//---------------------------------------------------------------- // Copyright (c) Microsoft Corporation. All rights reserved. //---------------------------------------------------------------- namespace System.ServiceModel.Discovery { using System; using System.Collections.ObjectModel; using System.Runtime; using System.ServiceModel; using System.ServiceModel.Description; using System.ServiceModel.Dispatcher; using System.Xml; using System.Xml.Linq; using SR2 = System.ServiceModel.Discovery.SR; [Fx.Tag.XamlVisible(false)] public class EndpointDiscoveryMetadata { static XmlQualifiedName metadataContractName; EndpointAddress endpointAddress; OpenableContractTypeNameCollection contractTypeNames; OpenableScopeCollection scopes; OpenableCollection listenUris; OpenableCollection extensions; int metadataVersion; string[] compiledScopes; bool isOpen; public EndpointDiscoveryMetadata() { this.endpointAddress = new EndpointAddress(EndpointAddress.AnonymousUri); } public Collection ContractTypeNames { get { if (this.contractTypeNames == null) { this.contractTypeNames = new OpenableContractTypeNameCollection(this.isOpen); } return this.contractTypeNames; } } public EndpointAddress Address { get { return this.endpointAddress; } set { ThrowIfOpen(); if (value == null) { throw FxTrace.Exception.ArgumentNull("value"); } this.endpointAddress = value; } } public Collection Extensions { get { if (this.extensions == null) { this.extensions = new OpenableCollection(this.isOpen); } return this.extensions; } } public Collection ListenUris { get { if (this.listenUris == null) { this.listenUris = new OpenableCollection(this.isOpen); } return this.listenUris; } } public Collection Scopes { get { if (this.scopes == null) { this.scopes = new OpenableScopeCollection(this.isOpen); } return this.scopes; } } public int Version { get { return this.metadataVersion; } set { ThrowIfOpen(); if (value < 0) { throw FxTrace.Exception.ArgumentOutOfRange("value", value, SR2.DiscoveryMetadataVersionLessThanZero); } this.metadataVersion = value; } } internal static XmlQualifiedName MetadataContractName { get { if (metadataContractName == null) { ContractDescription metadataContract = ContractDescription.GetContract(typeof(IMetadataExchange)); metadataContractName = new XmlQualifiedName(metadataContract.Name, metadataContract.Namespace); } return metadataContractName; } } internal Collection InternalContractTypeNames { get { return this.contractTypeNames; } } internal string[] CompiledScopes { get { Fx.Assert(IsOpen, "The CompiledScopes property is valid only if this EndpointDiscoveryMetadata instance is open."); return this.compiledScopes; } } internal bool IsOpen { get { return this.isOpen; } } public static EndpointDiscoveryMetadata FromServiceEndpoint(ServiceEndpoint endpoint) { if (endpoint == null) { throw FxTrace.Exception.ArgumentNull("endpoint"); } return GetEndpointDiscoveryMetadata(endpoint, endpoint.ListenUri); } public static EndpointDiscoveryMetadata FromServiceEndpoint(ServiceEndpoint endpoint, EndpointDispatcher endpointDispatcher) { if (endpoint == null) { throw FxTrace.Exception.ArgumentNull("endpoint"); } if (endpointDispatcher == null) { throw FxTrace.Exception.ArgumentNull("endpointDispatcher"); } EndpointDiscoveryMetadata endpointDiscoveryMetadata; if ((endpointDispatcher.ChannelDispatcher != null) && (endpointDispatcher.ChannelDispatcher.Listener != null)) { endpointDiscoveryMetadata = GetEndpointDiscoveryMetadata(endpoint, endpointDispatcher.ChannelDispatcher.Listener.Uri); } else { endpointDiscoveryMetadata = GetEndpointDiscoveryMetadata(endpoint, endpoint.ListenUri); } if ((endpointDiscoveryMetadata != null) && IsMetadataEndpoint(endpoint) && CanHaveMetadataEndpoints(endpointDispatcher)) { AddContractTypeScopes(endpointDiscoveryMetadata, endpointDispatcher.ChannelDispatcher.Host.Description); } return endpointDiscoveryMetadata; } static EndpointDiscoveryMetadata GetEndpointDiscoveryMetadata(ServiceEndpoint endpoint, Uri listenUri) { EndpointDiscoveryMetadata endpointDiscoveryMetadata = new EndpointDiscoveryMetadata(); endpointDiscoveryMetadata.Address = endpoint.Address; endpointDiscoveryMetadata.ListenUris.Add(listenUri); EndpointDiscoveryBehavior endpointDiscoveryBehavior = endpoint.Behaviors.Find(); if (endpointDiscoveryBehavior != null) { if (!endpointDiscoveryBehavior.Enabled) { if (TD.EndpointDiscoverabilityDisabledIsEnabled()) { TD.EndpointDiscoverabilityDisabled(endpoint.Address.ToString(), listenUri.ToString()); } return null; } if (TD.EndpointDiscoverabilityEnabledIsEnabled()) { TD.EndpointDiscoverabilityEnabled(endpoint.Address.ToString(), listenUri.ToString()); } if (endpointDiscoveryBehavior.InternalContractTypeNames != null) { foreach (XmlQualifiedName contractTypeName in endpointDiscoveryBehavior.InternalContractTypeNames) { endpointDiscoveryMetadata.ContractTypeNames.Add(contractTypeName); } } if (endpointDiscoveryBehavior.InternalScopes != null) { foreach (Uri scope in endpointDiscoveryBehavior.InternalScopes) { endpointDiscoveryMetadata.Scopes.Add(scope); } } if (endpointDiscoveryBehavior.InternalExtensions != null) { foreach (XElement xElement in endpointDiscoveryBehavior.InternalExtensions) { endpointDiscoveryMetadata.Extensions.Add(xElement); } } } XmlQualifiedName defaultContractTypeName = new XmlQualifiedName(endpoint.Contract.Name, endpoint.Contract.Namespace); if (!endpointDiscoveryMetadata.ContractTypeNames.Contains(defaultContractTypeName)) { endpointDiscoveryMetadata.ContractTypeNames.Add(defaultContractTypeName); } return endpointDiscoveryMetadata; } static void AddContractTypeScopes(EndpointDiscoveryMetadata endpointDiscoveryMetadata, ServiceDescription serviceDescription) { foreach (ServiceEndpoint endpoint in serviceDescription.Endpoints) { if (IsMetadataEndpoint(endpoint) || IsDiscoverySystemEndpoint(endpoint)) { continue; } endpointDiscoveryMetadata.Scopes.Add(FindCriteria.GetContractTypeNameScope( new XmlQualifiedName(endpoint.Contract.Name, endpoint.Contract.Namespace))); } } static bool CanHaveMetadataEndpoints(EndpointDispatcher endpointDispatcher) { if ((endpointDispatcher.ChannelDispatcher == null) || (endpointDispatcher.ChannelDispatcher.Host == null)) { return false; } ServiceDescription description = endpointDispatcher.ChannelDispatcher.Host.Description; if (description.Behaviors != null && description.Behaviors.Find() == null) { return false; } if (description.ServiceType != null && description.ServiceType.GetInterface(typeof(IMetadataExchange).Name) != null) { return false; } return true; } internal static bool IsDiscoverySystemEndpoint(EndpointDispatcher endpointDispatcher) { return (endpointDispatcher.IsSystemEndpoint && IsDiscoveryContract(endpointDispatcher.ContractName, endpointDispatcher.ContractNamespace)); } internal static bool IsDiscoverySystemEndpoint(ServiceEndpoint endpoint) { return (endpoint.IsSystemEndpoint && IsDiscoveryContract(endpoint.Contract.Name, endpoint.Contract.Namespace)); } static bool IsDiscoveryContract(string contractName, string contractNamespace) { return (IsDiscoveryContractName(contractName) && IsDiscoveryContractNamespace(contractNamespace)); } static bool IsDiscoveryContractName(string contractName) { return ((string.CompareOrdinal(contractName, ProtocolStrings.ContractNames.DiscoveryAdhocContractName) == 0) || (string.CompareOrdinal(contractName, ProtocolStrings.ContractNames.DiscoveryManagedContractName) == 0)); } static bool IsDiscoveryContractNamespace(string contractNamespace) { return ((string.CompareOrdinal(contractNamespace, ProtocolStrings.VersionApril2005.Namespace) == 0) || (string.CompareOrdinal(contractNamespace, ProtocolStrings.Version11.Namespace) == 0) || (string.CompareOrdinal(contractNamespace, ProtocolStrings.VersionCD1.Namespace) == 0)); } internal static bool IsMetadataEndpoint(ServiceEndpoint endpoint) { return ((string.CompareOrdinal(endpoint.Contract.Name, MetadataContractName.Name) == 0) && (string.CompareOrdinal(endpoint.Contract.Namespace, MetadataContractName.Namespace) == 0)); } [Fx.Tag.Throws(typeof(XmlException), "throws on incorrect xml data")] internal void ReadFrom(DiscoveryVersion discoveryVersion, XmlReader reader) { ThrowIfOpen(); if (discoveryVersion == null) { throw FxTrace.Exception.ArgumentNull("discoveryVersion"); } if (reader == null) { throw FxTrace.Exception.ArgumentNull("reader"); } this.endpointAddress = new EndpointAddress(EndpointAddress.AnonymousUri); this.contractTypeNames = null; this.scopes = null; this.listenUris = null; this.metadataVersion = 0; this.extensions = null; this.isOpen = false; reader.MoveToContent(); if (reader.IsEmptyElement) { throw FxTrace.Exception.AsError(new XmlException(SR2.DiscoveryXmlEndpointNull)); } int startDepth = reader.Depth; reader.ReadStartElement(); this.endpointAddress = SerializationUtility.ReadEndpointAddress(discoveryVersion, reader); if (reader.IsStartElement(ProtocolStrings.SchemaNames.TypesElement, discoveryVersion.Namespace)) { this.contractTypeNames = new OpenableContractTypeNameCollection(false); SerializationUtility.ReadContractTypeNames(this.contractTypeNames, reader); } if (reader.IsStartElement(ProtocolStrings.SchemaNames.ScopesElement, discoveryVersion.Namespace)) { this.scopes = new OpenableScopeCollection(false); SerializationUtility.ReadScopes(this.scopes, reader); } if (reader.IsStartElement(ProtocolStrings.SchemaNames.XAddrsElement, discoveryVersion.Namespace)) { this.listenUris = new OpenableCollection(false); SerializationUtility.ReadListenUris(listenUris, reader); } if (reader.IsStartElement(ProtocolStrings.SchemaNames.MetadataVersionElement, discoveryVersion.Namespace)) { this.metadataVersion = SerializationUtility.ReadMetadataVersion(reader); } while (true) { reader.MoveToContent(); if ((reader.NodeType == XmlNodeType.EndElement) && (reader.Depth == startDepth)) { break; } else if (reader.IsStartElement()) { this.Extensions.Add(XElement.ReadFrom(reader) as XElement); } else { reader.Read(); } } reader.ReadEndElement(); } internal void WriteTo(DiscoveryVersion discoveryVersion, XmlWriter writer) { if (discoveryVersion == null) { throw FxTrace.Exception.ArgumentNull("discoveryVersion"); } if (writer == null) { throw FxTrace.Exception.ArgumentNull("writer"); } SerializationUtility.WriteEndPointAddress(discoveryVersion, this.endpointAddress, writer); SerializationUtility.WriteContractTypeNames(discoveryVersion, this.contractTypeNames, writer); SerializationUtility.WriteScopes(discoveryVersion, this.scopes, null, writer); SerializationUtility.WriteListenUris(discoveryVersion, this.listenUris, writer); SerializationUtility.WriteMetadataVersion(discoveryVersion, this.metadataVersion, writer); if (this.extensions != null) { foreach (XElement xElement in Extensions) { xElement.WriteTo(writer); } } } internal void Open() { if (this.contractTypeNames != null) { this.contractTypeNames.Open(); } if (this.scopes != null) { this.scopes.Open(); this.compiledScopes = ScopeCompiler.Compile(this.scopes); } if (this.listenUris != null) { this.listenUris.Open(); } if (this.extensions != null) { this.extensions.Open(); } this.isOpen = true; } void ThrowIfOpen() { if (this.isOpen) { throw FxTrace.Exception.AsError(new InvalidOperationException(SR2.DiscoveryMetadataAlreadyOpen)); } } class OpenableCollection : NonNullItemCollection { bool isOpen; public OpenableCollection(bool opened) { this.isOpen = opened; } void ThrowIfOpen() { if (this.isOpen) { throw FxTrace.Exception.AsError(new InvalidOperationException(SR2.DiscoverySdmCollectionIsOpen(typeof(T).Name))); } } internal void Open() { this.isOpen = true; } protected override void ClearItems() { ThrowIfOpen(); base.ClearItems(); } protected override void InsertItem(int index, T item) { ThrowIfOpen(); base.InsertItem(index, item); } protected override void RemoveItem(int index) { ThrowIfOpen(); base.RemoveItem(index); } protected override void SetItem(int index, T item) { ThrowIfOpen(); base.SetItem(index, item); } } class OpenableContractTypeNameCollection : OpenableCollection { public OpenableContractTypeNameCollection(bool opened) : base(opened) { } protected override void InsertItem(int index, XmlQualifiedName item) { if ((item != null) && (item.Name == string.Empty)) { throw FxTrace.Exception.AsError(new ArgumentException(SR2.DiscoveryArgumentEmptyContractTypeName)); } base.InsertItem(index, item); } protected override void SetItem(int index, XmlQualifiedName item) { if ((item != null) && (item.Name == string.Empty)) { throw FxTrace.Exception.AsError(new ArgumentException(SR2.DiscoveryArgumentEmptyContractTypeName)); } base.SetItem(index, item); } } class OpenableScopeCollection : OpenableCollection { public OpenableScopeCollection(bool opened) : base(opened) { } protected override void InsertItem(int index, Uri item) { if (item != null && !item.IsAbsoluteUri) { throw FxTrace.Exception.AsError(new ArgumentException(SR2.DiscoveryArgumentInvalidScopeUri(item))); } base.InsertItem(index, item); } protected override void SetItem(int index, Uri item) { if (item != null && !item.IsAbsoluteUri) { throw FxTrace.Exception.AsError(new ArgumentException(SR2.DiscoveryArgumentInvalidScopeUri(item))); } base.SetItem(index, item); } } } }