PeerHelpers.cs source code in C# .NET

Source code for the .NET framework in C#

                        

Code:

/ WCF / WCF / 3.5.30729.1 / untmp / Orcas / SP / ndp / cdf / src / WCF / ServiceModel / System / ServiceModel / Channels / PeerHelpers.cs / 1 / PeerHelpers.cs

                            //------------------------------------------------------------ 
// Copyright (c) Microsoft Corporation.  All rights reserved.
//-----------------------------------------------------------
namespace System.ServiceModel.Channels
{ 
    using System.Collections.Generic;
    using System.Collections.ObjectModel; 
    using System.ComponentModel; 
    using System.Diagnostics;
    using System.Globalization; 
    using System.Net;
    using System.Net.Sockets;
    using System.Reflection;
    using System.Runtime.InteropServices; 
    using System.Runtime.Serialization;
    using System.ServiceModel; 
    using System.ServiceModel.Description; 
    using System.ServiceModel.Diagnostics;
    using System.ServiceModel.Dispatcher; 
    using System.ServiceModel.Security;
    using System.Text;
    using System.Xml;
    using System.ServiceModel.PeerResolvers; 
    using System.Threading;
 
    class PeerQuotaHelper 
    {
        int enqueuedCount = 0; 
        int quota = 64;
        AutoResetEvent waiter = new AutoResetEvent(false);

        public PeerQuotaHelper(int limit) 
        {
            this.quota = limit; 
        } 

        public void    ReadyToEnqueueItem() 
        {
            int value = Interlocked.Increment(ref enqueuedCount);
            if (value > quota)
            { 
                waiter.WaitOne();
            } 
        } 

        public void ItemDequeued() 
        {
            int value = Interlocked.Decrement(ref enqueuedCount);
            DiagnosticUtility.DebugAssert(value >= 0, "queue below empty");
            if (value >= quota) 
            {
                waiter.Set(); 
            } 
        }
 
    }

    class PeerNodeConfig
    { 
        int connectTimeout;
        MessageEncoder encoder; 
        PeerNodeAddress listenAddress;  // EPR + IP addresses 
        IPAddress listenIPAddress;
        Uri listenUri; 
        long maxReceivedMessageSize;
        int minNeighbors;                                             //Neighbor parameters
        int idealNeighbors;
        int maxNeighbors; 
        int maxReferrals;
        int maxReferralCacheSize; 
        int maxResolveAddresses; 
        string meshId;
        PeerMessagePropagationFilter messagePropagationFilter; 
        ulong nodeId;
        int port;
        PeerResolver resolver;
        int maintainerInterval; 
        TimeSpan maintainerRetryInterval;
        TimeSpan maintainerTimeout; 
        TimeSpan unregisterTimeout; 
        PeerSecurityManager securityManager;
        int maxIncomingConcurrentCalls=128; 
        int maxConcurrentSessions=64;
        XmlDictionaryReaderQuotas readerQuotas = new XmlDictionaryReaderQuotas();
        long maxBufferPoolSize;
        int maxSendQueueSize=128; 

        public PeerNodeConfig(string meshId, ulong nodeId, 
            PeerResolver resolver, 
            PeerMessagePropagationFilter messagePropagationFilter,
            MessageEncoder encoder, 
            Uri listenUri, IPAddress listenIPAddress, int port,
            long maxReceivedMessageSize,
            int minNeighbors, int idealNeighbors, int maxNeighbors,
            int maxReferrals, 
            int connectTimeout,
            int maintainerInterval, 
            PeerSecurityManager securityManager, 
            XmlDictionaryReaderQuotas readerQuotas,
            long maxBufferPool, 
            int maxSendQueueSize,
            int maxReceiveQueueSize)
        {
            this.connectTimeout = connectTimeout; 
            this.listenIPAddress = listenIPAddress;
            this.listenUri = listenUri; 
            this.maxReceivedMessageSize = maxReceivedMessageSize; 
            this.minNeighbors = minNeighbors;
            this.idealNeighbors = idealNeighbors; 
            this.maxNeighbors = maxNeighbors;
            this.maxReferrals = maxReferrals;
            this.maxReferralCacheSize = PeerTransportConstants.MaxReferralCacheSize;
            this.maxResolveAddresses = PeerTransportConstants.MaxResolveAddresses; 
            this.meshId = meshId;
            this.encoder = encoder; 
            this.messagePropagationFilter = messagePropagationFilter; 
            this.nodeId = nodeId;
            this.port = port; 
            this.resolver = resolver;
            this.maintainerInterval = maintainerInterval;
            this.maintainerRetryInterval = new TimeSpan(PeerTransportConstants.MaintainerRetryInterval * 10000);
            this.maintainerTimeout = new TimeSpan(PeerTransportConstants.MaintainerTimeout * 10000); 
            this.unregisterTimeout = new TimeSpan(PeerTransportConstants.UnregisterTimeout * 10000);
            this.securityManager = securityManager; 
            readerQuotas.CopyTo(this.readerQuotas); 
            this.maxBufferPoolSize = maxBufferPool;
            this.maxIncomingConcurrentCalls = maxReceiveQueueSize; 
            this.maxSendQueueSize = maxSendQueueSize;
        }

        internal PeerSecurityManager SecurityManager 
        {
            get { return securityManager; } 
        } 

        public int ConnectTimeout 
        {
            get { return connectTimeout; }
        }
 
        public IPAddress ListenIPAddress
        { 
            get { return listenIPAddress; } 
        }
 
        public int ListenerPort
        {
            get { return listenAddress.EndpointAddress.Uri.Port; }
        } 

        public Uri ListenUri 
        { 
            get { return listenUri; }
        } 

        public int IdealNeighbors
        {
            get { return idealNeighbors; } 
        }
 
        public int MaintainerInterval 
        {
            get { return maintainerInterval; } 
        }

        public TimeSpan MaintainerRetryInterval
        { 
            get { return maintainerRetryInterval; }
        } 
 
        public TimeSpan MaintainerTimeout
        { 
            get { return maintainerTimeout; }
        }

        public long MaxBufferPoolSize 
        {
            get { return maxBufferPoolSize; } 
        } 

        public long MaxReceivedMessageSize 
        {
            get { return maxReceivedMessageSize; }
        }
 
        public int MaxNeighbors
        { 
            get { return maxNeighbors; } 
        }
 
        public int MaxReferrals
        {
            get { return maxReferrals; }
        } 

        public int MaxReferralCacheSize 
        { 
            get { return maxReferralCacheSize; }
        } 

        public int MaxResolveAddresses
        {
            get { return maxResolveAddresses; } 
        }
 
        public int MaxPendingIncomingCalls 
        {
            get { return this.maxIncomingConcurrentCalls;} 
        }

        public int MaxPendingOutgoingCalls
        { 
            get { return this.maxSendQueueSize;}
        } 
 
        public int MaxConcurrentSessions
        { 
            get { return this.maxConcurrentSessions;}
        }
        public int MinNeighbors
        { 
            get { return minNeighbors; }
        } 
 
        public string MeshId
        { 
            get { return meshId; }
        }

        public MessageEncoder MessageEncoder 
        {
            get { return encoder; } 
        } 

        public PeerMessagePropagationFilter MessagePropagationFilter 
        {
            get { return messagePropagationFilter; }
        }
 
        public ulong NodeId
        { 
            get { return nodeId; } 
        }
 
        public int Port
        {
            get { return port; }
        } 

        public XmlDictionaryReaderQuotas ReaderQuotas 
        { 
            get { return this.readerQuotas; }
        } 

        public PeerResolver Resolver
        {
            get { return resolver; } 
        }
 
        public TimeSpan UnregisterTimeout 
        {
            get { return unregisterTimeout; } 
        }

        // Returns the actual address that the node service is listening on.
        // If retrieving the address in order to send it over the wire, maskScopeId should be true 
        // (scope IDs are not sent over the wire).
        public PeerNodeAddress GetListenAddress(bool maskScopeId) 
        { 
            PeerNodeAddress localAddress = this.listenAddress;
            return new PeerNodeAddress(localAddress.EndpointAddress, PeerIPHelper.CloneAddresses(localAddress.IPAddresses, maskScopeId)); 
        }

        public void SetListenAddress(PeerNodeAddress address)
        { 
            this.listenAddress = address;
        } 
 
        static Uri BuildUri(string host, int port, Guid guid)
        { 
            UriBuilder uriBuilder = new UriBuilder();
            uriBuilder.Host = host;
            if (port > 0)
                uriBuilder.Port = port; 
            uriBuilder.Path = PeerStrings.KnownServiceUriPrefix + '/' + guid;
            uriBuilder.Scheme = Uri.UriSchemeNetTcp; 
            TcpChannelListener.FixIpv6Hostname(uriBuilder, uriBuilder.Uri); 
            return uriBuilder.Uri;
        } 

