//----------------------------------------------------------------------------- // Copyright (c) Microsoft Corporation. All rights reserved. //----------------------------------------------------------------------------- namespace System.ServiceModel.Dispatcher { using System; using System.Collections.Generic; using System.ComponentModel; using System.Collections.ObjectModel; using System.Diagnostics.CodeAnalysis; using System.Net; using System.Runtime; using System.ServiceModel.Channels; using System.ServiceModel.Security; using System.Xml; [SuppressMessage("Microsoft.Naming", "CA1704:IdentifiersShouldBeSpelledCorrectly", MessageId = "Compat", Justification = "Compat is an accepted abbreviation")] [EditorBrowsable(EditorBrowsableState.Never)] public class ClientRuntimeCompatBase { internal ClientRuntimeCompatBase() { } [EditorBrowsable(EditorBrowsableState.Never)] [Obsolete("This API supports the .NET Framework infrastructure and is not intended to be used directly from your code.", true)] public IList MessageInspectors { get { return this.messageInspectors; } } [EditorBrowsable(EditorBrowsableState.Never)] [Obsolete("This API supports the .NET Framework infrastructure and is not intended to be used directly from your code.", true)] public KeyedCollection Operations { get { return this.compatOperations; } } internal SynchronizedCollection messageInspectors; internal SynchronizedKeyedCollection operations; internal KeyedCollection compatOperations; } public sealed class ClientRuntime : ClientRuntimeCompatBase { bool addTransactionFlowProperties = true; Type callbackProxyType; ProxyBehaviorCollection channelInitializers; string contractName; string contractNamespace; Type contractProxyType; DispatchRuntime dispatchRuntime; IdentityVerifier identityVerifier; ProxyBehaviorCollection interactiveChannelInitializers; IClientOperationSelector operationSelector; ImmutableClientRuntime runtime; ClientOperation unhandled; bool useSynchronizationContext = true; Uri via; SharedRuntimeState shared; int maxFaultSize; bool messageVersionNoneFaultsEnabled; internal ClientRuntime(DispatchRuntime dispatchRuntime, SharedRuntimeState shared) : this(dispatchRuntime.EndpointDispatcher.ContractName, dispatchRuntime.EndpointDispatcher.ContractNamespace, shared) { if (dispatchRuntime == null) throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("dispatchRuntime"); this.dispatchRuntime = dispatchRuntime; this.shared = shared; Fx.Assert(shared.IsOnServer, "Server constructor called on client?"); } internal ClientRuntime(string contractName, string contractNamespace) : this(contractName, contractNamespace, new SharedRuntimeState(false)) { Fx.Assert(!shared.IsOnServer, "Client constructor called on server?"); } ClientRuntime(string contractName, string contractNamespace, SharedRuntimeState shared) { this.contractName = contractName; this.contractNamespace = contractNamespace; this.shared = shared; OperationCollection operations = new OperationCollection(this); this.operations = operations; this.compatOperations = new OperationCollectionWrapper(operations); this.channelInitializers = new ProxyBehaviorCollection(this); this.messageInspectors = new ProxyBehaviorCollection(this); this.interactiveChannelInitializers = new ProxyBehaviorCollection(this); this.unhandled = new ClientOperation(this, "*", MessageHeaders.WildcardAction, MessageHeaders.WildcardAction); this.unhandled.InternalFormatter = new MessageOperationFormatter(); this.maxFaultSize = TransportDefaults.MaxFaultSize; } internal bool AddTransactionFlowProperties { get { return this.addTransactionFlowProperties; } set { lock (this.ThisLock) { this.InvalidateRuntime(); this.addTransactionFlowProperties = value; } } } public Type CallbackClientType { get { return this.callbackProxyType; } set { lock (this.ThisLock) { this.InvalidateRuntime(); this.callbackProxyType = value; } } } public SynchronizedCollection ChannelInitializers { get { return this.channelInitializers; } } public string ContractName { get { return this.contractName; } } public string ContractNamespace { get { return this.contractNamespace; } } public Type ContractClientType { get { return this.contractProxyType; } set { lock (this.ThisLock) { this.InvalidateRuntime(); this.contractProxyType = value; } } } internal IdentityVerifier IdentityVerifier { get { if (this.identityVerifier == null) { this.identityVerifier = IdentityVerifier.CreateDefault(); } return this.identityVerifier; } set { if (value == null) { throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("value"); } this.InvalidateRuntime(); this.identityVerifier = value; } } public Uri Via { get { return this.via; } set { lock (this.ThisLock) { this.InvalidateRuntime(); this.via = value; } } } public bool ValidateMustUnderstand { get { return this.shared.ValidateMustUnderstand; } set { lock (this.ThisLock) { this.InvalidateRuntime(); this.shared.ValidateMustUnderstand = value; } } } public bool MessageVersionNoneFaultsEnabled { get { return this.messageVersionNoneFaultsEnabled; } set { this.InvalidateRuntime(); this.messageVersionNoneFaultsEnabled = value; } } internal DispatchRuntime DispatchRuntime { get { return this.dispatchRuntime; } } public DispatchRuntime CallbackDispatchRuntime { get { if (this.dispatchRuntime == null) this.dispatchRuntime = new DispatchRuntime(this, this.shared); return this.dispatchRuntime; } } internal bool EnableFaults { get { if (this.IsOnServer) { return this.dispatchRuntime.EnableFaults; } else { return this.shared.EnableFaults; } } set { lock (this.ThisLock) { if (this.IsOnServer) { string text = SR.GetString(SR.SFxSetEnableFaultsOnChannelDispatcher0); throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(text)); } else { this.InvalidateRuntime(); this.shared.EnableFaults = value; } } } } public SynchronizedCollection InteractiveChannelInitializers { get { return this.interactiveChannelInitializers; } } public int MaxFaultSize { get { return this.maxFaultSize; } set { this.InvalidateRuntime(); this.maxFaultSize = value; } } internal bool IsOnServer { get { return this.shared.IsOnServer; } } public bool ManualAddressing { get { if (this.IsOnServer) { return this.dispatchRuntime.ManualAddressing; } else { return this.shared.ManualAddressing; } } set { lock (this.ThisLock) { if (this.IsOnServer) { string text = SR.GetString(SR.SFxSetManualAddresssingOnChannelDispatcher0); throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(text)); } else { this.InvalidateRuntime(); this.shared.ManualAddressing = value; } } } } internal int MaxParameterInspectors { get { lock (this.ThisLock) { int max = 0; for (int i = 0; i < this.operations.Count; i++) max = System.Math.Max(max, this.operations[i].ParameterInspectors.Count); return max; } } } public ICollection ClientMessageInspectors { get { return this.MessageInspectors; } } [EditorBrowsable(EditorBrowsableState.Never)] public new SynchronizedCollection MessageInspectors { get { return this.messageInspectors; } } public ICollection ClientOperations { get { return this.Operations; } } [EditorBrowsable(EditorBrowsableState.Never)] public new SynchronizedKeyedCollection Operations { get { return this.operations; } } public IClientOperationSelector OperationSelector { get { return this.operationSelector; } set { lock (this.ThisLock) { this.InvalidateRuntime(); this.operationSelector = value; } } } internal object ThisLock { get { return this.shared; } } public ClientOperation UnhandledClientOperation { get { return this.unhandled; } } internal bool UseSynchronizationContext { get { return this.useSynchronizationContext; } set { lock (this.ThisLock) { this.InvalidateRuntime(); this.useSynchronizationContext = value; } } } internal T[] GetArray(SynchronizedCollection collection) { lock (collection.SyncRoot) { if (collection.Count == 0) { return EmptyArray.Instance; } else { T[] array = new T[collection.Count]; collection.CopyTo(array, 0); return array; } } } internal ImmutableClientRuntime GetRuntime() { lock (this.ThisLock) { if (this.runtime == null) this.runtime = new ImmutableClientRuntime(this); return this.runtime; } } internal void InvalidateRuntime() { lock (this.ThisLock) { this.shared.ThrowIfImmutable(); this.runtime = null; } } internal void LockDownProperties() { this.shared.LockDownProperties(); } internal SynchronizedCollection NewBehaviorCollection() { return new ProxyBehaviorCollection(this); } internal bool IsFault(ref Message reply) { if (reply == null) { return false; } if (reply.IsFault) { return true; } if (this.MessageVersionNoneFaultsEnabled && IsMessageVersionNoneFault(ref reply, this.MaxFaultSize)) { return true; } return false; } internal static bool IsMessageVersionNoneFault(ref Message message, int maxFaultSize) { if (message.Version != MessageVersion.None || message.IsEmpty) { return false; } HttpResponseMessageProperty prop = message.Properties[HttpResponseMessageProperty.Name] as HttpResponseMessageProperty; if (prop == null || prop.StatusCode != HttpStatusCode.InternalServerError) { return false; } using (MessageBuffer buffer = message.CreateBufferedCopy(maxFaultSize)) { message.Close(); message = buffer.CreateMessage(); using (Message copy = buffer.CreateMessage()) { using (XmlDictionaryReader reader = copy.GetReaderAtBodyContents()) { return reader.IsStartElement(XD.MessageDictionary.Fault, MessageVersion.None.Envelope.DictionaryNamespace); } } } } class ProxyBehaviorCollection : SynchronizedCollection { ClientRuntime outer; internal ProxyBehaviorCollection(ClientRuntime outer) : base(outer.ThisLock) { this.outer = outer; } protected override void ClearItems() { this.outer.InvalidateRuntime(); base.ClearItems(); } protected override void InsertItem(int index, T item) { if (item == null) { throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("item"); } this.outer.InvalidateRuntime(); base.InsertItem(index, item); } protected override void RemoveItem(int index) { this.outer.InvalidateRuntime(); base.RemoveItem(index); } protected override void SetItem(int index, T item) { if (item == null) { throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("item"); } this.outer.InvalidateRuntime(); base.SetItem(index, item); } } class OperationCollection : SynchronizedKeyedCollection { ClientRuntime outer; internal OperationCollection(ClientRuntime outer) : base(outer.ThisLock) { this.outer = outer; } protected override void ClearItems() { this.outer.InvalidateRuntime(); base.ClearItems(); } protected override string GetKeyForItem(ClientOperation item) { return item.Name; } protected override void InsertItem(int index, ClientOperation item) { if (item == null) throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("item"); if (item.Parent != this.outer) throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument(SR.GetString(SR.SFxMismatchedOperationParent)); this.outer.InvalidateRuntime(); base.InsertItem(index, item); } protected override void RemoveItem(int index) { this.outer.InvalidateRuntime(); base.RemoveItem(index); } protected override void SetItem(int index, ClientOperation item) { if (item == null) throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("item"); if (item.Parent != this.outer) throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument(SR.GetString(SR.SFxMismatchedOperationParent)); this.outer.InvalidateRuntime(); base.SetItem(index, item); } internal void InternalClearItems() { this.ClearItems(); } internal string InternalGetKeyForItem(ClientOperation item) { return this.GetKeyForItem(item); } internal void InternalInsertItem(int index, ClientOperation item) { this.InsertItem(index, item); } internal void InternalRemoveItem(int index) { this.RemoveItem(index); } internal void InternalSetItem(int index, ClientOperation item) { this.SetItem(index, item); } } class OperationCollectionWrapper : KeyedCollection { OperationCollection inner; internal OperationCollectionWrapper(OperationCollection inner) { this.inner = inner; } protected override void ClearItems() { this.inner.InternalClearItems(); } protected override string GetKeyForItem(ClientOperation item) { return this.inner.InternalGetKeyForItem(item); } protected override void InsertItem(int index, ClientOperation item) { this.inner.InternalInsertItem(index, item); } protected override void RemoveItem(int index) { this.inner.InternalRemoveItem(index); } protected override void SetItem(int index, ClientOperation item) { this.inner.InternalSetItem(index, item); } } } }