| 
									
										
										
										
											2016-08-03 10:59:49 +00:00
										 |  |  | //------------------------------------------------------------ | 
					
						
							|  |  |  | // Copyright (c) Microsoft Corporation.  All rights reserved. | 
					
						
							|  |  |  | //------------------------------------------------------------ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace System.ServiceModel.Channels | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     using System.Runtime; | 
					
						
							|  |  |  |     using System.ServiceModel.Dispatcher; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     sealed class InternalDuplexBindingElement : BindingElement | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         InputChannelDemuxer clientChannelDemuxer; | 
					
						
							|  |  |  |         bool providesCorrelation; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         public InternalDuplexBindingElement() | 
					
						
							|  |  |  |             : this(false) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         //  | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         internal InternalDuplexBindingElement(bool providesCorrelation) | 
					
						
							|  |  |  |             : base() | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             this.providesCorrelation = providesCorrelation; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         InternalDuplexBindingElement(InternalDuplexBindingElement elementToBeCloned) | 
					
						
							|  |  |  |             : base(elementToBeCloned) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             this.clientChannelDemuxer = elementToBeCloned.ClientChannelDemuxer; | 
					
						
							|  |  |  |             this.providesCorrelation = elementToBeCloned.ProvidesCorrelation; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         internal InputChannelDemuxer ClientChannelDemuxer | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             get { return this.clientChannelDemuxer; } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         internal bool ProvidesCorrelation | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             get { return this.providesCorrelation; } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         public override BindingElement Clone() | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             return new InternalDuplexBindingElement(this); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         public override IChannelFactory<TChannel> BuildChannelFactory<TChannel>(BindingContext context) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if (context == null) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("context"); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!this.CanBuildChannelFactory<TChannel>(context)) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument( | 
					
						
							|  |  |  |                     "TChannel", SR.GetString(SR.ChannelTypeNotSupported, typeof(TChannel))); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             IChannelFactory<IOutputChannel> innerChannelFactory = context.Clone().BuildInnerChannelFactory<IOutputChannel>(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (this.clientChannelDemuxer == null) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 this.clientChannelDemuxer = new InputChannelDemuxer(context); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |             { | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | #pragma warning suppress 56506 // Microsoft, context.RemainingBindingElements will never be null | 
					
						
							| 
									
										
										
										
											2016-08-03 10:59:49 +00:00
										 |  |  |                 context.RemainingBindingElements.Clear(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             LocalAddressProvider localAddressProvider = context.BindingParameters.Remove<LocalAddressProvider>(); | 
					
						
							|  |  |  |             return (IChannelFactory<TChannel>)(object) | 
					
						
							|  |  |  |                 new InternalDuplexChannelFactory(this, context, this.clientChannelDemuxer, innerChannelFactory, localAddressProvider); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         public override IChannelListener<TChannel> BuildChannelListener<TChannel>(BindingContext context) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if (context == null) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("context"); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (typeof(TChannel) != typeof(IDuplexChannel)) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument("TChannel", | 
					
						
							|  |  |  |                     SR.GetString(SR.ChannelTypeNotSupported, typeof(TChannel))); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return (IChannelListener<TChannel>)(object)new InternalDuplexChannelListener(this, context); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         public override bool CanBuildChannelFactory<TChannel>(BindingContext context) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if (context == null) | 
					
						
							|  |  |  |                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("context"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return typeof(TChannel) == typeof(IDuplexChannel) | 
					
						
							|  |  |  |                 && context.CanBuildInnerChannelFactory<IOutputChannel>() | 
					
						
							|  |  |  |                 && context.CanBuildInnerChannelListener<IInputChannel>(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         public override bool CanBuildChannelListener<TChannel>(BindingContext context) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if (context == null) | 
					
						
							|  |  |  |                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("context"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return typeof(TChannel) == typeof(IDuplexChannel) | 
					
						
							|  |  |  |                 && context.CanBuildInnerChannelFactory<IOutputChannel>() | 
					
						
							|  |  |  |                 && context.CanBuildInnerChannelListener<IInputChannel>(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         internal static T GetSecurityCapabilities<T>(ISecurityCapabilities lowerCapabilities) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             Fx.Assert(typeof(T) == typeof(ISecurityCapabilities), "Can only be used with ISecurityCapabilities"); | 
					
						
							|  |  |  |             if (lowerCapabilities != null) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 // composite duplex cannot ensure that messages it receives are from the part it sends | 
					
						
							|  |  |  |                 // messages to. So it cannot offer server auth | 
					
						
							|  |  |  |                 return (T)(object)(new SecurityCapabilities(lowerCapabilities.SupportsClientAuthentication, | 
					
						
							|  |  |  |                     false, lowerCapabilities.SupportsClientWindowsIdentity, lowerCapabilities.SupportedRequestProtectionLevel, | 
					
						
							|  |  |  |                     System.Net.Security.ProtectionLevel.None)); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 return (T)(object)null; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         public override T GetProperty<T>(BindingContext context) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if (typeof(T) == typeof(ISecurityCapabilities) && !this.ProvidesCorrelation) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 return InternalDuplexBindingElement.GetSecurityCapabilities<T>(context.GetInnerProperty<ISecurityCapabilities>()); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 return context.GetInnerProperty<T>(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         internal override bool IsMatch(BindingElement b) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if (b == null) | 
					
						
							|  |  |  |                 return false; | 
					
						
							|  |  |  |             return (b is InternalDuplexBindingElement); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         public static void AddDuplexFactorySupport(BindingContext context, ref InternalDuplexBindingElement internalDuplexBindingElement) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if (context.CanBuildInnerChannelFactory<IDuplexChannel>()) | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |             if (context.RemainingBindingElements.Find<CompositeDuplexBindingElement>() == null) | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (context.CanBuildInnerChannelFactory<IOutputChannel>() && | 
					
						
							|  |  |  |                 context.CanBuildInnerChannelListener<IInputChannel>()) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 if (context.CanBuildInnerChannelFactory<IRequestChannel>()) | 
					
						
							|  |  |  |                     return; | 
					
						
							|  |  |  |                 if (context.CanBuildInnerChannelFactory<IRequestSessionChannel>()) | 
					
						
							|  |  |  |                     return; | 
					
						
							|  |  |  |                 if (context.CanBuildInnerChannelFactory<IOutputSessionChannel>()) | 
					
						
							|  |  |  |                     return; | 
					
						
							|  |  |  |                 if (context.CanBuildInnerChannelFactory<IDuplexSessionChannel>()) | 
					
						
							|  |  |  |                     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (internalDuplexBindingElement == null) | 
					
						
							|  |  |  |                     internalDuplexBindingElement = new InternalDuplexBindingElement(); | 
					
						
							|  |  |  |                 context.RemainingBindingElements.Insert(0, internalDuplexBindingElement); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         public static void AddDuplexListenerSupport(BindingContext context, ref InternalDuplexBindingElement internalDuplexBindingElement) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if (context.CanBuildInnerChannelListener<IDuplexChannel>()) | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |             if (context.RemainingBindingElements.Find<CompositeDuplexBindingElement>() == null) | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (context.CanBuildInnerChannelFactory<IOutputChannel>() && | 
					
						
							|  |  |  |                 context.CanBuildInnerChannelListener<IInputChannel>()) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 if (context.CanBuildInnerChannelListener<IReplyChannel>()) | 
					
						
							|  |  |  |                     return; | 
					
						
							|  |  |  |                 if (context.CanBuildInnerChannelListener<IReplySessionChannel>()) | 
					
						
							|  |  |  |                     return; | 
					
						
							|  |  |  |                 if (context.CanBuildInnerChannelListener<IInputSessionChannel>()) | 
					
						
							|  |  |  |                     return; | 
					
						
							|  |  |  |                 if (context.CanBuildInnerChannelListener<IDuplexSessionChannel>()) | 
					
						
							|  |  |  |                     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (internalDuplexBindingElement == null) | 
					
						
							|  |  |  |                     internalDuplexBindingElement = new InternalDuplexBindingElement(); | 
					
						
							|  |  |  |                 context.RemainingBindingElements.Insert(0, internalDuplexBindingElement); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         public static void AddDuplexListenerSupport(CustomBinding binding, ref InternalDuplexBindingElement internalDuplexBindingElement) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if (binding.CanBuildChannelListener<IDuplexChannel>()) | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |             if (binding.Elements.Find<CompositeDuplexBindingElement>() == null) | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (binding.CanBuildChannelFactory<IOutputChannel>() && | 
					
						
							|  |  |  |                 binding.CanBuildChannelListener<IInputChannel>()) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 if (binding.CanBuildChannelListener<IReplyChannel>()) | 
					
						
							|  |  |  |                     return; | 
					
						
							|  |  |  |                 if (binding.CanBuildChannelListener<IReplySessionChannel>()) | 
					
						
							|  |  |  |                     return; | 
					
						
							|  |  |  |                 if (binding.CanBuildChannelListener<IInputSessionChannel>()) | 
					
						
							|  |  |  |                     return; | 
					
						
							|  |  |  |                 if (binding.CanBuildChannelListener<IDuplexSessionChannel>()) | 
					
						
							|  |  |  |                     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (internalDuplexBindingElement == null) | 
					
						
							|  |  |  |                     internalDuplexBindingElement = new InternalDuplexBindingElement(); | 
					
						
							|  |  |  |                 binding.Elements.Insert(0, internalDuplexBindingElement); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     class LocalAddressProvider | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         EndpointAddress localAddress; | 
					
						
							|  |  |  |         MessageFilter filter; | 
					
						
							|  |  |  |         int priority; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         public LocalAddressProvider(EndpointAddress localAddress, MessageFilter filter) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if (localAddress == null) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("localAddress"); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (filter == null) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("filter"); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             this.localAddress = localAddress; | 
					
						
							|  |  |  |             this.filter = filter; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (localAddress.Headers.FindHeader(XD.UtilityDictionary.UniqueEndpointHeaderName.Value, | 
					
						
							|  |  |  |                     XD.UtilityDictionary.UniqueEndpointHeaderNamespace.Value) == null) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 this.priority = Int32.MaxValue - 1; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 this.priority = Int32.MaxValue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         public EndpointAddress LocalAddress | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             get { return this.localAddress; } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         public MessageFilter Filter | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             get { return this.filter; } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         public int Priority | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             get { return this.priority; } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |