You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			387 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			387 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //------------------------------------------------------------ | ||
|  | // Copyright (c) Microsoft Corporation.  All rights reserved. | ||
|  | //------------------------------------------------------------ | ||
|  | namespace System.ServiceModel.Dispatcher | ||
|  | { | ||
|  |     using System.Collections.ObjectModel; | ||
|  |     using System.ServiceModel.Channels; | ||
|  |     using System.ServiceModel; | ||
|  |     using System.ServiceModel.Description; | ||
|  |     using System.Collections.Generic; | ||
|  |     using System.Runtime.CompilerServices; | ||
|  |     using System.ServiceModel.Transactions; | ||
|  | 
 | ||
|  |     class TransactionValidationBehavior : IEndpointBehavior, IServiceBehavior | ||
|  |     { | ||
|  |         static TransactionValidationBehavior instance; | ||
|  | 
 | ||
|  |         internal static TransactionValidationBehavior Instance | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 if (instance == null) | ||
|  |                     instance = new TransactionValidationBehavior(); | ||
|  |                 return instance; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         TransactionValidationBehavior() { } | ||
|  | 
 | ||
|  |         void ValidateTransactionFlowRequired(string resource, string name, ServiceEndpoint endpoint) | ||
|  |         { | ||
|  |             bool anOperationRequiresTxFlow = false; | ||
|  |             for (int i = 0; i < endpoint.Contract.Operations.Count; i++) | ||
|  |             { | ||
|  |                 OperationDescription operationDescription = endpoint.Contract.Operations[i]; | ||
|  |                 TransactionFlowAttribute transactionFlow = operationDescription.Behaviors.Find<TransactionFlowAttribute>(); | ||
|  |                 if (transactionFlow != null && transactionFlow.Transactions == TransactionFlowOption.Mandatory) | ||
|  |                 { | ||
|  |                     anOperationRequiresTxFlow = true; | ||
|  |                     break; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             if (anOperationRequiresTxFlow) | ||
|  |             { | ||
|  |                 CustomBinding binding = new CustomBinding(endpoint.Binding); | ||
|  |                 TransactionFlowBindingElement transactionFlowBindingElement = | ||
|  |                                               binding.Elements.Find<TransactionFlowBindingElement>(); | ||
|  | 
 | ||
|  |                 if (transactionFlowBindingElement == null || !transactionFlowBindingElement.Transactions) | ||
|  |                 { | ||
|  |                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException( | ||
|  |                         String.Format(Globalization.CultureInfo.CurrentCulture, SR.GetString(resource), name, binding.Name))); | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         void IEndpointBehavior.Validate(ServiceEndpoint serviceEndpoint) | ||
|  |         { | ||
|  |             if (serviceEndpoint == null) | ||
|  |                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("serviceEndpoint"); | ||
|  |             ValidateTransactionFlowRequired(SR.ChannelHasAtLeastOneOperationWithTransactionFlowEnabled, | ||
|  |                            serviceEndpoint.Contract.Name, | ||
|  |                            serviceEndpoint); | ||
|  |             EnsureNoOneWayTransactions(serviceEndpoint); | ||
|  |             ValidateNoMSMQandTransactionFlow(serviceEndpoint); | ||
|  |             ValidateCallbackBehaviorAttributeWithNoScopeRequired(serviceEndpoint); | ||
|  |             OperationDescription autoCompleteFalseOperation = GetAutoCompleteFalseOperation(serviceEndpoint); | ||
|  |             if (autoCompleteFalseOperation != null) | ||
|  |             { | ||
|  |                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException( | ||
|  |                     SR.GetString(SR.SFxTransactionAutoCompleteFalseOnCallbackContract, autoCompleteFalseOperation.Name, serviceEndpoint.Contract.Name))); | ||
|  |             } | ||
|  | 
 | ||
|  |         } | ||
|  | 
 | ||
|  |         void ValidateCallbackBehaviorAttributeWithNoScopeRequired(ServiceEndpoint endpoint) | ||
|  |         { | ||
|  |             // If the endpoint has no operations with TransactionScopeRequired=true, disallow any | ||
|  |             // transaction-related properties on the CallbackBehaviorAttribute | ||
|  |             if (!HasTransactedOperations(endpoint)) | ||
|  |             { | ||
|  |                 CallbackBehaviorAttribute attribute = endpoint.Behaviors.Find<CallbackBehaviorAttribute>(); | ||
|  |                 if (attribute != null) | ||
|  |                 { | ||
|  |                     if (attribute.TransactionTimeoutSet) | ||
|  |                     { | ||
|  |                         throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException( | ||
|  |                             SR.GetString(SR.SFxTransactionTransactionTimeoutNeedsScope, endpoint.Contract.Name))); | ||
|  |                     } | ||
|  | 
 | ||
|  |                     if (attribute.IsolationLevelSet) | ||
|  |                     { | ||
|  |                         throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException( | ||
|  |                             SR.GetString(SR.SFxTransactionIsolationLevelNeedsScope, endpoint.Contract.Name))); | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         void IEndpointBehavior.AddBindingParameters(ServiceEndpoint serviceEndpoint, BindingParameterCollection bindingParameters) | ||
|  |         { | ||
|  |         } | ||
|  | 
 | ||
|  |         void IEndpointBehavior.ApplyDispatchBehavior(ServiceEndpoint serviceEndpoint, EndpointDispatcher endpointDispatcher) | ||
|  |         { | ||
|  |         } | ||
|  | 
 | ||
|  |         void IEndpointBehavior.ApplyClientBehavior(ServiceEndpoint serviceEndpoint, ClientRuntime behavior) | ||
|  |         { | ||
|  |         } | ||
|  | 
 | ||
|  |         void IServiceBehavior.AddBindingParameters(ServiceDescription description, ServiceHostBase serviceHostBase, Collection<ServiceEndpoint> endpoints, BindingParameterCollection parameters) | ||
|  |         { | ||
|  |         } | ||
|  | 
 | ||
|  |         void IServiceBehavior.ApplyDispatchBehavior(ServiceDescription service, ServiceHostBase serviceHostBase) | ||
|  |         { | ||
|  |         } | ||
|  | 
 | ||
|  |         void IServiceBehavior.Validate(ServiceDescription service, ServiceHostBase serviceHostBase) | ||
|  |         { | ||
|  |             if (service == null) | ||
|  |                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("service"); | ||
|  | 
 | ||
|  |             ValidateNotConcurrentWhenReleaseServiceInstanceOnTxComplete(service); | ||
|  |             bool singleThreaded = IsSingleThreaded(service); | ||
|  | 
 | ||
|  |             for (int i = 0; i < service.Endpoints.Count; i++) | ||
|  |             { | ||
|  |                 ServiceEndpoint endpoint = service.Endpoints[i]; | ||
|  | 
 | ||
|  |                 ValidateTransactionFlowRequired(SR.ServiceHasAtLeastOneOperationWithTransactionFlowEnabled, | ||
|  |                                service.Name, | ||
|  |                                endpoint); | ||
|  |                 EnsureNoOneWayTransactions(endpoint); | ||
|  |                 ValidateNoMSMQandTransactionFlow(endpoint); | ||
|  | 
 | ||
|  |                 ContractDescription contract = endpoint.Contract; | ||
|  |                 for (int j = 0; j < contract.Operations.Count; j++) | ||
|  |                 { | ||
|  |                     OperationDescription operation = contract.Operations[j]; | ||
|  |                     ValidateScopeRequiredAndAutoComplete(operation, singleThreaded, contract.Name); | ||
|  |                 } | ||
|  | 
 | ||
|  |                 ValidateAutoCompleteFalseRequirements(service, endpoint); | ||
|  |             } | ||
|  | 
 | ||
|  |             ValidateServiceBehaviorAttributeWithNoScopeRequired(service); | ||
|  |             ValidateTransactionAutoCompleteOnSessionCloseHasSession(service); | ||
|  |         } | ||
|  | 
 | ||
|  | 
 | ||
|  |         void ValidateAutoCompleteFalseRequirements(ServiceDescription service, ServiceEndpoint endpoint) | ||
|  |         { | ||
|  |             OperationDescription autoCompleteFalseOperation = GetAutoCompleteFalseOperation(endpoint); | ||
|  | 
 | ||
|  |             if (autoCompleteFalseOperation != null) | ||
|  |             { | ||
|  |                 // Does the service have InstanceContextMode.PerSession or Shareable? | ||
|  |                 ServiceBehaviorAttribute serviceBehavior = service.Behaviors.Find<ServiceBehaviorAttribute>(); | ||
|  |                 if (serviceBehavior != null) | ||
|  |                 { | ||
|  |                     InstanceContextMode instanceMode = serviceBehavior.InstanceContextMode; | ||
|  |                     if (instanceMode != InstanceContextMode.PerSession) | ||
|  |                     { | ||
|  |                         throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException( | ||
|  |                             SR.GetString(SR.SFxTransactionAutoCompleteFalseAndInstanceContextMode, | ||
|  |                             endpoint.Contract.Name, autoCompleteFalseOperation.Name))); | ||
|  |                     } | ||
|  |                 } | ||
|  | 
 | ||
|  |                 // Does the binding support sessions? | ||
|  |                 if (!autoCompleteFalseOperation.IsInsideTransactedReceiveScope) | ||
|  |                 { | ||
|  |                     if (!RequiresSessions(endpoint)) | ||
|  |                     { | ||
|  |                         throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException( | ||
|  |                             SR.GetString(SR.SFxTransactionAutoCompleteFalseAndSupportsSession, | ||
|  |                             endpoint.Contract.Name, autoCompleteFalseOperation.Name))); | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         OperationDescription GetAutoCompleteFalseOperation(ServiceEndpoint endpoint) | ||
|  |         { | ||
|  |             foreach (OperationDescription operation in endpoint.Contract.Operations) | ||
|  |             { | ||
|  |                 if (!IsAutoComplete(operation)) | ||
|  |                 { | ||
|  |                     return operation; | ||
|  |                 } | ||
|  |             } | ||
|  |             return null; | ||
|  |         } | ||
|  | 
 | ||
|  |         void ValidateTransactionAutoCompleteOnSessionCloseHasSession(ServiceDescription service) | ||
|  |         { | ||
|  |             ServiceBehaviorAttribute serviceBehavior = service.Behaviors.Find<ServiceBehaviorAttribute>(); | ||
|  | 
 | ||
|  |             if (serviceBehavior != null) | ||
|  |             { | ||
|  |                 InstanceContextMode instanceMode = serviceBehavior.InstanceContextMode; | ||
|  |                 if (serviceBehavior.TransactionAutoCompleteOnSessionClose && | ||
|  |                     instanceMode != InstanceContextMode.PerSession) | ||
|  |                 { | ||
|  |                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException( | ||
|  |                         SR.GetString(SR.SFxTransactionAutoCompleteOnSessionCloseNoSession, service.Name))); | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |         } | ||
|  | 
 | ||
|  | 
 | ||
|  |         void ValidateServiceBehaviorAttributeWithNoScopeRequired(ServiceDescription service) | ||
|  |         { | ||
|  |             // If the service has no operations with TransactionScopeRequired=true, disallow any | ||
|  |             // transaction-related properties on the ServiceBehaviorAttribute | ||
|  |             if (!HasTransactedOperations(service)) | ||
|  |             { | ||
|  |                 ServiceBehaviorAttribute attribute = service.Behaviors.Find<ServiceBehaviorAttribute>(); | ||
|  |                 if (attribute != null) | ||
|  |                 { | ||
|  |                     if (attribute.TransactionTimeoutSet) | ||
|  |                     { | ||
|  |                         throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException( | ||
|  |                             SR.GetString(SR.SFxTransactionTransactionTimeoutNeedsScope, service.Name))); | ||
|  |                     } | ||
|  | 
 | ||
|  |                     if (attribute.IsolationLevelSet) | ||
|  |                     { | ||
|  |                         throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException( | ||
|  |                             SR.GetString(SR.SFxTransactionIsolationLevelNeedsScope, service.Name))); | ||
|  |                     } | ||
|  | 
 | ||
|  |                     if (attribute.ReleaseServiceInstanceOnTransactionCompleteSet) | ||
|  |                     { | ||
|  |                         throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException( | ||
|  |                             SR.GetString(SR.SFxTransactionReleaseServiceInstanceOnTransactionCompleteNeedsScope, service.Name))); | ||
|  |                     } | ||
|  | 
 | ||
|  |                     if (attribute.TransactionAutoCompleteOnSessionCloseSet) | ||
|  |                     { | ||
|  |                         throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException( | ||
|  |                             SR.GetString(SR.SFxTransactionTransactionAutoCompleteOnSessionCloseNeedsScope, service.Name))); | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         void EnsureNoOneWayTransactions(ServiceEndpoint endpoint) | ||
|  |         { | ||
|  |             CustomBinding binding = new CustomBinding(endpoint.Binding); | ||
|  |             TransactionFlowBindingElement txFlowBindingElement = binding.Elements.Find<TransactionFlowBindingElement>(); | ||
|  |             if (txFlowBindingElement != null) | ||
|  |             { | ||
|  |                 for (int i = 0; i < endpoint.Contract.Operations.Count; i++) | ||
|  |                 { | ||
|  |                     OperationDescription operation = endpoint.Contract.Operations[i]; | ||
|  |                     if (operation.IsOneWay) | ||
|  |                     { | ||
|  |                         TransactionFlowAttribute tfbp = operation.Behaviors.Find<TransactionFlowAttribute>(); | ||
|  |                         TransactionFlowOption transactions; | ||
|  |                         if (tfbp != null) | ||
|  |                         { | ||
|  |                             transactions = tfbp.Transactions; | ||
|  |                         } | ||
|  |                         else | ||
|  |                         { | ||
|  |                             transactions = TransactionFlowOption.NotAllowed; | ||
|  |                         } | ||
|  |                         if (TransactionFlowOptionHelper.AllowedOrRequired(transactions)) | ||
|  |                         { | ||
|  |                             throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException( | ||
|  |                                 SR.GetString(SR.SFxOneWayAndTransactionsIncompatible, endpoint.Contract.Name, operation.Name))); | ||
|  |                         } | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         bool HasTransactedOperations(ServiceDescription service) | ||
|  |         { | ||
|  |             for (int i = 0; i < service.Endpoints.Count; i++) | ||
|  |             { | ||
|  |                 if (HasTransactedOperations(service.Endpoints[i])) | ||
|  |                 { | ||
|  |                     return true; | ||
|  |                 } | ||
|  |             } | ||
|  |             return false; | ||
|  |         } | ||
|  | 
 | ||
|  |         bool HasTransactedOperations(ServiceEndpoint endpoint) | ||
|  |         { | ||
|  |             for (int j = 0; j < endpoint.Contract.Operations.Count; j++) | ||
|  |             { | ||
|  |                 OperationDescription operation = endpoint.Contract.Operations[j]; | ||
|  |                 OperationBehaviorAttribute attribute = operation.Behaviors.Find<OperationBehaviorAttribute>(); | ||
|  | 
 | ||
|  |                 if (attribute != null && attribute.TransactionScopeRequired) | ||
|  |                 { | ||
|  |                     return true; | ||
|  |                 } | ||
|  |             } | ||
|  |             return false; | ||
|  |         } | ||
|  | 
 | ||
|  |         bool IsSingleThreaded(ServiceDescription service) | ||
|  |         { | ||
|  |             ServiceBehaviorAttribute attribute = service.Behaviors.Find<ServiceBehaviorAttribute>(); | ||
|  | 
 | ||
|  |             if (attribute != null) | ||
|  |             { | ||
|  |                 return (attribute.ConcurrencyMode == ConcurrencyMode.Single); | ||
|  |             } | ||
|  | 
 | ||
|  |             // The default is ConcurrencyMode.Single | ||
|  |             return true; | ||
|  |         } | ||
|  | 
 | ||
|  |         bool IsAutoComplete(OperationDescription operation) | ||
|  |         { | ||
|  |             OperationBehaviorAttribute attribute = operation.Behaviors.Find<OperationBehaviorAttribute>(); | ||
|  | 
 | ||
|  |             if (attribute != null) | ||
|  |             { | ||
|  |                 return attribute.TransactionAutoComplete; | ||
|  |             } | ||
|  | 
 | ||
|  |             // The default is TransactionAutoComplete=true | ||
|  |             return true; | ||
|  |         } | ||
|  | 
 | ||
|  |         bool RequiresSessions(ServiceEndpoint endpoint) | ||
|  |         { | ||
|  |             return endpoint.Contract.SessionMode == SessionMode.Required; | ||
|  |         } | ||
|  | 
 | ||
|  |         void ValidateScopeRequiredAndAutoComplete(OperationDescription operation, | ||
|  |                                                   bool singleThreaded, | ||
|  |                                                   string contractName) | ||
|  |         { | ||
|  |             OperationBehaviorAttribute attribute = operation.Behaviors.Find<OperationBehaviorAttribute>(); | ||
|  | 
 | ||
|  |             if (attribute != null) | ||
|  |             { | ||
|  |                 if (!singleThreaded && !attribute.TransactionAutoComplete) | ||
|  |                 { | ||
|  |                     string id = SR.SFxTransactionNonConcurrentOrAutoComplete2; | ||
|  |                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException( | ||
|  |                         SR.GetString(id, contractName, operation.Name))); | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         void ValidateNoMSMQandTransactionFlow(ServiceEndpoint endpoint) | ||
|  |         { | ||
|  |             BindingElementCollection bindingElements = endpoint.Binding.CreateBindingElements(); | ||
|  | 
 | ||
|  |             if (bindingElements.Find<TransactionFlowBindingElement>() != null && | ||
|  |                 bindingElements.Find<MsmqTransportBindingElement>() != null) | ||
|  |             { | ||
|  |                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException( | ||
|  |                         SR.GetString(SR.SFxTransactionFlowAndMSMQ, endpoint.Address.Uri.AbsoluteUri))); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         void ValidateNotConcurrentWhenReleaseServiceInstanceOnTxComplete(ServiceDescription service) | ||
|  |         { | ||
|  |             ServiceBehaviorAttribute attribute = service.Behaviors.Find<ServiceBehaviorAttribute>(); | ||
|  | 
 | ||
|  |             if (attribute != null && HasTransactedOperations(service)) | ||
|  |             { | ||
|  |                 if (attribute.ReleaseServiceInstanceOnTransactionComplete && !IsSingleThreaded(service)) | ||
|  |                 { | ||
|  |                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperError( | ||
|  |                         new InvalidOperationException(SR.GetString( | ||
|  |                             SR.SFxTransactionNonConcurrentOrReleaseServiceInstanceOnTxComplete, service.Name))); | ||
|  |                 } | ||
|  | 
 | ||
|  |             } | ||
|  |         } | ||
|  |     } | ||
|  | } |