        public Uri GetSelfUri()
        {
            Uri uri = null; 
            Guid serviceGuid = Guid.NewGuid();
 
            if (this.listenIPAddress == null) 
            {
                uri = BuildUri(DnsCache.MachineName, port, serviceGuid); 
            }
            else
            {
                uri = BuildUri(this.listenIPAddress.ToString(), port, serviceGuid); 
            }
            return uri; 
        } 

        public Uri GetMeshUri() 
        {
            UriBuilder uriBuilder = new UriBuilder();
            uriBuilder.Host = this.meshId;
            uriBuilder.Scheme = PeerStrings.Scheme; 
            return uriBuilder.Uri;
        } 
    } 

    static class PeerMessageHelpers 
    {
        // delegates used to call the callback to process an incoming message
        public delegate void CleanupCallback(IPeerNeighbor neighbor, PeerCloseReason reason, Exception exception);
 
        public static string GetHeaderString(MessageHeaders headers, string name, string ns)
        { 
            string result = null; 
            int index = headers.FindHeader(name, ns);
            if(index >= 0) 
            {
                using(XmlDictionaryReader reader = headers.GetReaderAtHeader(index))
                {
                    result = reader.ReadElementString(); 
                }
            } 
            return result; 
        }
 
        public static Uri GetHeaderUri(MessageHeaders headers, string name, string ns)
        {
            Uri result = null;
            string rawString = GetHeaderString(headers, name, ns); 
            if(rawString != null)
                result = new Uri(rawString); 
            return result; 
        }
 
        public static ulong GetHeaderULong(MessageHeaders headers, int index)
        {
            ulong result = PeerTransportConstants.MaxHopCount;
            if(index >= 0) 
            {
                using(XmlDictionaryReader reader = headers.GetReaderAtHeader(index)) 
                { 
                    result = XmlConvert.ToUInt64(reader.ReadElementString());
                } 
            }
            return result;
        }
 
        public static UniqueId GetHeaderUniqueId(MessageHeaders headers, string name, string ns)
        { 
            UniqueId result = null; 
            int index = headers.FindHeader(name, ns);
            if(index >= 0) 
            {
                using(XmlDictionaryReader reader = headers.GetReaderAtHeader(index))
                {
                    result = reader.ReadElementContentAsUniqueId(); 
                }
            } 
            return result; 
        }
 
    }

    static class PeerStrings
    { 
        public static Dictionary protocolActions;
 
        // Namespace for infrastructure messages 
        public const string Namespace = "http://schemas.microsoft.com/net/2006/05/peer";
 
        // PeerService contract name
        public const string ServiceContractName = "PeerService";

        // Infrastructure message actions 
        // PeerConnector
        public const string ConnectAction  = Namespace + "/Connect"; 
        public const string WelcomeAction = Namespace + "/Welcome"; 
        public const string RefuseAction = Namespace + "/Refuse";
        public const string DisconnectAction = Namespace + "/Disconnect"; 

        // PeerFlooder
        public const string FloodAction = Namespace + "/Flood";
        public const string InternalFloodAction = Namespace + "/IntFlood"; 
        public const string LinkUtilityAction = Namespace + "/LinkUtility";
        public const string RequestSecurityTokenAction = "RequestSecurityToken"; 
        public const string RequestSecurityTokenResponseAction = "RequestSecurityTokenResponse"; 
        public const string HopCountElementName = "Hops";
        public const string HopCountElementNamespace = Namespace + "/HopCount"; 
        public const string PingAction = Namespace+ "/Ping";

        // Uri
        public const string Scheme = "net.p2p"; 
        public const string KnownServiceUriPrefix = "PeerChannelEndpoints";
        public const string PeerCustomResolver = "PeerCustomResolver"; 
 
        public const string SkipLocalChannels="SkipLocalChannels";
        public const string Via = "PeerVia"; 
        public const string MessageVerified = "MessageVerified";
        public const string CacheMiss = "CacheMiss";
        public const string PeerProperty = "PeerProperty";
        public const string MessageId = "MessageID"; 

        static PeerStrings() 
        { 
            protocolActions = new Dictionary();
            PopulateProtocolActions(); 
        }

