You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			1090 lines
		
	
	
		
			38 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			1090 lines
		
	
	
		
			38 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //------------------------------------------------------------
 | |
| // Copyright (c) Microsoft Corporation.  All rights reserved.
 | |
| //------------------------------------------------------------
 | |
| 
 | |
| namespace System.ServiceModel.Channels
 | |
| {
 | |
|     using System;
 | |
|     using System.Diagnostics;
 | |
|     using System.Runtime;
 | |
|     using System.ServiceModel;
 | |
|     using System.ServiceModel.Dispatcher;
 | |
|     using System.ServiceModel.Security;
 | |
| 
 | |
|     abstract class ServerReliableChannelBinder<TChannel> : ReliableChannelBinder<TChannel>,
 | |
|         IServerReliableChannelBinder
 | |
|         where TChannel : class, IChannel
 | |
|     {
 | |
|         static string addressedPropertyName = "MessageAddressedByBinderProperty";
 | |
|         IChannelListener<TChannel> listener;
 | |
|         static AsyncCallback onAcceptChannelComplete = Fx.ThunkCallback(new AsyncCallback(OnAcceptChannelCompleteStatic));
 | |
|         EndpointAddress cachedLocalAddress;
 | |
|         TChannel pendingChannel;
 | |
|         InterruptibleWaitObject pendingChannelEvent = new InterruptibleWaitObject(false, false);
 | |
|         EndpointAddress remoteAddress;
 | |
| 
 | |
|         protected ServerReliableChannelBinder(ChannelBuilder builder,
 | |
|             EndpointAddress remoteAddress, MessageFilter filter, int priority, MaskingMode maskingMode,
 | |
|             TolerateFaultsMode faultMode, TimeSpan defaultCloseTimeout,
 | |
|             TimeSpan defaultSendTimeout)
 | |
|             : base(null, maskingMode, faultMode, defaultCloseTimeout, defaultSendTimeout)
 | |
|         {
 | |
|             this.listener = builder.BuildChannelListener<TChannel>(filter, priority);
 | |
|             this.remoteAddress = remoteAddress;
 | |
|         }
 | |
| 
 | |
|         protected ServerReliableChannelBinder(TChannel channel, EndpointAddress cachedLocalAddress,
 | |
|             EndpointAddress remoteAddress, MaskingMode maskingMode,
 | |
|             TolerateFaultsMode faultMode, TimeSpan defaultCloseTimeout,
 | |
|             TimeSpan defaultSendTimeout)
 | |
|             : base(channel, maskingMode, faultMode, defaultCloseTimeout, defaultSendTimeout)
 | |
|         {
 | |
|             this.cachedLocalAddress = cachedLocalAddress;
 | |
|             this.remoteAddress = remoteAddress;
 | |
|         }
 | |
| 
 | |
|         protected override bool CanGetChannelForReceive
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 return true;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public override EndpointAddress LocalAddress
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 if (this.cachedLocalAddress != null)
 | |
|                 {
 | |
|                     return this.cachedLocalAddress;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     return this.GetInnerChannelLocalAddress();
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         protected override bool MustCloseChannel
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 return this.MustOpenChannel || this.HasSession;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         protected override bool MustOpenChannel
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 return this.listener != null;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public override EndpointAddress RemoteAddress
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 return this.remoteAddress;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         void AddAddressedProperty(Message message)
 | |
|         {
 | |
|             message.Properties.Add(addressedPropertyName, new object());
 | |
|         }
 | |
| 
 | |
|         protected override void AddOutputHeaders(Message message)
 | |
|         {
 | |
|             if (this.GetAddressedProperty(message) == null)
 | |
|             {
 | |
|                 this.RemoteAddress.ApplyTo(message);
 | |
|                 this.AddAddressedProperty(message);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public bool AddressResponse(Message request, Message response)
 | |
|         {
 | |
|             if (this.GetAddressedProperty(response) != null)
 | |
|             {
 | |
|                 throw Fx.AssertAndThrow("The binder can't address a response twice");
 | |
|             }
 | |
| 
 | |
|             try
 | |
|             {
 | |
|                 RequestReplyCorrelator.PrepareReply(response, request);
 | |
|             }
 | |
|             catch (MessageHeaderException exception)
 | |
|             {
 | |
|                 // ---- it - we don't need to correlate the reply if the MessageId header is bad
 | |
|                 if (DiagnosticUtility.ShouldTraceInformation)
 | |
|                     DiagnosticUtility.TraceHandledException(exception, TraceEventType.Information);
 | |
|             }
 | |
| 
 | |
|             bool sendResponse = true;
 | |
|             try
 | |
|             {
 | |
|                 sendResponse = RequestReplyCorrelator.AddressReply(response, request);
 | |
|             }
 | |
|             catch (MessageHeaderException exception)
 | |
|             {
 | |
|                 // ---- it - we don't need to address the reply if the addressing headers are bad
 | |
|                 if (DiagnosticUtility.ShouldTraceInformation)
 | |
|                     DiagnosticUtility.TraceHandledException(exception, TraceEventType.Information);
 | |
|             }
 | |
| 
 | |
|             if (sendResponse)
 | |
|                 this.AddAddressedProperty(response);
 | |
| 
 | |
|             return sendResponse;
 | |
|         }
 | |
| 
 | |
|         protected override IAsyncResult BeginTryGetChannel(TimeSpan timeout,
 | |
|             AsyncCallback callback, object state)
 | |
|         {
 | |
|             return this.pendingChannelEvent.BeginTryWait(timeout, callback, state);
 | |
|         }
 | |
| 
 | |
|         public IAsyncResult BeginWaitForRequest(TimeSpan timeout, AsyncCallback callback,
 | |
|             object state)
 | |
|         {
 | |
|             if (this.DefaultMaskingMode != MaskingMode.None)
 | |
|             {
 | |
|                 throw Fx.AssertAndThrow("This method was implemented only for the case where we do not mask exceptions.");
 | |
|             }
 | |
| 
 | |
|             if (this.ValidateInputOperation(timeout))
 | |
|             {
 | |
|                 return new WaitForRequestAsyncResult(this, timeout, callback, state);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 return new CompletedAsyncResult(callback, state);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         bool CompleteAcceptChannel(IAsyncResult result)
 | |
|         {
 | |
|             TChannel channel = this.listener.EndAcceptChannel(result);
 | |
| 
 | |
|             if (channel == null)
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             if (!this.UseNewChannel(channel))
 | |
|             {
 | |
|                 channel.Abort();
 | |
|             }
 | |
| 
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         public static IServerReliableChannelBinder CreateBinder(ChannelBuilder builder,
 | |
|             EndpointAddress remoteAddress, MessageFilter filter, int priority,
 | |
|             TolerateFaultsMode faultMode, TimeSpan defaultCloseTimeout,
 | |
|             TimeSpan defaultSendTimeout)
 | |
|         {
 | |
|             Type type = typeof(TChannel);
 | |
| 
 | |
|             if (type == typeof(IDuplexChannel))
 | |
|             {
 | |
|                 return new DuplexServerReliableChannelBinder(builder, remoteAddress, filter,
 | |
|                     priority, MaskingMode.None, defaultCloseTimeout, defaultSendTimeout);
 | |
|             }
 | |
|             else if (type == typeof(IDuplexSessionChannel))
 | |
|             {
 | |
|                 return new DuplexSessionServerReliableChannelBinder(builder, remoteAddress, filter,
 | |
|                     priority, MaskingMode.None, faultMode, defaultCloseTimeout,
 | |
|                     defaultSendTimeout);
 | |
|             }
 | |
|             else if (type == typeof(IReplyChannel))
 | |
|             {
 | |
|                 return new ReplyServerReliableChannelBinder(builder, remoteAddress, filter,
 | |
|                     priority, MaskingMode.None, defaultCloseTimeout, defaultSendTimeout);
 | |
|             }
 | |
|             else if (type == typeof(IReplySessionChannel))
 | |
|             {
 | |
|                 return new ReplySessionServerReliableChannelBinder(builder, remoteAddress, filter,
 | |
|                     priority, MaskingMode.None, faultMode, defaultCloseTimeout,
 | |
|                     defaultSendTimeout);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 throw Fx.AssertAndThrow("ServerReliableChannelBinder supports creation of IDuplexChannel, IDuplexSessionChannel, IReplyChannel, and IReplySessionChannel only.");
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public static IServerReliableChannelBinder CreateBinder(TChannel channel,
 | |
|             EndpointAddress cachedLocalAddress, EndpointAddress remoteAddress,
 | |
|             TolerateFaultsMode faultMode, TimeSpan defaultCloseTimeout,
 | |
|             TimeSpan defaultSendTimeout)
 | |
|         {
 | |
|             Type type = typeof(TChannel);
 | |
| 
 | |
|             if (type == typeof(IDuplexChannel))
 | |
|             {
 | |
|                 return new DuplexServerReliableChannelBinder((IDuplexChannel)channel,
 | |
|                     cachedLocalAddress, remoteAddress, MaskingMode.All, defaultCloseTimeout,
 | |
|                     defaultSendTimeout);
 | |
|             }
 | |
|             else if (type == typeof(IDuplexSessionChannel))
 | |
|             {
 | |
|                 return new DuplexSessionServerReliableChannelBinder((IDuplexSessionChannel)channel,
 | |
|                     cachedLocalAddress, remoteAddress, MaskingMode.All, faultMode,
 | |
|                     defaultCloseTimeout, defaultSendTimeout);
 | |
|             }
 | |
|             else if (type == typeof(IReplyChannel))
 | |
|             {
 | |
|                 return new ReplyServerReliableChannelBinder((IReplyChannel)channel,
 | |
|                     cachedLocalAddress, remoteAddress, MaskingMode.All, defaultCloseTimeout,
 | |
|                     defaultSendTimeout);
 | |
|             }
 | |
|             else if (type == typeof(IReplySessionChannel))
 | |
|             {
 | |
|                 return new ReplySessionServerReliableChannelBinder((IReplySessionChannel)channel,
 | |
|                     cachedLocalAddress, remoteAddress, MaskingMode.All, faultMode,
 | |
|                     defaultCloseTimeout, defaultSendTimeout);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 throw Fx.AssertAndThrow("ServerReliableChannelBinder supports creation of IDuplexChannel, IDuplexSessionChannel, IReplyChannel, and IReplySessionChannel only.");
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         protected override bool EndTryGetChannel(IAsyncResult result)
 | |
|         {
 | |
|             if (!this.pendingChannelEvent.EndTryWait(result))
 | |
|                 return false;
 | |
| 
 | |
|             TChannel abortChannel = null;
 | |
| 
 | |
|             lock (this.ThisLock)
 | |
|             {
 | |
|                 if (this.State != CommunicationState.Faulted &&
 | |
|                     this.State != CommunicationState.Closing &&
 | |
|                     this.State != CommunicationState.Closed)
 | |
|                 {
 | |
|                     if (!this.Synchronizer.SetChannel(this.pendingChannel))
 | |
|                     {
 | |
|                         abortChannel = this.pendingChannel;
 | |
|                     }
 | |
| 
 | |
|                     this.pendingChannel = null;
 | |
|                     this.pendingChannelEvent.Reset();
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (abortChannel != null)
 | |
|             {
 | |
|                 abortChannel.Abort();
 | |
|             }
 | |
| 
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         public bool EndWaitForRequest(IAsyncResult result)
 | |
|         {
 | |
|             WaitForRequestAsyncResult waitForRequestResult = result as WaitForRequestAsyncResult;
 | |
| 
 | |
|             if (waitForRequestResult != null)
 | |
|             {
 | |
|                 return waitForRequestResult.End();
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 CompletedAsyncResult.End(result);
 | |
|                 return true;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         object GetAddressedProperty(Message message)
 | |
|         {
 | |
|             object property;
 | |
| 
 | |
|             message.Properties.TryGetValue(addressedPropertyName, out property);
 | |
|             return property;
 | |
|         }
 | |
| 
 | |
|         protected abstract EndpointAddress GetInnerChannelLocalAddress();
 | |
| 
 | |
|         bool IsListenerExceptionNullOrHandleable(Exception e)
 | |
|         {
 | |
|             if (e == null)
 | |
|             {
 | |
|                 return true;
 | |
|             }
 | |
| 
 | |
|             if (this.listener.State == CommunicationState.Faulted)
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             return this.IsHandleable(e);
 | |
|         }
 | |
| 
 | |
|         protected override void OnAbort()
 | |
|         {
 | |
|             if (this.listener != null)
 | |
|             {
 | |
|                 this.listener.Abort();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         void OnAcceptChannelComplete(IAsyncResult result)
 | |
|         {
 | |
|             Exception expectedException = null;
 | |
|             Exception unexpectedException = null;
 | |
|             bool gotChannel = false;
 | |
| 
 | |
|             try
 | |
|             {
 | |
|                 gotChannel = this.CompleteAcceptChannel(result);
 | |
|             }
 | |
|             catch (Exception e)
 | |
|             {
 | |
|                 if (Fx.IsFatal(e))
 | |
|                 {
 | |
|                     throw;
 | |
|                 }
 | |
| 
 | |
|                 if (this.IsHandleable(e))
 | |
|                 {
 | |
|                     expectedException = e;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     unexpectedException = e;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (gotChannel)
 | |
|             {
 | |
|                 this.StartAccepting();
 | |
|             }
 | |
|             else if (unexpectedException != null)
 | |
|             {
 | |
|                 this.Fault(unexpectedException);
 | |
|             }
 | |
|             else if ((expectedException != null)
 | |
|                 && (this.listener.State == CommunicationState.Opened))
 | |
|             {
 | |
|                 this.StartAccepting();
 | |
|             }
 | |
|             else if (this.listener.State == CommunicationState.Faulted)
 | |
|             {
 | |
|                 this.Fault(expectedException);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         static void OnAcceptChannelCompleteStatic(IAsyncResult result)
 | |
|         {
 | |
|             if (!result.CompletedSynchronously)
 | |
|             {
 | |
|                 ServerReliableChannelBinder<TChannel> binder =
 | |
|                     (ServerReliableChannelBinder<TChannel>)result.AsyncState;
 | |
| 
 | |
|                 binder.OnAcceptChannelComplete(result);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         protected override IAsyncResult OnBeginClose(TimeSpan timeout, AsyncCallback callback,
 | |
|             object state)
 | |
|         {
 | |
|             if (this.listener != null)
 | |
|             {
 | |
|                 return this.listener.BeginClose(timeout, callback, state);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 return new CompletedAsyncResult(callback, state);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         protected override IAsyncResult OnBeginOpen(TimeSpan timeout, AsyncCallback callback,
 | |
|             object state)
 | |
|         {
 | |
|             if (this.listener != null)
 | |
|             {
 | |
|                 return this.listener.BeginOpen(timeout, callback, state);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 return new CompletedAsyncResult(callback, state);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         protected abstract IAsyncResult OnBeginWaitForRequest(TChannel channel, TimeSpan timeout,
 | |
|             AsyncCallback callback, object state);
 | |
| 
 | |
|         protected override void OnClose(TimeSpan timeout)
 | |
|         {
 | |
|             if (this.listener != null)
 | |
|             {
 | |
|                 this.listener.Close(timeout);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         protected override void OnShutdown()
 | |
|         {
 | |
|             TChannel channel = null;
 | |
| 
 | |
|             lock (this.ThisLock)
 | |
|             {
 | |
|                 channel = this.pendingChannel;
 | |
|                 this.pendingChannel = null;
 | |
|                 this.pendingChannelEvent.Set();
 | |
|             }
 | |
| 
 | |
|             if (channel != null)
 | |
|                 channel.Abort();
 | |
|         }
 | |
| 
 | |
|         protected abstract bool OnWaitForRequest(TChannel channel, TimeSpan timeout);
 | |
| 
 | |
|         protected override void OnEndClose(IAsyncResult result)
 | |
|         {
 | |
|             if (this.listener != null)
 | |
|             {
 | |
|                 this.listener.EndClose(result);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 CompletedAsyncResult.End(result);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         protected override void OnEndOpen(IAsyncResult result)
 | |
|         {
 | |
|             if (this.listener != null)
 | |
|             {
 | |
|                 this.listener.EndOpen(result);
 | |
|                 this.StartAccepting();
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 CompletedAsyncResult.End(result);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         protected abstract bool OnEndWaitForRequest(TChannel channel, IAsyncResult result);
 | |
| 
 | |
|         protected override void OnOpen(TimeSpan timeout)
 | |
|         {
 | |
|             if (this.listener != null)
 | |
|             {
 | |
|                 this.listener.Open(timeout);
 | |
|                 this.StartAccepting();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         void StartAccepting()
 | |
|         {
 | |
|             Exception expectedException = null;
 | |
|             Exception unexpectedException = null;
 | |
| 
 | |
|             while (this.listener.State == CommunicationState.Opened)
 | |
|             {
 | |
|                 expectedException = null;
 | |
|                 unexpectedException = null;
 | |
| 
 | |
|                 try
 | |
|                 {
 | |
|                     IAsyncResult result = this.listener.BeginAcceptChannel(TimeSpan.MaxValue,
 | |
|                         onAcceptChannelComplete, this);
 | |
| 
 | |
|                     if (!result.CompletedSynchronously)
 | |
|                     {
 | |
|                         return;
 | |
|                     }
 | |
|                     else if (!this.CompleteAcceptChannel(result))
 | |
|                     {
 | |
|                         break;
 | |
|                     }
 | |
|                 }
 | |
|                 catch (Exception e)
 | |
|                 {
 | |
|                     if (Fx.IsFatal(e))
 | |
|                     {
 | |
|                         throw;
 | |
|                     }
 | |
| 
 | |
|                     if (this.IsHandleable(e))
 | |
|                     {
 | |
|                         expectedException = e;
 | |
|                         continue;
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         unexpectedException = e;
 | |
|                         break;
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (unexpectedException != null)
 | |
|             {
 | |
|                 this.Fault(unexpectedException);
 | |
|             }
 | |
|             else if (this.listener.State == CommunicationState.Faulted)
 | |
|             {
 | |
|                 this.Fault(expectedException);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         protected override bool TryGetChannel(TimeSpan timeout)
 | |
|         {
 | |
|             if (!this.pendingChannelEvent.Wait(timeout))
 | |
|                 return false;
 | |
| 
 | |
|             TChannel abortChannel = null;
 | |
| 
 | |
|             lock (this.ThisLock)
 | |
|             {
 | |
|                 if (this.State != CommunicationState.Faulted &&
 | |
|                     this.State != CommunicationState.Closing &&
 | |
|                     this.State != CommunicationState.Closed)
 | |
|                 {
 | |
|                     if (!this.Synchronizer.SetChannel(this.pendingChannel))
 | |
|                     {
 | |
|                         abortChannel = this.pendingChannel;
 | |
|                     }
 | |
| 
 | |
|                     this.pendingChannel = null;
 | |
|                     this.pendingChannelEvent.Reset();
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (abortChannel != null)
 | |
|             {
 | |
|                 abortChannel.Abort();
 | |
|             }
 | |
| 
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         public bool UseNewChannel(IChannel channel)
 | |
|         {
 | |
|             TChannel oldPendingChannel = null;
 | |
|             TChannel oldBinderChannel = null;
 | |
| 
 | |
|             lock (this.ThisLock)
 | |
|             {
 | |
|                 if (!this.Synchronizer.TolerateFaults ||
 | |
|                     this.State == CommunicationState.Faulted ||
 | |
|                     this.State == CommunicationState.Closing ||
 | |
|                     this.State == CommunicationState.Closed)
 | |
|                 {
 | |
|                     return false;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     oldPendingChannel = this.pendingChannel;
 | |
|                     this.pendingChannel = (TChannel)channel;
 | |
|                     oldBinderChannel = this.Synchronizer.AbortCurentChannel();
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (oldPendingChannel != null)
 | |
|             {
 | |
|                 oldPendingChannel.Abort();
 | |
|             }
 | |
| 
 | |
|             this.pendingChannelEvent.Set();
 | |
| 
 | |
|             if (oldBinderChannel != null)
 | |
|             {
 | |
|                 oldBinderChannel.Abort();
 | |
|             }
 | |
| 
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         public bool WaitForRequest(TimeSpan timeout)
 | |
|         {
 | |
|             if (this.DefaultMaskingMode != MaskingMode.None)
 | |
|             {
 | |
|                 throw Fx.AssertAndThrow("This method was implemented only for the case where we do not mask exceptions.");
 | |
|             }
 | |
| 
 | |
|             if (!this.ValidateInputOperation(timeout))
 | |
|             {
 | |
|                 return true;
 | |
|             }
 | |
| 
 | |
|             TimeoutHelper timeoutHelper = new TimeoutHelper(timeout);
 | |
| 
 | |
|             while (true)
 | |
|             {
 | |
|                 bool autoAborted = false;
 | |
| 
 | |
|                 try
 | |
|                 {
 | |
|                     TChannel channel;
 | |
|                     bool success = !this.Synchronizer.TryGetChannelForInput(true, timeoutHelper.RemainingTime(),
 | |
|                         out channel);
 | |
| 
 | |
|                     if (channel == null)
 | |
|                     {
 | |
|                         return success;
 | |
|                     }
 | |
| 
 | |
|                     try
 | |
|                     {
 | |
|                         return this.OnWaitForRequest(channel, timeoutHelper.RemainingTime());
 | |
|                     }
 | |
|                     finally
 | |
|                     {
 | |
|                         autoAborted = this.Synchronizer.Aborting;
 | |
|                         this.Synchronizer.ReturnChannel();
 | |
|                     }
 | |
|                 }
 | |
|                 catch (Exception e)
 | |
|                 {
 | |
|                     if (Fx.IsFatal(e))
 | |
|                         throw;
 | |
| 
 | |
|                     if (!this.HandleException(e, this.DefaultMaskingMode, autoAborted))
 | |
|                     {
 | |
|                         throw;
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         continue;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         abstract class DuplexServerReliableChannelBinder<TDuplexChannel>
 | |
|             : ServerReliableChannelBinder<TDuplexChannel>
 | |
|             where TDuplexChannel : class, IDuplexChannel
 | |
|         {
 | |
|             protected DuplexServerReliableChannelBinder(ChannelBuilder builder,
 | |
|                 EndpointAddress remoteAddress, MessageFilter filter, int priority,
 | |
|                 MaskingMode maskingMode, TolerateFaultsMode faultMode,
 | |
|                 TimeSpan defaultCloseTimeout, TimeSpan defaultSendTimeout)
 | |
|                 : base(builder, remoteAddress, filter, priority, maskingMode, faultMode,
 | |
|                 defaultCloseTimeout, defaultSendTimeout)
 | |
|             {
 | |
|             }
 | |
| 
 | |
|             protected DuplexServerReliableChannelBinder(TDuplexChannel channel,
 | |
|                 EndpointAddress cachedLocalAddress, EndpointAddress remoteAddress,
 | |
|                 MaskingMode maskingMode, TolerateFaultsMode faultMode,
 | |
|                 TimeSpan defaultCloseTimeout, TimeSpan defaultSendTimeout)
 | |
|                 : base(channel, cachedLocalAddress, remoteAddress, maskingMode, faultMode,
 | |
|                 defaultCloseTimeout, defaultSendTimeout)
 | |
|             {
 | |
|             }
 | |
| 
 | |
|             public override bool CanSendAsynchronously
 | |
|             {
 | |
|                 get
 | |
|                 {
 | |
|                     return true;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             protected override EndpointAddress GetInnerChannelLocalAddress()
 | |
|             {
 | |
|                 IDuplexChannel channel = this.Synchronizer.CurrentChannel;
 | |
|                 EndpointAddress localAddress = (channel == null) ? null : channel.LocalAddress;
 | |
|                 return localAddress;
 | |
|             }
 | |
| 
 | |
|             protected override IAsyncResult OnBeginSend(TDuplexChannel channel, Message message,
 | |
|                 TimeSpan timeout, AsyncCallback callback, object state)
 | |
|             {
 | |
|                 return channel.BeginSend(message, timeout, callback, state);
 | |
|             }
 | |
| 
 | |
|             protected override IAsyncResult OnBeginTryReceive(TDuplexChannel channel,
 | |
|                 TimeSpan timeout, AsyncCallback callback, object state)
 | |
|             {
 | |
|                 return channel.BeginTryReceive(timeout, callback, state);
 | |
|             }
 | |
| 
 | |
|             protected override IAsyncResult OnBeginWaitForRequest(TDuplexChannel channel,
 | |
|                 TimeSpan timeout, AsyncCallback callback, object state)
 | |
|             {
 | |
|                 return channel.BeginWaitForMessage(timeout, callback, state);
 | |
|             }
 | |
| 
 | |
|             protected override void OnEndSend(TDuplexChannel channel, IAsyncResult result)
 | |
|             {
 | |
|                 channel.EndSend(result);
 | |
|             }
 | |
| 
 | |
|             protected override bool OnEndTryReceive(TDuplexChannel channel, IAsyncResult result,
 | |
|                 out RequestContext requestContext)
 | |
|             {
 | |
|                 Message message;
 | |
|                 bool success = channel.EndTryReceive(result, out message);
 | |
|                 if (success)
 | |
|                 {
 | |
|                     this.OnMessageReceived(message);
 | |
|                 }
 | |
|                 requestContext = this.WrapMessage(message);
 | |
|                 return success;
 | |
|             }
 | |
| 
 | |
|             protected override bool OnEndWaitForRequest(TDuplexChannel channel,
 | |
|                 IAsyncResult result)
 | |
|             {
 | |
|                 return channel.EndWaitForMessage(result);
 | |
|             }
 | |
| 
 | |
|             protected abstract void OnMessageReceived(Message message);
 | |
| 
 | |
|             protected override void OnSend(TDuplexChannel channel, Message message,
 | |
|                 TimeSpan timeout)
 | |
|             {
 | |
|                 channel.Send(message, timeout);
 | |
|             }
 | |
| 
 | |
|             protected override bool OnTryReceive(TDuplexChannel channel, TimeSpan timeout,
 | |
|                 out RequestContext requestContext)
 | |
|             {
 | |
|                 Message message;
 | |
|                 bool success = channel.TryReceive(timeout, out message);
 | |
|                 if (success)
 | |
|                 {
 | |
|                     this.OnMessageReceived(message);
 | |
|                 }
 | |
|                 requestContext = this.WrapMessage(message);
 | |
|                 return success;
 | |
|             }
 | |
| 
 | |
|             protected override bool OnWaitForRequest(TDuplexChannel channel, TimeSpan timeout)
 | |
|             {
 | |
|                 return channel.WaitForMessage(timeout);
 | |
|             }
 | |
| 
 | |
|         }
 | |
| 
 | |
|         sealed class DuplexServerReliableChannelBinder
 | |
|             : DuplexServerReliableChannelBinder<IDuplexChannel>
 | |
|         {
 | |
|             public DuplexServerReliableChannelBinder(ChannelBuilder builder,
 | |
|                 EndpointAddress remoteAddress, MessageFilter filter, int priority,
 | |
|                 MaskingMode maskingMode, TimeSpan defaultCloseTimeout, TimeSpan defaultSendTimeout)
 | |
|                 : base(builder, remoteAddress, filter, priority, maskingMode,
 | |
|                 TolerateFaultsMode.Never, defaultCloseTimeout, defaultSendTimeout)
 | |
|             {
 | |
|             }
 | |
| 
 | |
|             public DuplexServerReliableChannelBinder(IDuplexChannel channel,
 | |
|                 EndpointAddress cachedLocalAddress, EndpointAddress remoteAddress,
 | |
|                 MaskingMode maskingMode, TimeSpan defaultCloseTimeout, TimeSpan defaultSendTimeout)
 | |
|                 : base(channel, cachedLocalAddress, remoteAddress, maskingMode, TolerateFaultsMode.Never,
 | |
|                 defaultCloseTimeout, defaultSendTimeout)
 | |
|             {
 | |
|             }
 | |
| 
 | |
|             public override bool HasSession
 | |
|             {
 | |
|                 get
 | |
|                 {
 | |
|                     return false;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             public override ISession GetInnerSession()
 | |
|             {
 | |
|                 return null;
 | |
|             }
 | |
| 
 | |
|             protected override bool HasSecuritySession(IDuplexChannel channel)
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             protected override void OnMessageReceived(Message message)
 | |
|             {
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         sealed class DuplexSessionServerReliableChannelBinder
 | |
|             : DuplexServerReliableChannelBinder<IDuplexSessionChannel>
 | |
|         {
 | |
|             public DuplexSessionServerReliableChannelBinder(ChannelBuilder builder,
 | |
|                 EndpointAddress remoteAddress, MessageFilter filter, int priority,
 | |
|                 MaskingMode maskingMode, TolerateFaultsMode faultMode,
 | |
|                 TimeSpan defaultCloseTimeout, TimeSpan defaultSendTimeout)
 | |
|                 : base(builder, remoteAddress, filter, priority, maskingMode, faultMode,
 | |
|                 defaultCloseTimeout, defaultSendTimeout)
 | |
|             {
 | |
|             }
 | |
| 
 | |
|             public DuplexSessionServerReliableChannelBinder(IDuplexSessionChannel channel,
 | |
|                 EndpointAddress cachedLocalAddress, EndpointAddress remoteAddress,
 | |
|                 MaskingMode maskingMode, TolerateFaultsMode faultMode,
 | |
|                 TimeSpan defaultCloseTimeout, TimeSpan defaultSendTimeout)
 | |
|                 : base(channel, cachedLocalAddress, remoteAddress, maskingMode, faultMode,
 | |
|                 defaultCloseTimeout, defaultSendTimeout)
 | |
|             {
 | |
|             }
 | |
| 
 | |
|             public override bool HasSession
 | |
|             {
 | |
|                 get
 | |
|                 {
 | |
|                     return true;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             protected override IAsyncResult BeginCloseChannel(IDuplexSessionChannel channel,
 | |
|                 TimeSpan timeout, AsyncCallback callback, object state)
 | |
|             {
 | |
|                 return ReliableChannelBinderHelper.BeginCloseDuplexSessionChannel(this, channel,
 | |
|                     timeout, callback, state);
 | |
|             }
 | |
| 
 | |
|             protected override void CloseChannel(IDuplexSessionChannel channel, TimeSpan timeout)
 | |
|             {
 | |
|                 ReliableChannelBinderHelper.CloseDuplexSessionChannel(this, channel, timeout);
 | |
|             }
 | |
| 
 | |
|             protected override void EndCloseChannel(IDuplexSessionChannel channel,
 | |
|                 IAsyncResult result)
 | |
|             {
 | |
|                 ReliableChannelBinderHelper.EndCloseDuplexSessionChannel(channel, result);
 | |
|             }
 | |
| 
 | |
|             public override ISession GetInnerSession()
 | |
|             {
 | |
|                 return this.Synchronizer.CurrentChannel.Session;
 | |
|             }
 | |
| 
 | |
|             protected override bool HasSecuritySession(IDuplexSessionChannel channel)
 | |
|             {
 | |
|                 return channel.Session is ISecuritySession;
 | |
|             }
 | |
| 
 | |
|             protected override void OnMessageReceived(Message message)
 | |
|             {
 | |
|                 if (message == null)
 | |
|                     this.Synchronizer.OnReadEof();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         abstract class ReplyServerReliableChannelBinder<TReplyChannel>
 | |
|             : ServerReliableChannelBinder<TReplyChannel>
 | |
|             where TReplyChannel : class, IReplyChannel
 | |
|         {
 | |
|             public ReplyServerReliableChannelBinder(ChannelBuilder builder,
 | |
|                 EndpointAddress remoteAddress, MessageFilter filter, int priority,
 | |
|                 MaskingMode maskingMode, TolerateFaultsMode faultMode,
 | |
|                 TimeSpan defaultCloseTimeout, TimeSpan defaultSendTimeout)
 | |
|                 : base(builder, remoteAddress, filter, priority, maskingMode, faultMode,
 | |
|                 defaultCloseTimeout, defaultSendTimeout)
 | |
|             {
 | |
|             }
 | |
| 
 | |
|             public ReplyServerReliableChannelBinder(TReplyChannel channel,
 | |
|                 EndpointAddress cachedLocalAddress, EndpointAddress remoteAddress,
 | |
|                 MaskingMode maskingMode, TolerateFaultsMode faultMode,
 | |
|                 TimeSpan defaultCloseTimeout, TimeSpan defaultSendTimeout)
 | |
|                 : base(channel, cachedLocalAddress, remoteAddress, maskingMode, faultMode,
 | |
|                 defaultCloseTimeout, defaultSendTimeout)
 | |
|             {
 | |
|             }
 | |
| 
 | |
|             public override bool CanSendAsynchronously
 | |
|             {
 | |
|                 get
 | |
|                 {
 | |
|                     return false;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             protected override EndpointAddress GetInnerChannelLocalAddress()
 | |
|             {
 | |
|                 IReplyChannel channel = this.Synchronizer.CurrentChannel;
 | |
|                 EndpointAddress localAddress = (channel == null) ? null : channel.LocalAddress;
 | |
|                 return localAddress;
 | |
|             }
 | |
| 
 | |
|             protected override IAsyncResult OnBeginTryReceive(TReplyChannel channel,
 | |
|                 TimeSpan timeout, AsyncCallback callback, object state)
 | |
|             {
 | |
|                 return channel.BeginTryReceiveRequest(timeout, callback, state);
 | |
|             }
 | |
| 
 | |
|             protected override IAsyncResult OnBeginWaitForRequest(TReplyChannel channel,
 | |
|                 TimeSpan timeout, AsyncCallback callback, object state)
 | |
|             {
 | |
|                 return channel.BeginWaitForRequest(timeout, callback, state);
 | |
|             }
 | |
| 
 | |
|             protected override bool OnEndTryReceive(TReplyChannel channel, IAsyncResult result,
 | |
|                 out RequestContext requestContext)
 | |
|             {
 | |
|                 bool success = channel.EndTryReceiveRequest(result, out requestContext);
 | |
|                 if (success && (requestContext == null))
 | |
|                 {
 | |
|                     this.OnReadNullMessage();
 | |
|                 }
 | |
|                 requestContext = this.WrapRequestContext(requestContext);
 | |
|                 return success;
 | |
|             }
 | |
| 
 | |
|             protected override bool OnEndWaitForRequest(TReplyChannel channel, IAsyncResult result)
 | |
|             {
 | |
|                 return channel.EndWaitForRequest(result);
 | |
|             }
 | |
| 
 | |
|             protected virtual void OnReadNullMessage()
 | |
|             {
 | |
|             }
 | |
| 
 | |
|             protected override bool OnTryReceive(TReplyChannel channel, TimeSpan timeout,
 | |
|                 out RequestContext requestContext)
 | |
|             {
 | |
|                 bool success = channel.TryReceiveRequest(timeout, out requestContext);
 | |
|                 if (success && (requestContext == null))
 | |
|                 {
 | |
|                     this.OnReadNullMessage();
 | |
|                 }
 | |
|                 requestContext = this.WrapRequestContext(requestContext);
 | |
|                 return success;
 | |
|             }
 | |
| 
 | |
|             protected override bool OnWaitForRequest(TReplyChannel channel, TimeSpan timeout)
 | |
|             {
 | |
|                 return channel.WaitForRequest(timeout);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         sealed class ReplyServerReliableChannelBinder
 | |
|             : ReplyServerReliableChannelBinder<IReplyChannel>
 | |
|         {
 | |
|             public ReplyServerReliableChannelBinder(ChannelBuilder builder,
 | |
|                 EndpointAddress remoteAddress, MessageFilter filter, int priority,
 | |
|                 MaskingMode maskingMode, TimeSpan defaultCloseTimeout, TimeSpan defaultSendTimeout)
 | |
|                 : base(builder, remoteAddress, filter, priority, maskingMode,
 | |
|                 TolerateFaultsMode.Never, defaultCloseTimeout, defaultSendTimeout)
 | |
|             {
 | |
|             }
 | |
| 
 | |
|             public ReplyServerReliableChannelBinder(IReplyChannel channel,
 | |
|                 EndpointAddress cachedLocalAddress, EndpointAddress remoteAddress,
 | |
|                 MaskingMode maskingMode, TimeSpan defaultCloseTimeout, TimeSpan defaultSendTimeout)
 | |
|                 : base(channel, cachedLocalAddress, remoteAddress, maskingMode,
 | |
|                 TolerateFaultsMode.Never, defaultCloseTimeout, defaultSendTimeout)
 | |
|             {
 | |
|             }
 | |
| 
 | |
|             public override bool HasSession
 | |
|             {
 | |
|                 get
 | |
|                 {
 | |
|                     return false;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             public override ISession GetInnerSession()
 | |
|             {
 | |
|                 return null;
 | |
|             }
 | |
| 
 | |
|             protected override bool HasSecuritySession(IReplyChannel channel)
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         sealed class ReplySessionServerReliableChannelBinder
 | |
|             : ReplyServerReliableChannelBinder<IReplySessionChannel>
 | |
|         {
 | |
|             public ReplySessionServerReliableChannelBinder(ChannelBuilder builder,
 | |
|                 EndpointAddress remoteAddress, MessageFilter filter, int priority,
 | |
|                 MaskingMode maskingMode, TolerateFaultsMode faultMode,
 | |
|                 TimeSpan defaultCloseTimeout, TimeSpan defaultSendTimeout)
 | |
|                 : base(builder, remoteAddress, filter, priority, maskingMode, faultMode,
 | |
|                 defaultCloseTimeout, defaultSendTimeout)
 | |
|             {
 | |
|             }
 | |
| 
 | |
|             public ReplySessionServerReliableChannelBinder(IReplySessionChannel channel,
 | |
|                 EndpointAddress cachedLocalAddress, EndpointAddress remoteAddress,
 | |
|                 MaskingMode maskingMode, TolerateFaultsMode faultMode,
 | |
|                 TimeSpan defaultCloseTimeout, TimeSpan defaultSendTimeout)
 | |
|                 : base(channel, cachedLocalAddress, remoteAddress, maskingMode, faultMode,
 | |
|                 defaultCloseTimeout, defaultSendTimeout)
 | |
|             {
 | |
|             }
 | |
| 
 | |
|             public override bool HasSession
 | |
|             {
 | |
|                 get
 | |
|                 {
 | |
|                     return true;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             protected override IAsyncResult BeginCloseChannel(IReplySessionChannel channel,
 | |
|                TimeSpan timeout, AsyncCallback callback, object state)
 | |
|             {
 | |
|                 return ReliableChannelBinderHelper.BeginCloseReplySessionChannel(this, channel,
 | |
|                     timeout, callback, state);
 | |
|             }
 | |
| 
 | |
|             protected override void CloseChannel(IReplySessionChannel channel, TimeSpan timeout)
 | |
|             {
 | |
|                 ReliableChannelBinderHelper.CloseReplySessionChannel(this, channel, timeout);
 | |
|             }
 | |
| 
 | |
|             protected override void EndCloseChannel(IReplySessionChannel channel,
 | |
|                 IAsyncResult result)
 | |
|             {
 | |
|                 ReliableChannelBinderHelper.EndCloseReplySessionChannel(channel, result);
 | |
|             }
 | |
| 
 | |
|             public override ISession GetInnerSession()
 | |
|             {
 | |
|                 return this.Synchronizer.CurrentChannel.Session;
 | |
|             }
 | |
| 
 | |
|             protected override bool HasSecuritySession(IReplySessionChannel channel)
 | |
|             {
 | |
|                 return channel.Session is ISecuritySession;
 | |
|             }
 | |
| 
 | |
|             protected override void OnReadNullMessage()
 | |
|             {
 | |
|                 this.Synchronizer.OnReadEof();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         sealed class WaitForRequestAsyncResult
 | |
|             : InputAsyncResult<ServerReliableChannelBinder<TChannel>>
 | |
|         {
 | |
|             public WaitForRequestAsyncResult(ServerReliableChannelBinder<TChannel> binder,
 | |
|                 TimeSpan timeout, AsyncCallback callback, object state)
 | |
|                 : base(binder, true, timeout, binder.DefaultMaskingMode, callback, state)
 | |
|             {
 | |
|                 if (this.Start())
 | |
|                     this.Complete(true);
 | |
|             }
 | |
| 
 | |
|             protected override IAsyncResult BeginInput(
 | |
|                 ServerReliableChannelBinder<TChannel> binder, TChannel channel, TimeSpan timeout,
 | |
|                 AsyncCallback callback, object state)
 | |
|             {
 | |
|                 return binder.OnBeginWaitForRequest(channel, timeout, callback, state);
 | |
|             }
 | |
| 
 | |
|             protected override bool EndInput(ServerReliableChannelBinder<TChannel> binder,
 | |
|                 TChannel channel, IAsyncResult result, out bool complete)
 | |
|             {
 | |
|                 complete = true;
 | |
|                 return binder.OnEndWaitForRequest(channel, result);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 |