You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			227 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			227 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //----------------------------------------------------------------------------
 | |
| // Copyright (c) Microsoft Corporation.  All rights reserved.
 | |
| //----------------------------------------------------------------------------
 | |
| 
 | |
| namespace System.ServiceModel.Channels
 | |
| {
 | |
|     using System.Collections.Generic;
 | |
|     using System.Runtime;
 | |
|     using System.ServiceModel;
 | |
|     using System.ServiceModel.Activation;
 | |
|     using System.Threading;
 | |
|     using System.Web;
 | |
|     using System.Web.Hosting;
 | |
| 
 | |
|     class MsmqHostedTransportManager : TransportManager
 | |
|     {
 | |
|         string[] hosts;
 | |
|         List<MsmqBindingMonitor> bindingMonitors;
 | |
|         HostedBindingFilter filter;
 | |
|         MsmqUri.IAddressTranslator addressing;
 | |
|         Action messageReceivedCallback;
 | |
| 
 | |
|         public MsmqHostedTransportManager(string[] hosts, MsmqUri.IAddressTranslator addressing)
 | |
|         {
 | |
|             AspNetPartialTrustHelpers.FailIfInPartialTrustOutsideAspNet();
 | |
| 
 | |
|             this.hosts = hosts;
 | |
|             this.bindingMonitors = new List<MsmqBindingMonitor>();
 | |
|             this.addressing = addressing;
 | |
|             this.filter = new HostedBindingFilter(HostingEnvironment.ApplicationVirtualPath, addressing);
 | |
| 
 | |
|             foreach (string host in this.hosts)
 | |
|             {
 | |
|                 MsmqBindingMonitor monitor = new MsmqBindingMonitor(host, TimeSpan.FromMinutes(5), true);
 | |
|                 monitor.AddFilter(this.filter);
 | |
|                 monitor.Open();
 | |
|                 this.bindingMonitors.Add(monitor);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public Uri[] GetBaseAddresses(string virtualPath)
 | |
|         {
 | |
|             // Make sure this is not called until initialization is done:
 | |
|             foreach (MsmqBindingMonitor monitor in this.bindingMonitors)
 | |
|             {
 | |
|                 monitor.WaitForFirstRoundComplete();
 | |
|             }
 | |
| 
 | |
|             string absoluteVirtualPath = VirtualPathUtility.ToAbsolute(virtualPath, HostingEnvironment.ApplicationVirtualPath);
 | |
| 
 | |
|             List<Uri> baseAddresses = new List<Uri>(this.hosts.Length);
 | |
|             string queueName = absoluteVirtualPath.Substring(1);
 | |
| 
 | |
|             foreach (string host in this.hosts)
 | |
|             {
 | |
|                 bool isPrivate = this.filter.IsPrivateMatch(queueName);
 | |
| 
 | |
|                 Uri uri = this.addressing.CreateUri(host, queueName, isPrivate);
 | |
|                 baseAddresses.Add(uri);
 | |
|                 MsmqDiagnostics.FoundBaseAddress(uri, absoluteVirtualPath);
 | |
|             }
 | |
|             return baseAddresses.ToArray();
 | |
|         }
 | |
| 
 | |
|         internal override string Scheme
 | |
|         {
 | |
|             get { return this.addressing.Scheme; }
 | |
|         }
 | |
| 
 | |
|         internal override void OnClose(TimeSpan timeout)
 | |
|         {
 | |
|             foreach (MsmqBindingMonitor monitor in this.bindingMonitors)
 | |
|             {
 | |
|                 monitor.Close();
 | |
|             }
 | |
| 
 | |
|             this.bindingMonitors.Clear();
 | |
|         }
 | |
| 
 | |
|         internal override void OnOpen()
 | |
|         {
 | |
|             // Nothing to do - we only use the transport manager for WebHosted case.
 | |
|         }
 | |
| 
 | |
|         internal override void Register(TransportChannelListener channelListener)
 | |
|         {
 | |
|             channelListener.SetMessageReceivedCallback(new Action(OnMessageReceived));
 | |
|         }
 | |
| 
 | |
|         internal void Start(Action messageReceivedCallback)
 | |
|         {
 | |
|             this.messageReceivedCallback = messageReceivedCallback;
 | |
|         }
 | |
| 
 | |
|         internal override void Unregister(TransportChannelListener channelListener)
 | |
|         {
 | |
|             // Nothing to do - we never use the transport manager during normal
 | |
|             // operation.
 | |
|         }
 | |
| 
 | |
|         void OnMessageReceived()
 | |
|         {
 | |
|             Action callback = this.messageReceivedCallback;
 | |
|             if (callback != null)
 | |
|             {
 | |
|                 callback();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         class HostedBindingFilter : MsmqBindingFilter
 | |
|         {
 | |
|             Dictionary<string, string> privateMatches = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);
 | |
|             object thisLock = new object();
 | |
| 
 | |
|             public HostedBindingFilter(string path, MsmqUri.IAddressTranslator addressing)
 | |
|                 : base(path, addressing)
 | |
|             {
 | |
|                 AspNetPartialTrustHelpers.FailIfInPartialTrustOutsideAspNet();
 | |
|             }
 | |
| 
 | |
|             public override object MatchFound(string host, string name, bool isPrivate)
 | |
|             {
 | |
|                 string processedVirtualPath = CreateRelativeVirtualPath(host, name, isPrivate);
 | |
|                 string relativeServiceFile = ServiceHostingEnvironment.NormalizeVirtualPath(processedVirtualPath);
 | |
| 
 | |
|                 // Compute the remainder path:
 | |
|                 if (isPrivate)
 | |
|                 {
 | |
|                     lock (this.thisLock)
 | |
|                     {
 | |
|                         string baseQueue = CreateBaseQueue(relativeServiceFile);
 | |
|                         this.privateMatches[baseQueue] = baseQueue;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 // Start the service on a different thread so we can complete 
 | |
|                 // initialization
 | |
|                 if (CheckServiceExists(relativeServiceFile))
 | |
|                 {
 | |
|                     MsmqDiagnostics.StartingService(host, name, isPrivate, processedVirtualPath);
 | |
|                     ActionItem.Schedule(StartService, processedVirtualPath);
 | |
|                 }
 | |
| 
 | |
|                 // no callback state here...
 | |
|                 return null;
 | |
|             }
 | |
| 
 | |
|             public bool IsPrivateMatch(string processedVirtualPath)
 | |
|             {
 | |
|                 lock (this.thisLock)
 | |
|                 {
 | |
|                     return this.privateMatches.ContainsKey(processedVirtualPath);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             public override void MatchLost(string host, string name, bool isPrivate, object callbackState)
 | |
|             {
 | |
|                 // We don't do anything here - the service will stay alive,
 | |
|                 // and if the queue ever comes back, then it will begin to
 | |
|                 // process again.
 | |
|             }
 | |
| 
 | |
|             string CreateRelativeVirtualPath(string host, string name, bool isPrivate)
 | |
|             {
 | |
|                 // the canonical prefix looks something like: "invoices/"
 | |
|                 // Because the queue name matched, it looks like "invoices/..."
 | |
|                 // remove the common piece, and prefix with the "~/" home specifier
 | |
|                 return "~/" + name.Substring(CanonicalPrefix.Length);
 | |
|             }
 | |
| 
 | |
|             string CreateBaseQueue(string serviceFile)
 | |
|             {
 | |
|                 // Clean up the service file...
 | |
|                 if (serviceFile.StartsWith("~", StringComparison.OrdinalIgnoreCase))
 | |
|                     serviceFile = serviceFile.Substring(1);
 | |
|                 if (serviceFile.StartsWith("/", StringComparison.OrdinalIgnoreCase))
 | |
|                     serviceFile = serviceFile.Substring(1);
 | |
| 
 | |
|                 string virtualPath = HostingEnvironment.ApplicationVirtualPath;
 | |
|                 if (virtualPath.EndsWith("/", StringComparison.OrdinalIgnoreCase))
 | |
|                     virtualPath = virtualPath.Substring(0, virtualPath.Length - 1);
 | |
|                 if (virtualPath.StartsWith("/", StringComparison.OrdinalIgnoreCase))
 | |
|                     virtualPath = virtualPath.Substring(1);
 | |
| 
 | |
|                 return virtualPath + "/" + serviceFile;
 | |
|             }
 | |
| 
 | |
|             bool CheckServiceExists(string serviceFile)
 | |
|             {
 | |
|                 try
 | |
|                 {
 | |
|                     return ((ServiceHostingEnvironment.IsConfigurationBasedService(serviceFile) 
 | |
|                              || HostingEnvironmentWrapper.ServiceFileExists(serviceFile)) 
 | |
|                              && AspNetEnvironment.Current.IsWithinApp(VirtualPathUtility.ToAbsolute(serviceFile)));
 | |
|                 }
 | |
|                 catch (ArgumentException ex)
 | |
|                 {
 | |
|                     MsmqDiagnostics.ExpectedException(ex);
 | |
|                     return false;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             void StartService(object state)
 | |
|             {
 | |
|                 try
 | |
|                 {
 | |
|                     string processedVirtualPath = (string)state;
 | |
|                     ServiceHostingEnvironment.EnsureServiceAvailable(processedVirtualPath);
 | |
|                 }
 | |
|                 catch (ServiceActivationException e)
 | |
|                 {
 | |
|                     // Non-fatal exceptions from the user code are wrapped in ServiceActivationException
 | |
|                     // The best we can do is to trace them
 | |
|                     MsmqDiagnostics.ExpectedException(e);
 | |
|                 }
 | |
|                 catch (EndpointNotFoundException e)
 | |
|                 {
 | |
|                     // This means that the server disappeared between the time we
 | |
|                     // saw the service, and the time we tried to start it.
 | |
|                     // That's okay.
 | |
|                     MsmqDiagnostics.ExpectedException(e);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 |