        static void PopulateProtocolActions()
        { 
            protocolActions.Add(PeerStrings.ConnectAction, PeerOperationNames.Connect);
            protocolActions.Add(PeerStrings.WelcomeAction, PeerOperationNames.Welcome); 
            protocolActions.Add(PeerStrings.RefuseAction, PeerOperationNames.Refuse); 
            protocolActions.Add(PeerStrings.DisconnectAction, PeerOperationNames.Disconnect);
            protocolActions.Add(PeerStrings.RequestSecurityTokenAction, PeerOperationNames.ProcessRequestSecurityToken); 
            protocolActions.Add(PeerStrings.RequestSecurityTokenResponseAction, PeerOperationNames.RequestSecurityTokenResponse);
            protocolActions.Add(PeerStrings.LinkUtilityAction, PeerOperationNames.LinkUtility);
            protocolActions.Add(Addressing10Strings.FaultAction, PeerOperationNames.Fault);
            protocolActions.Add(PeerStrings.PingAction, PeerOperationNames.Ping); 
        }
 
        public static string FindAction(string action) 
        {
            string result = null; 
            protocolActions.TryGetValue(action, out result);
            return result;
        }
 

    } 
 
    class PeerOperationNames
    { 
        public const string Connect = "Connect";
        public const string Disconnect = "Disconnect";
        public const string Refuse = "Refuse";
        public const string Welcome = "Welcome"; 
        public const string LinkUtility = "LinkUtility";
        public const string ProcessRequestSecurityToken = "ProcessRequestSecurityToken"; 
        public const string RequestSecurityTokenResponse = "RequestSecurityTokenResponse"; 
        public const string Flood = "FloodMessage";
        public const string Demuxer = "PeerFlooder"; 
        public const string PeerVia = "PeerVia";
        public const string Fault = "Fault";
        public const string PeerTo = "PeerTo";
        public const string Ping = "Ping"; 

    } 
 
    class PeerResolverStrings
    { 
        public const string Namespace=PeerStrings.Namespace+"/resolver";
        public const string RegisterAction = Namespace+"/Register";
        public const string RegisterResponseAction = Namespace+"/RegisterResponse";
        public const string UnregisterAction = Namespace+"/Unregister"; 
        public const string ResolveAction = Namespace+"/Resolve";
        public const string ResolveResponseAction = Namespace+"/ResolveResponse"; 
        public const string UpdateAction = Namespace+"/Update"; 
        public const string UpdateResponseAction = Namespace+"/UpdateResponse";
        public const string RefreshAction = Namespace+"/Refresh"; 
        public const string RefreshResponseAction = Namespace+"/RefreshResponse";
        public const string GetServiceSettingsAction = Namespace+"/GetServiceSettings";
        public const string GetServiceSettingsResponseAction = Namespace+"/GetServiceSettingsResponse";
    } 

    // constants used by more than one component 
    static class PeerTransportConstants 
    {
        public const int ConnectTimeout = 60 * 1000;            // 1 minute 
        public const ulong InvalidNodeId = 0;
        public const int MinNeighbors = 2;
        public const int IdealNeighbors = 3;
        public const int MaxResolveAddresses = IdealNeighbors;  // We only to resolve Ideal connections 

        public const int MaxNeighbors = 7; 
        public const int MaxReferrals = 10; 
        public const int MaxReferralCacheSize = 50;             // Cache no more than 50 referrals
 
        public const int MaintainerInterval = 5 * 60 * 1000;    // 5 Minutes
        public const int MaintainerRetryInterval = 10000;       // 10 seconds
        public const int MaintainerTimeout = 2 * 60 * 1000;     // 2 Minutes
        public const int UnregisterTimeout = 2 * 60 * 1000;     // 2 Minutes 

        //how long do we want wait before sending each batch of acks? 
        public const int AckTimeout = 30*1000; 

        //how many acks do we want max in each ack message? 
        public const uint AckWindow = 32;

        public const long MinMessageSize = 16384;
 
        public const int MinPort = IPEndPoint.MinPort;
        public const int MaxPort = IPEndPoint.MaxPort; 
        public const ulong MaxHopCount = ulong.MaxValue; 
        public static TimeSpan ForwardInterval = TimeSpan.FromSeconds(10);
        public static TimeSpan ForwardTimeout = TimeSpan.FromSeconds(60); 
        public static int MaxOutgoingMessages = 128;
        public const int MessageThreshold = 32;
    }
 
    static class PeerValidateHelper
    { 
        public static void ValidateListenIPAddress(IPAddress address) 
        {
            // null is Okay 
            if (address==null)
                return;

            // If an incorrect IP address is given throw. 
            if (address.Equals(IPAddress.Any) ||
                address.Equals(IPAddress.IPv6Any) || 
                address.Equals(IPAddress.IPv6None) || 
                address.Equals(IPAddress.None) ||
                address.Equals(IPAddress.Broadcast) || 
                address.IsIPv6Multicast ||
                IPAddress.IsLoopback(address))
            {
                throw DiagnosticUtility.ExceptionUtility.ThrowHelperError( 
                    new ArgumentException(SR.GetString(SR.PeerListenIPAddressInvalid, address), "address", null));
            } 
        } 

