You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			423 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			423 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //------------------------------------------------------------ | ||
|  | // Copyright (c) Microsoft Corporation.  All rights reserved. | ||
|  | //------------------------------------------------------------ | ||
|  | 
 | ||
|  | namespace System.ServiceModel.Channels | ||
|  | { | ||
|  |     using System; | ||
|  |     using System.Diagnostics; | ||
|  |     using System.IO; | ||
|  |     using System.Runtime; | ||
|  |     using System.Runtime.Diagnostics; | ||
|  |     using System.ServiceModel; | ||
|  |     using System.ServiceModel.Diagnostics; | ||
|  |     using System.ServiceModel.Diagnostics.Application; | ||
|  | 
 | ||
|  | 
 | ||
|  |     delegate IConnectionOrientedTransportFactorySettings TransportSettingsCallback(Uri via); | ||
|  |     delegate void ConnectionClosedCallback(InitialServerConnectionReader connectionReader); | ||
|  | 
 | ||
|  |     // Host for a connection that deals with structured close/abort and notifying the owner appropriately | ||
|  |     // used for cases where no one else (channel, etc) actually owns the reader | ||
|  |     abstract class InitialServerConnectionReader : IDisposable | ||
|  |     { | ||
|  |         int maxViaSize; | ||
|  |         int maxContentTypeSize; | ||
|  |         IConnection connection; | ||
|  |         Action connectionDequeuedCallback; | ||
|  |         ConnectionClosedCallback closedCallback; | ||
|  |         bool isClosed; | ||
|  | 
 | ||
|  |         protected InitialServerConnectionReader(IConnection connection, ConnectionClosedCallback closedCallback) | ||
|  |             : this(connection, closedCallback, | ||
|  |             ConnectionOrientedTransportDefaults.MaxViaSize, ConnectionOrientedTransportDefaults.MaxContentTypeSize) | ||
|  |         { | ||
|  |         } | ||
|  | 
 | ||
|  |         protected InitialServerConnectionReader(IConnection connection, ConnectionClosedCallback closedCallback, int maxViaSize, int maxContentTypeSize) | ||
|  |         { | ||
|  |             if (connection == null) | ||
|  |             { | ||
|  |                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("connection"); | ||
|  |             } | ||
|  | 
 | ||
|  |             if (closedCallback == null) | ||
|  |             { | ||
|  |                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("closedCallback"); | ||
|  |             } | ||
|  | 
 | ||
|  |             this.connection = connection; | ||
|  |             this.closedCallback = closedCallback; | ||
|  |             this.maxContentTypeSize = maxContentTypeSize; | ||
|  |             this.maxViaSize = maxViaSize; | ||
|  |         } | ||
|  | 
 | ||
|  |         public IConnection Connection | ||
|  |         { | ||
|  |             get { return connection; } | ||
|  |         } | ||
|  | 
 | ||
|  |         public Action ConnectionDequeuedCallback | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 return this.connectionDequeuedCallback; | ||
|  |             } | ||
|  | 
 | ||
|  |             set | ||
|  |             { | ||
|  |                 this.connectionDequeuedCallback = value; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         public Action GetConnectionDequeuedCallback() | ||
|  |         { | ||
|  |             Action dequeuedCallback = this.connectionDequeuedCallback; | ||
|  |             this.connectionDequeuedCallback = null; | ||
|  |             return dequeuedCallback; | ||
|  |         } | ||
|  | 
 | ||
|  |         protected bool IsClosed | ||
|  |         { | ||
|  |             get { return isClosed; } | ||
|  |         } | ||
|  | 
 | ||
|  |         protected int MaxContentTypeSize | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 return maxContentTypeSize; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         protected int MaxViaSize | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 return maxViaSize; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         object ThisLock | ||
|  |         { | ||
|  |             get { return this; } | ||
|  |         } | ||
|  | 
 | ||
|  |         // used by the listener to release the connection object so it can be closed at a later time | ||
|  |         public void ReleaseConnection() | ||
|  |         { | ||
|  |             isClosed = true; | ||
|  |             connection = null; | ||
|  |         } | ||
|  | 
 | ||
|  |         // for cached connections -- try to shut down gracefully if possible | ||
|  |         public void CloseFromPool(TimeSpan timeout) | ||
|  |         { | ||
|  |             try | ||
|  |             { | ||
|  |                 Close(timeout); | ||
|  |             } | ||
|  |             catch (CommunicationException communicationException) | ||
|  |             { | ||
|  |                 DiagnosticUtility.TraceHandledException(communicationException, TraceEventType.Information); | ||
|  |             } | ||
|  |             catch (TimeoutException timeoutException) | ||
|  |             { | ||
|  |                 if (TD.CloseTimeoutIsEnabled()) | ||
|  |                 { | ||
|  |                     TD.CloseTimeout(timeoutException.Message); | ||
|  |                 } | ||
|  |                 DiagnosticUtility.TraceHandledException(timeoutException, TraceEventType.Information); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         public void Dispose() | ||
|  |         { | ||
|  |             lock (ThisLock) | ||
|  |             { | ||
|  |                 if (isClosed) | ||
|  |                 { | ||
|  |                     return; | ||
|  |                 } | ||
|  | 
 | ||
|  |                 this.isClosed = true; | ||
|  |             } | ||
|  | 
 | ||
|  |             IConnection connection = this.connection; | ||
|  |             if (connection != null) | ||
|  |             { | ||
|  |                 connection.Abort(); | ||
|  |             } | ||
|  | 
 | ||
|  |             if (this.connectionDequeuedCallback != null) | ||
|  |             { | ||
|  |                 this.connectionDequeuedCallback(); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         protected void Abort() | ||
|  |         { | ||
|  |             Abort(null); | ||
|  |         } | ||
|  | 
 | ||
|  |         internal void Abort(Exception e) | ||
|  |         { | ||
|  |             lock (ThisLock) | ||
|  |             { | ||
|  |                 if (isClosed) | ||
|  |                     return; | ||
|  | 
 | ||
|  |                 isClosed = true; | ||
|  |             } | ||
|  | 
 | ||
|  |             try | ||
|  |             { | ||
|  |                 if (e != null) | ||
|  |                 { | ||
|  |                     if (DiagnosticUtility.ShouldTraceError) | ||
|  |                     { | ||
|  |                         TraceUtility.TraceEvent(TraceEventType.Error, TraceCode.ChannelConnectionDropped, | ||
|  |                             SR.GetString(SR.TraceCodeChannelConnectionDropped), this, e); | ||
|  |                     } | ||
|  |                 } | ||
|  | 
 | ||
|  |                 connection.Abort(); | ||
|  |             } | ||
|  |             finally | ||
|  |             { | ||
|  |                 if (closedCallback != null) | ||
|  |                 { | ||
|  |                     closedCallback(this); | ||
|  |                 } | ||
|  | 
 | ||
|  |                 if (this.connectionDequeuedCallback != null) | ||
|  |                 { | ||
|  |                     this.connectionDequeuedCallback(); | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         protected void Close(TimeSpan timeout) | ||
|  |         { | ||
|  |             lock (ThisLock) | ||
|  |             { | ||
|  |                 if (isClosed) | ||
|  |                     return; | ||
|  | 
 | ||
|  |                 isClosed = true; | ||
|  |             } | ||
|  | 
 | ||
|  |             bool success = false; | ||
|  |             try | ||
|  |             { | ||
|  |                 connection.Close(timeout, true); | ||
|  |                 success = true; | ||
|  |             } | ||
|  |             finally | ||
|  |             { | ||
|  |                 if (!success) | ||
|  |                 { | ||
|  |                     connection.Abort(); | ||
|  |                 } | ||
|  | 
 | ||
|  |                 if (closedCallback != null) | ||
|  |                 { | ||
|  |                     closedCallback(this); | ||
|  |                 } | ||
|  | 
 | ||
|  |                 if (this.connectionDequeuedCallback != null) | ||
|  |                 { | ||
|  |                     this.connectionDequeuedCallback(); | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static void SendFault(IConnection connection, string faultString, byte[] drainBuffer, TimeSpan sendTimeout, int maxRead) | ||
|  |         { | ||
|  | 
 | ||
|  |             if (TD.ConnectionReaderSendFaultIsEnabled()) | ||
|  |             { | ||
|  |                 TD.ConnectionReaderSendFault(faultString); | ||
|  |             } | ||
|  | 
 | ||
|  |             EncodedFault encodedFault = new EncodedFault(faultString); | ||
|  |             TimeoutHelper timeoutHelper = new TimeoutHelper(sendTimeout); | ||
|  |             try | ||
|  |             { | ||
|  |                 connection.Write(encodedFault.EncodedBytes, 0, encodedFault.EncodedBytes.Length, true, timeoutHelper.RemainingTime()); | ||
|  |                 connection.Shutdown(timeoutHelper.RemainingTime()); | ||
|  |             } | ||
|  |             catch (CommunicationException e) | ||
|  |             { | ||
|  |                 DiagnosticUtility.TraceHandledException(e, TraceEventType.Information); | ||
|  |                 connection.Abort(); | ||
|  |                 return; | ||
|  |             } | ||
|  |             catch (TimeoutException e) | ||
|  |             { | ||
|  |                 if (TD.SendTimeoutIsEnabled()) | ||
|  |                 { | ||
|  |                     TD.SendTimeout(e.Message); | ||
|  |                 } | ||
|  |                 DiagnosticUtility.TraceHandledException(e, TraceEventType.Information); | ||
|  |                 connection.Abort(); | ||
|  |                 return; | ||
|  |             } | ||
|  | 
 | ||
|  |             // make sure we read until EOF or a quota is hit | ||
|  |             int read = 0; | ||
|  |             int readTotal = 0; | ||
|  |             for (;;) | ||
|  |             { | ||
|  |                 try | ||
|  |                 { | ||
|  |                     read = connection.Read(drainBuffer, 0, drainBuffer.Length, timeoutHelper.RemainingTime()); | ||
|  |                 } | ||
|  |                 catch (CommunicationException e) | ||
|  |                 { | ||
|  |                     DiagnosticUtility.TraceHandledException(e, TraceEventType.Information); | ||
|  |                     connection.Abort(); | ||
|  |                     return; | ||
|  |                 } | ||
|  |                 catch (TimeoutException e) | ||
|  |                 { | ||
|  |                     if (TD.SendTimeoutIsEnabled()) | ||
|  |                     { | ||
|  |                         TD.SendTimeout(e.Message); | ||
|  |                     } | ||
|  |                     DiagnosticUtility.TraceHandledException(e, TraceEventType.Information); | ||
|  |                     connection.Abort(); | ||
|  |                     return; | ||
|  |                 } | ||
|  | 
 | ||
|  |                 if (read == 0) | ||
|  |                     break; | ||
|  | 
 | ||
|  |                 readTotal += read; | ||
|  |                 if (readTotal > maxRead || timeoutHelper.RemainingTime() <= TimeSpan.Zero) | ||
|  |                 { | ||
|  |                     connection.Abort(); | ||
|  |                     return; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             ConnectionUtilities.CloseNoThrow(connection, timeoutHelper.RemainingTime()); | ||
|  |         } | ||
|  | 
 | ||
|  |         public static IAsyncResult BeginUpgradeConnection(IConnection connection, StreamUpgradeAcceptor upgradeAcceptor, | ||
|  |             IDefaultCommunicationTimeouts defaultTimeouts, AsyncCallback callback, object state) | ||
|  |         { | ||
|  |             return new UpgradeConnectionAsyncResult(connection, upgradeAcceptor, defaultTimeouts, callback, state); | ||
|  |         } | ||
|  | 
 | ||
|  |         public static IConnection EndUpgradeConnection(IAsyncResult result) | ||
|  |         { | ||
|  |             // get our upgraded connection | ||
|  |             return UpgradeConnectionAsyncResult.End(result); | ||
|  |         } | ||
|  | 
 | ||
|  |         public static IConnection UpgradeConnection(IConnection connection, StreamUpgradeAcceptor upgradeAcceptor, IDefaultCommunicationTimeouts defaultTimeouts) | ||
|  |         { | ||
|  |             ConnectionStream connectionStream = new ConnectionStream(connection, defaultTimeouts); | ||
|  |             Stream stream = upgradeAcceptor.AcceptUpgrade(connectionStream); | ||
|  |             if (upgradeAcceptor is StreamSecurityUpgradeAcceptor) | ||
|  |             { | ||
|  |                 if (DiagnosticUtility.ShouldTraceInformation) | ||
|  |                 { | ||
|  |                     TraceUtility.TraceEvent(TraceEventType.Information, | ||
|  |                         TraceCode.StreamSecurityUpgradeAccepted, SR.GetString(SR.TraceCodeStreamSecurityUpgradeAccepted), | ||
|  |                         new StringTraceRecord("Type", upgradeAcceptor.GetType().ToString()), connection, null); | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             return new StreamConnection(stream, connectionStream); | ||
|  |         } | ||
|  | 
 | ||
|  |         class UpgradeConnectionAsyncResult : AsyncResult | ||
|  |         { | ||
|  |             ConnectionStream connectionStream; | ||
|  |             static AsyncCallback onAcceptUpgrade = Fx.ThunkCallback(new AsyncCallback(OnAcceptUpgrade)); | ||
|  |             IConnection connection; | ||
|  |             StreamUpgradeAcceptor upgradeAcceptor; | ||
|  | 
 | ||
|  |             public UpgradeConnectionAsyncResult(IConnection connection, | ||
|  |                 StreamUpgradeAcceptor upgradeAcceptor, IDefaultCommunicationTimeouts defaultTimeouts, | ||
|  |                 AsyncCallback callback, object state) | ||
|  |                 : base(callback, state) | ||
|  |             { | ||
|  |                 this.upgradeAcceptor = upgradeAcceptor; | ||
|  |                 this.connectionStream = new ConnectionStream(connection, defaultTimeouts); | ||
|  |                 bool completeSelf = false; | ||
|  | 
 | ||
|  |                 IAsyncResult result = upgradeAcceptor.BeginAcceptUpgrade(connectionStream, onAcceptUpgrade, this); | ||
|  | 
 | ||
|  |                 if (result.CompletedSynchronously) | ||
|  |                 { | ||
|  |                     CompleteAcceptUpgrade(result); | ||
|  |                     completeSelf = true; | ||
|  |                 } | ||
|  | 
 | ||
|  |                 if (completeSelf) | ||
|  |                 { | ||
|  |                     base.Complete(true); | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             public static IConnection End(IAsyncResult result) | ||
|  |             { | ||
|  |                 UpgradeConnectionAsyncResult thisPtr = AsyncResult.End<UpgradeConnectionAsyncResult>(result); | ||
|  |                 return thisPtr.connection; | ||
|  |             } | ||
|  | 
 | ||
|  |             void CompleteAcceptUpgrade(IAsyncResult result) | ||
|  |             { | ||
|  |                 Stream stream; | ||
|  |                 bool endSucceeded = false; | ||
|  |                 try | ||
|  |                 { | ||
|  |                     stream = this.upgradeAcceptor.EndAcceptUpgrade(result); | ||
|  |                     endSucceeded = true; | ||
|  |                 } | ||
|  |                 finally | ||
|  |                 { | ||
|  |                     if (upgradeAcceptor is StreamSecurityUpgradeAcceptor) | ||
|  |                     { | ||
|  |                         if (DiagnosticUtility.ShouldTraceInformation && endSucceeded) | ||
|  |                         { | ||
|  |                             TraceUtility.TraceEvent(TraceEventType.Information, | ||
|  |                                 TraceCode.StreamSecurityUpgradeAccepted, SR.GetString(SR.TraceCodeStreamSecurityUpgradeAccepted), | ||
|  |                                 new StringTraceRecord("Type", upgradeAcceptor.GetType().ToString()), this, null); | ||
|  |                         } | ||
|  |                     } | ||
|  |                 } | ||
|  |                 this.connection = new StreamConnection(stream, this.connectionStream); | ||
|  |             } | ||
|  | 
 | ||
|  |             static void OnAcceptUpgrade(IAsyncResult result) | ||
|  |             { | ||
|  |                 if (result.CompletedSynchronously) | ||
|  |                     return; | ||
|  | 
 | ||
|  |                 UpgradeConnectionAsyncResult thisPtr = (UpgradeConnectionAsyncResult)result.AsyncState; | ||
|  |                 Exception completionException = null; | ||
|  |                 try | ||
|  |                 { | ||
|  |                     thisPtr.CompleteAcceptUpgrade(result); | ||
|  |                 } | ||
|  | #pragma warning suppress 56500 // [....], transferring exception to another thread | ||
|  |                 catch (Exception e) | ||
|  |                 { | ||
|  |                     if (Fx.IsFatal(e)) | ||
|  |                     { | ||
|  |                         throw; | ||
|  |                     } | ||
|  | 
 | ||
|  |                     completionException = e; | ||
|  |                 } | ||
|  | 
 | ||
|  |                 thisPtr.Complete(false, completionException); | ||
|  |             } | ||
|  |         } | ||
|  |     } | ||
|  | } |