You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			192 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			192 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //------------------------------------------------------------
 | |
| // Copyright (c) Microsoft Corporation.  All rights reserved.
 | |
| //------------------------------------------------------------
 | |
| 
 | |
| namespace System.ServiceModel.Channels
 | |
| {
 | |
|     using System;
 | |
|     using System.Runtime;
 | |
|     using System.ServiceModel;
 | |
|     using System.ServiceModel.Security;
 | |
| 
 | |
|     class PeerMessageDispatcher<ChannelInterfaceType, TChannel> : CommunicationObject
 | |
|         where ChannelInterfaceType : class, IChannel
 | |
|         where TChannel : InputQueueChannel<Message>
 | |
|     {
 | |
|         public class PeerMessageQueueAdapter
 | |
|         {
 | |
|             SingletonChannelAcceptor<ChannelInterfaceType, TChannel, Message> singletonChannelAcceptor;
 | |
|             InputQueueChannel<Message> inputQueueChannel;
 | |
| 
 | |
|             public PeerMessageQueueAdapter(SingletonChannelAcceptor<ChannelInterfaceType, TChannel, Message> singletonChannelAcceptor)
 | |
|             {
 | |
|                 this.singletonChannelAcceptor = singletonChannelAcceptor;
 | |
|             }
 | |
| 
 | |
|             public PeerMessageQueueAdapter(InputQueueChannel<Message> inputQueueChannel)
 | |
|             {
 | |
|                 this.inputQueueChannel = inputQueueChannel;
 | |
|             }
 | |
| 
 | |
|             public void EnqueueAndDispatch(Message message, Action callback)
 | |
|             {
 | |
|                 if (singletonChannelAcceptor != null)
 | |
|                 {
 | |
|                     singletonChannelAcceptor.Enqueue(message, callback);
 | |
|                 }
 | |
|                 else if (inputQueueChannel != null)
 | |
|                 {
 | |
|                     inputQueueChannel.EnqueueAndDispatch(message, callback);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         Uri via;
 | |
|         EndpointAddress to;
 | |
|         SecurityProtocol securityProtocol;
 | |
|         PeerNodeImplementation peerNode;
 | |
|         PeerMessageQueueAdapter queueHandler;
 | |
|         ChannelManagerBase channelManager;
 | |
|         PeerQuotaHelper quotaHelper = new PeerQuotaHelper(Int32.MaxValue);
 | |
|         bool registered;
 | |
| 
 | |
|         public PeerMessageDispatcher(PeerMessageQueueAdapter queueHandler, PeerNodeImplementation peerNode, ChannelManagerBase channelManager, EndpointAddress to, Uri via)
 | |
|         {
 | |
|             PeerNodeImplementation.ValidateVia(via);
 | |
| 
 | |
|             this.queueHandler = queueHandler;
 | |
|             this.peerNode = peerNode;
 | |
|             this.to = to;
 | |
|             this.via = via;
 | |
|             this.channelManager = channelManager;
 | |
|             EndpointAddress filterTo = null;
 | |
| 
 | |
|             this.securityProtocol = ((IPeerFactory)channelManager).SecurityManager.CreateSecurityProtocol<ChannelInterfaceType>(to, ServiceDefaults.SendTimeout);
 | |
| 
 | |
|             if (typeof(IDuplexChannel).IsAssignableFrom(typeof(ChannelInterfaceType)))
 | |
|                 filterTo = to;
 | |
| 
 | |
|             //Register this handler
 | |
|             PeerMessageFilter[] filters = new PeerMessageFilter[] { new PeerMessageFilter(via, filterTo) };
 | |
|             peerNode.RegisterMessageFilter(this, this.via, filters, (ITransportFactorySettings)this.channelManager,
 | |
|                                            new PeerNodeImplementation.MessageAvailableCallback(OnMessageAvailable), securityProtocol);
 | |
|             registered = true;
 | |
|         }
 | |
| 
 | |
|         protected override TimeSpan DefaultCloseTimeout
 | |
|         {
 | |
|             get { return channelManager.InternalCloseTimeout; }
 | |
|         }
 | |
| 
 | |
|         protected override TimeSpan DefaultOpenTimeout
 | |
|         {
 | |
|             get { return channelManager.InternalOpenTimeout; }
 | |
|         }
 | |
| 
 | |
|         public SecurityProtocol SecurityProtocol
 | |
|         {
 | |
|             get { return securityProtocol; }
 | |
|         }
 | |
| 
 | |
|         protected override void OnAbort()
 | |
|         {
 | |
|         }
 | |
| 
 | |
|         protected override IAsyncResult OnBeginClose(TimeSpan timeout, AsyncCallback callback, object state)
 | |
|         {
 | |
|             this.OnClose(timeout);
 | |
|             return new CompletedAsyncResult(callback, state);
 | |
|         }
 | |
| 
 | |
|         protected override void OnEndClose(IAsyncResult result)
 | |
|         {
 | |
|             CompletedAsyncResult.End(result);
 | |
|         }
 | |
| 
 | |
|         protected override void OnClose(TimeSpan timeout)
 | |
|         {
 | |
|             Unregister(true);
 | |
|         }
 | |
| 
 | |
|         internal void Unregister()
 | |
|         {
 | |
|             Unregister(false);
 | |
|         }
 | |
| 
 | |
|         internal void Unregister(bool release)
 | |
|         {
 | |
|             PeerNodeImplementation node = this.peerNode;
 | |
|             if (node != null)
 | |
|             {
 | |
|                 if (registered)
 | |
|                 {
 | |
|                     node.UnregisterMessageFilter(this, via);
 | |
|                     registered = false;
 | |
|                 }
 | |
|                 if (release)
 | |
|                     node.Release();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         protected override void OnOpen(TimeSpan timeout)
 | |
|         {
 | |
|         }
 | |
| 
 | |
|         protected override IAsyncResult OnBeginOpen(TimeSpan timeout, AsyncCallback callback, object state)
 | |
|         {
 | |
|             return new CompletedAsyncResult(callback, state);
 | |
|         }
 | |
| 
 | |
|         protected override void OnEndOpen(IAsyncResult result)
 | |
|         {
 | |
|             CompletedAsyncResult.End(result);
 | |
|         }
 | |
| 
 | |
|         public void OnMessageAvailable(Message message)
 | |
|         {
 | |
|             quotaHelper.ReadyToEnqueueItem();
 | |
|             queueHandler.EnqueueAndDispatch(message, quotaHelper.ItemDequeued);
 | |
|         }
 | |
|     }
 | |
|     class PeerMessageFilter
 | |
|     {
 | |
|         Uri via;
 | |
|         Uri actingAs;
 | |
| 
 | |
|         public PeerMessageFilter(Uri via) : this(via, null) { }
 | |
|         public PeerMessageFilter(Uri via, EndpointAddress to)
 | |
|         {
 | |
|             Fx.Assert(via != null, "PeerMessageFilter via can not be set to null");
 | |
|             this.via = via;
 | |
|             if (to != null)
 | |
|                 this.actingAs = to.Uri;
 | |
|         }
 | |
| 
 | |
|         public bool Match(Uri peerVia, Uri toCond)
 | |
|         {
 | |
|             bool result = false;
 | |
|             if (peerVia == null)
 | |
|             {
 | |
|                 result = false;
 | |
|             }
 | |
|             else if (Uri.Compare(this.via, peerVia,
 | |
|                (UriComponents.Scheme | UriComponents.UserInfo | UriComponents.Host | UriComponents.Port | UriComponents.Path),
 | |
|                UriFormat.SafeUnescaped, StringComparison.OrdinalIgnoreCase) != 0)
 | |
|             {
 | |
|                 result = false;
 | |
|             }
 | |
|             else if (this.actingAs != null)
 | |
|             {
 | |
|                 result = Uri.Compare(this.actingAs, toCond,
 | |
|                (UriComponents.Scheme | UriComponents.UserInfo | UriComponents.Host | UriComponents.Port | UriComponents.Path),
 | |
|                UriFormat.SafeUnescaped, StringComparison.OrdinalIgnoreCase) == 0;
 | |
|             }
 | |
|             else
 | |
|                 result = true;
 | |
| 
 | |
|             return result;
 | |
|         }
 | |
| 
 | |
|     }
 | |
| }
 |