        public static void ValidateMaxMessageSize(long value) 
        {
            if (value < PeerTransportConstants.MinMessageSize)
            {
                string message = SR.GetString(SR.ArgumentOutOfRange, PeerTransportConstants.MinMessageSize, long.MaxValue); 
                throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentOutOfRangeException("value", value, message));
            } 
        } 

        public static void ValidatePort(int value) 
        {
            if (value < PeerTransportConstants.MinPort || value > PeerTransportConstants.MaxPort)
            {
                string message = SR.GetString(SR.ArgumentOutOfRange, PeerTransportConstants.MinPort, PeerTransportConstants.MaxPort); 
                throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentOutOfRangeException("value", value, message));
            } 
        } 

        public static bool ValidNodeAddress(PeerNodeAddress address) 
        {
            return (address != null
                && address.EndpointAddress != null
                && address.EndpointAddress.Uri != null 
                && address.IPAddresses != null
                && address.IPAddresses.Count > 0 
                && string.Compare(address.EndpointAddress.Uri.Scheme, Uri.UriSchemeNetTcp, StringComparison.OrdinalIgnoreCase) == 0 
                );
        } 

        public static bool ValidReferralNodeAddress(PeerNodeAddress address)
        {
            bool valid = true; 
            long scopeId = -1;
 
            foreach (IPAddress addr in address.IPAddresses) 
            {
                if (addr.IsIPv6LinkLocal) 
                {
                    if (scopeId == -1)
                    {
                        scopeId = addr.ScopeId; 
                    }
                    else if (scopeId != addr.ScopeId) 
                    { 
                        valid = false;
                        break; 
                    }
                }
            }
            return valid; 
        }
    } 
 
    // Indicates which side initiated neighbor close
    enum PeerCloseInitiator 
    {
        LocalNode,
        RemoteNode
    } 

    // Reason for closing a neighbor 
    // WARNING: If you add another enum value, consider whether a corresponding value 
    // should be added to PeerConnector.RefuseReason and PeerConnector.DisconnectReason.
    // WARNING: Upon adding a new enum value, consider if PeerCloseReasonHelper.IsDefined 
    // should be updated.
    enum PeerCloseReason
    {
        None = 0,               // Reserved value - never serialized used internally. 
        InvalidNeighbor,        // used when protocol violations are detected
        LeavingMesh,            // Closing because the node is leaving the mesh 
        NotUsefulNeighbor,      // Closing because the neighbor is not useful 
        DuplicateNeighbor,      // The node already has a neighbor session to this node
        DuplicateNodeId,        // The neighbor has the same node ID as the local node 
        NodeBusy,               // The node has too many neighbor sessions to accept a new session
        ConnectTimedOut,        // Connect processing timedout
        Faulted,                // When neighbor faults
        Closed,                 // When neighbor closes without Disconnect or Refuse 
        InternalFailure,        // Eg: an infrastructure msg send fails and requires closing the channel
        AuthenticationFailure,  // Eg, when in secure mode, wrong credentials. 
        NodeTooSlow,            //remote neighbor is too slow 
    }
 
    // Neighbor event args
    // EventArgs when a neighbor is closing or closed
    class PeerNeighborCloseEventArgs : EventArgs
    { 
        PeerCloseInitiator closeInitiator;
        Exception exception; 
        PeerCloseReason reason; 

        public PeerNeighborCloseEventArgs(PeerCloseReason reason, 
            PeerCloseInitiator closeInitiator, Exception exception)
        {
            this.reason = reason;
            this.closeInitiator = closeInitiator; 
            this.exception = exception;
        } 
 
        public PeerCloseInitiator CloseInitiator
        { 
            get { return this.closeInitiator; }
        }

        public Exception Exception 
        {
            get { return this.exception; } 
        } 

        public PeerCloseReason Reason 
        {
            get { return this.reason; }
        }
    } 

    class PeerExceptionHelper 
    { 
        static internal void ThrowInvalidOperation_InsufficientCryptoSupport(Exception innerException)
        { 
            throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.InsufficientCryptoSupport), innerException));
        }
        static internal void ThrowArgument_InsufficientCredentials(string property)
        { 
            throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentException(SR.GetString(SR.InsufficientCredentials, property)));
        } 
        static internal void ThrowArgumentOutOfRange_InvalidTransportCredentialType(int value) 
        {
            throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentOutOfRangeException("CredentialType", value, 
                SR.GetString(SR.ValueMustBeInRange, PeerTransportCredentialType.Password, PeerTransportCredentialType.Certificate)));
        }

        static internal void ThrowArgumentOutOfRange_InvalidSecurityMode(int value) 
        {
            throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentOutOfRangeException("Mode", value, 
                SR.GetString(SR.ValueMustBeInRange, SecurityMode.None, SecurityMode.TransportWithMessageCredential))); 
        }
 
        static internal void ThrowInvalidOperation_UnexpectedSecurityTokensDuringHandshake()
        {
            throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.UnexpectedSecurityTokensDuringHandshake)));
        } 

        static internal void ThrowArgument_PnrpAddressesExceedLimit() 
        { 
            throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentException(SR.GetString(SR.PnrpAddressesExceedLimit)));
        } 
        static internal void ThrowInvalidOperation_PnrpNoClouds()
        {
            throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.PnrpNoClouds)));
        } 
        static internal void ThrowInvalidOperation_PnrpAddressesUnsupported()
        { 
            throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.PnrpAddressesUnsupported))); 
        }
        static internal void ThrowArgument_InsufficientResolverSettings() 
        {
            throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentException(SR.GetString(SR.InsufficientResolverSettings)));
        }
        static internal void ThrowArgument_MustOverrideInitialize() 
        {
            throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentException(SR.GetString(SR.MustOverrideInitialize))); 
        } 
        static internal void ThrowArgument_InvalidResolverMode(PeerResolverMode mode)
        { 
            throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentException(SR.GetString(SR.InvalidResolverMode, mode)));
        }

        static internal void ThrowInvalidOperation_NotValidWhenOpen(string operation) 
        {
            throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.NotValidWhenOpen, operation))); 
        } 

        static internal void ThrowInvalidOperation_NotValidWhenClosed(string operation) 
        {
            throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.NotValidWhenClosed, operation)));
        }
 
        static internal void ThrowInvalidOperation_DuplicatePeerRegistration(string servicepath)
        { 
            throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.DuplicatePeerRegistration, servicepath))); 
        }
        static internal void ThrowPnrpError(int errorCode, string cloud) 
        {
            ThrowPnrpError(errorCode, cloud, true);
        }
        static internal void ThrowPnrpError(int errorCode, string cloud, bool trace) 
        {
            throw DiagnosticUtility.ExceptionUtility.ThrowHelper(new System.ServiceModel.Channels.PnrpPeerResolver.PnrpException(errorCode, cloud), trace? TraceEventType.Error : TraceEventType.Information); 
        } 

        static internal void ThrowInvalidOperation_PeerConflictingPeerNodeSettings(string propertyName) 
        {
            throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.PeerConflictingPeerNodeSettings, propertyName)));
        }
 
        static internal void ThrowInvalidOperation_PeerCertGenFailure(Exception innerException)
        { 
            throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.PeerCertGenFailure), innerException)); 
        }
        static internal void ThrowInvalidOperation_ConflictingHeader(string headerName) 
        {
            throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.PeerConflictingHeader, headerName, PeerStrings.Namespace)));
        }
 
        public static Exception GetLastException()
        { 
            return new Win32Exception(Marshal.GetLastWin32Error()); 
        }
 
    }

    class PeerBindingPropertyNames
    { 
        public static readonly string ListenUri = "ListenUri";
        public static readonly string Port = "Port"; 
        public static readonly string MaxReceivedMessageSize = "MaxReceivedMessageSize"; 
        public static readonly string Resolver = "Resolver";
        public static readonly string Security = "Security"; 
        public static readonly string SecurityDotMode = "Security.Mode";
        public static readonly string ListenIPAddress = "ListenIPAddress";
        public static readonly string Credentials = "Credentials";
        public static readonly string ResolverSettings = "ResolverSettings"; 
        public static readonly string Password = "Password";
        public static readonly string Certificate = "Certificate"; 
        public static readonly string MaxBufferPoolSize = "MaxBufferPoolSize"; 
        public static readonly string ReaderQuotasDotArrayLength = "ReaderQuotas.MaxArrayLength";
        public static readonly string ReaderQuotasDotStringLength = "ReaderQuotas.MaxStringContentLength"; 
        public static readonly string ReaderQuotasDotMaxDepth = "ReaderQuotas.MaxDepth";
        public static readonly string ReaderQuotasDotMaxCharCount = "ReaderQuotas.MaxNameTableCharCount";
        public static readonly string ReaderQuotasDotMaxBytesPerRead = "ReaderQuotas.MaxBytesPerRead";
    } 

    class PeerPropertyNames 
    { 
        public static readonly string MessageSenderAuthentication = "Credentials.Peer.MessageSenderAuthentication";
        public static readonly string Credentials = "SecurityCredentialsManager"; 
        public static readonly string Password = "Credentials.Peer.MeshPassword";
        public static readonly string Certificate = "Credentials.Peer.Certificate";
        public static readonly string PeerAuthentication = "Credentials.Peer.PeerAuthentication";
    } 

    class OperationSelector : IDispatchOperationSelector 
    { 
        IPeerNodeMessageHandling messageHandler;
 
        public OperationSelector(IPeerNodeMessageHandling messageHandler)
        {
            this.messageHandler = messageHandler;
        } 

        public static void TurnOffSecurityHeader(Message message) 
        { 
            int i = message.Headers.FindHeader(SecurityJan2004Strings.Security, SecurityJan2004Strings.Namespace);
            if (i >= 0) 
            {
                message.Headers.AddUnderstood(i);
            }
        } 

        public string SelectOperation(ref Message message) 
        { 

            string action = message.Headers.Action; 
            string demux = null;
            byte[] id = PeerNodeImplementation.DefaultId;
            string operation = PeerStrings.FindAction(action);
            Uri via = null; 
            Uri to = null;
            bool skipped = false; 
            PeerMessageProperty peerProperty = new PeerMessageProperty(); 

            if(operation != null) 
                return operation;
            try
            {
                demux = PeerMessageHelpers.GetHeaderString(message.Headers, PeerOperationNames.Flood, PeerStrings.Namespace); 
                via = PeerMessageHelpers.GetHeaderUri(message.Headers, PeerStrings.Via, PeerStrings.Namespace);
                to= PeerMessageHelpers.GetHeaderUri(message.Headers, PeerOperationNames.PeerTo, PeerStrings.Namespace); 
            } 
            catch(MessageHeaderException e)
            { 
                DiagnosticUtility.ExceptionUtility.TraceHandledException(e, TraceEventType.Warning);
                return PeerOperationNames.Fault;
            }
            catch(SerializationException e) 
            {
                DiagnosticUtility.ExceptionUtility.TraceHandledException(e, TraceEventType.Warning); 
                return PeerOperationNames.Fault; 
            }
            catch(XmlException e) 
            {
                DiagnosticUtility.ExceptionUtility.TraceHandledException(e, TraceEventType.Warning);
                return PeerOperationNames.Fault;
            } 
            peerProperty.PeerVia = via;
            peerProperty.PeerTo = to; 
            message.Properties.Add(PeerStrings.PeerProperty, peerProperty); 
            if (demux == PeerOperationNames.Demuxer)
            { 
                try
                {
                    if(!this.messageHandler.ValidateIncomingMessage(ref message, via))
                    { 
                        peerProperty.SkipLocalChannels = true;
                        skipped = true; 
                        TurnOffSecurityHeader(message); 
                    }
                    if (this.messageHandler.IsNotSeenBefore(message, out id, out peerProperty.CacheMiss)) 
                    {
                        peerProperty.MessageVerified = true;
                    }
                    else 
                    {
                        if(!skipped) 
                        { 
                            peerProperty.SkipLocalChannels = true;
                        } 
                    }
                    //means that the message doesnt contain legal id
                    if (id == PeerNodeImplementation.DefaultId)
                        return PeerOperationNames.Fault; 
                }
                catch (MessageHeaderException e) 
                { 
                    DiagnosticUtility.ExceptionUtility.TraceHandledException(e, TraceEventType.Warning);
                    return PeerOperationNames.Fault; 
                }
                catch (SerializationException e)
                {
                    DiagnosticUtility.ExceptionUtility.TraceHandledException(e, TraceEventType.Warning); 
                    return PeerOperationNames.Fault;
                } 
                catch (XmlException e) 
                {
                    DiagnosticUtility.ExceptionUtility.TraceHandledException(e, TraceEventType.Warning); 
                    return PeerOperationNames.Fault;
                }
                catch (MessageSecurityException e)
                { 
                    if(!e.ReplayDetected)
                        return PeerOperationNames.Fault; 
                    else 
                        DiagnosticUtility.ExceptionUtility.TraceHandledException(e, TraceEventType.Information);
                } 
                return PeerOperationNames.Flood;
            }
            else
                return null; 
        }
    } 
 
    internal class PeerOperationSelectorBehavior : IContractBehavior
    { 
        IPeerNodeMessageHandling messageHandler;

        internal PeerOperationSelectorBehavior(IPeerNodeMessageHandling messageHandler)
        { 
            this.messageHandler = messageHandler;
        } 
 
        void IContractBehavior.AddBindingParameters(ContractDescription description, ServiceEndpoint endpoint, BindingParameterCollection parameters)
        { 
        }

        void IContractBehavior.Validate(ContractDescription description, ServiceEndpoint endpoint)
        { 
        }
 
        void IContractBehavior.ApplyDispatchBehavior(ContractDescription description, ServiceEndpoint endpoint, DispatchRuntime 
        dispatch)
        { 
            dispatch.OperationSelector = new OperationSelector(this.messageHandler);

            if (dispatch.ClientRuntime != null)
            { 
                dispatch.ClientRuntime.OperationSelector = new OperationSelectorBehavior.MethodInfoOperationSelector(description, MessageDirection.Output);
            } 
        } 

        void IContractBehavior.ApplyClientBehavior(ContractDescription description, ServiceEndpoint endpoint, ClientRuntime proxy) 
        {
            proxy.OperationSelector = new OperationSelectorBehavior.MethodInfoOperationSelector(description, MessageDirection.Input);
            proxy.CallbackDispatchRuntime.OperationSelector = new OperationSelector(this.messageHandler);
        } 

    } 
 
    class PeerDictionaryHeader : DictionaryHeader
    { 
        string value;
        XmlDictionaryString name;
        XmlDictionaryString nameSpace;
 

        public override XmlDictionaryString DictionaryName 
        { 
            get { return name; }
        } 

        public override XmlDictionaryString DictionaryNamespace
        {
            get { return nameSpace; } 
        }
 
        public PeerDictionaryHeader(XmlDictionaryString name, XmlDictionaryString nameSpace, string value) 
        {
            this.name = name; 
            this.nameSpace = nameSpace;
            this.value = value;
        }
 

 
        public PeerDictionaryHeader(XmlDictionaryString name, XmlDictionaryString nameSpace, XmlDictionaryString value) 
        {
            this.name = name; 
            this.nameSpace = nameSpace;
            this.value = value.Value;
        }
 
        protected override void OnWriteHeaderContents(XmlDictionaryWriter writer, MessageVersion messageVersion)
        { 
            writer.WriteString(this.value); 
        }
 
        static internal PeerDictionaryHeader CreateHopCountHeader(ulong hopcount)
        {
            return new PeerDictionaryHeader(XD.PeerWireStringsDictionary.HopCount, XD.PeerWireStringsDictionary.HopCountNamespace, hopcount.ToString(CultureInfo.InvariantCulture));
        } 

        static internal PeerDictionaryHeader CreateViaHeader(Uri via) 
        { 
            return new PeerDictionaryHeader(XD.PeerWireStringsDictionary.PeerVia, XD.PeerWireStringsDictionary.Namespace, via.ToString());
        } 

        static internal PeerDictionaryHeader CreateFloodRole()
        {
            return new PeerDictionaryHeader(XD.PeerWireStringsDictionary.FloodAction, XD.PeerWireStringsDictionary.Namespace, XD.PeerWireStringsDictionary.Demuxer); 
        }
 
        static internal PeerDictionaryHeader CreateToHeader(Uri to) 
        {
            return new PeerDictionaryHeader(XD.PeerWireStringsDictionary.PeerTo, XD.PeerWireStringsDictionary.Namespace, to.ToString()); 
        }
        static internal PeerDictionaryHeader CreateMessageIdHeader(System.Xml.UniqueId messageId)
        {
            return new PeerDictionaryHeader(XD.AddressingDictionary.MessageId, XD.PeerWireStringsDictionary.Namespace, messageId.ToString()); 
        }
 
    } 

    class PeerMessageProperty 
    {

        public bool MessageVerified;
        public bool SkipLocalChannels; 
        public Uri PeerVia;
        public Uri PeerTo; 
        public int CacheMiss; 
    }
} 


// File provided for Reference Use Only by Microsoft Corporation (c) 2007.
// Copyright (c) Microsoft Corporation. All rights reserved.
                        

Link Menu

Network programming in C#, Network Programming in VB.NET, Network Programming in .NET
This book is available now!
Buy at Amazon US or
Buy at Amazon UK