//----------------------------------------------------------------------------- // Copyright (c) Microsoft Corporation. All rights reserved. //----------------------------------------------------------------------------- namespace System.ServiceModel.Description { using System; using System.Collections.Generic; using System.Diagnostics.CodeAnalysis; using System.Globalization; using System.IO; using System.Reflection; using System.Runtime; using System.Runtime.Serialization; using System.ServiceModel; using System.ServiceModel.Dispatcher; using System.Xml; using System.Xml.Schema; using System.Xml.Serialization; using WsdlNS = System.Web.Services.Description; abstract class MessageContractExporter { readonly protected WsdlContractConversionContext contractContext; readonly protected WsdlExporter exporter; readonly protected OperationDescription operation; readonly protected IOperationBehavior extension; static internal void ExportMessageBinding(WsdlExporter exporter, WsdlEndpointConversionContext endpointContext, Type messageContractExporterType, OperationDescription operation) { new MessageBindingExporter(exporter, endpointContext).ExportMessageBinding(operation, messageContractExporterType); } protected abstract object OnExportMessageContract(); protected abstract void ExportHeaders(int messageIndex, object state); protected abstract void ExportBody(int messageIndex, object state); protected abstract void ExportKnownTypes(); protected abstract bool IsRpcStyle(); protected abstract bool IsEncoded(); protected abstract object GetExtensionData(); protected MessageExportContext ExportedMessages { get { return GetMessageExportContext(exporter); } } void AddElementToSchema(XmlSchemaElement element, string elementNs, XmlSchemaSet schemaSet) { OperationDescription parentOperation = this.operation; if (parentOperation.OperationMethod != null) { XmlQualifiedName qname = new XmlQualifiedName(element.Name, elementNs); OperationElement existingElement; if (ExportedMessages.ElementTypes.TryGetValue(qname, out existingElement)) { if (existingElement.Operation.OperationMethod == parentOperation.OperationMethod) return; if (!SchemaHelper.IsMatch(element, existingElement.Element)) { throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.CannotHaveTwoOperationsWithTheSameElement5, parentOperation.OperationMethod.DeclaringType, parentOperation.OperationMethod.Name, qname, existingElement.Operation.OperationMethod.DeclaringType, existingElement.Operation.Name))); } return; } else { ExportedMessages.ElementTypes.Add(qname, new OperationElement(element, parentOperation)); } } SchemaHelper.AddElementToSchema(element, SchemaHelper.GetSchema(elementNs, schemaSet), schemaSet); } static MessageExportContext GetMessageExportContext(WsdlExporter exporter) { object messageExportContext; if (!exporter.State.TryGetValue(typeof(MessageExportContext), out messageExportContext)) { messageExportContext = new MessageExportContext(); exporter.State[typeof(MessageExportContext)] = messageExportContext; } return (MessageExportContext)messageExportContext; } protected MessageContractExporter(WsdlExporter exporter, WsdlContractConversionContext context, OperationDescription operation, IOperationBehavior extension) { this.exporter = exporter; this.contractContext = context; this.operation = operation; this.extension = extension; } internal void ExportMessageContract() { if (extension == null) return; object state = OnExportMessageContract(); OperationFormatter.Validate(operation, IsRpcStyle(), IsEncoded()); ExportKnownTypes(); for (int messageIndex = 0; messageIndex < operation.Messages.Count; messageIndex++) ExportMessage(messageIndex, state); if (!operation.IsOneWay) { ExportFaults(state); } foreach (XmlSchema schema in exporter.GeneratedXmlSchemas.Schemas()) EnsureXsdImport(schema.TargetNamespace, contractContext.WsdlPortType.ServiceDescription); } void ExportMessage(int messageIndex, object state) { try { MessageDescription description = operation.Messages[messageIndex]; WsdlNS.Message wsdlMessage; if (CreateMessage(description, messageIndex, out wsdlMessage)) { if (description.IsUntypedMessage) { ExportAnyMessage(wsdlMessage, description.Body.ReturnValue ?? description.Body.Parts[0]); return; } bool isRequest = (messageIndex == 0); StreamFormatter streamFormatter = StreamFormatter.Create(description, operation.Name, isRequest); if (streamFormatter != null) { ExportStreamBody(wsdlMessage, streamFormatter.WrapperName, streamFormatter.WrapperNamespace, streamFormatter.PartName, streamFormatter.PartNamespace, IsRpcStyle(), false /*IsOperationInherited(operation)*/); } else { ExportBody(messageIndex, state); } } if (!description.IsUntypedMessage) { ExportHeaders(messageIndex, state); } } finally { Compile(); } } protected virtual void ExportFaults(object state) { foreach (FaultDescription fault in operation.Faults) { ExportFault(fault); } } protected bool IsOperationInherited() { return operation.DeclaringContract != contractContext.Contract; } void ExportAnyMessage(WsdlNS.Message message, MessagePartDescription part) { XmlSchemaSet schemas = this.exporter.GeneratedXmlSchemas; XmlSchema schema = SchemaHelper.GetSchema(DataContractSerializerMessageContractImporter.GenericMessageTypeName.Namespace, schemas); if (!schema.SchemaTypes.Contains(DataContractSerializerMessageContractImporter.GenericMessageTypeName)) { XmlSchemaComplexType genericMessageType = new XmlSchemaComplexType(); genericMessageType.Name = DataContractSerializerMessageContractImporter.GenericMessageTypeName.Name; XmlSchemaSequence bodySequence = new XmlSchemaSequence(); genericMessageType.Particle = bodySequence; XmlSchemaAny anyElement = new XmlSchemaAny(); anyElement.MinOccurs = 0; anyElement.MaxOccurs = decimal.MaxValue; anyElement.Namespace = "##any"; bodySequence.Items.Add(anyElement); SchemaHelper.AddTypeToSchema(genericMessageType, schema, schemas); } string partName = string.IsNullOrEmpty(part.UniquePartName) ? part.Name : part.UniquePartName; WsdlNS.MessagePart wsdlPart = AddMessagePart(message, partName, XmlQualifiedName.Empty, DataContractSerializerMessageContractImporter.GenericMessageTypeName); part.UniquePartName = wsdlPart.Name; } protected void ExportStreamBody(WsdlNS.Message message, string wrapperName, string wrapperNs, string partName, string partNs, bool isRpc, bool skipSchemaExport) { XmlSchemaSet schemas = this.exporter.GeneratedXmlSchemas; XmlSchema schema = SchemaHelper.GetSchema(DataContractSerializerMessageContractImporter.StreamBodyTypeName.Namespace, schemas); if (!schema.SchemaTypes.Contains(DataContractSerializerMessageContractImporter.StreamBodyTypeName)) { XmlSchemaSimpleType streamBodyType = new XmlSchemaSimpleType(); streamBodyType.Name = DataContractSerializerMessageContractImporter.StreamBodyTypeName.Name; XmlSchemaSimpleTypeRestriction contentRestriction = new XmlSchemaSimpleTypeRestriction(); contentRestriction.BaseTypeName = XmlSchemaType.GetBuiltInSimpleType(XmlTypeCode.Base64Binary).QualifiedName; streamBodyType.Content = contentRestriction; SchemaHelper.AddTypeToSchema(streamBodyType, schema, schemas); } XmlSchemaSequence wrapperSequence = null; if (!isRpc && wrapperName != null) wrapperSequence = ExportWrappedPart(message, wrapperName, wrapperNs, schemas, skipSchemaExport); MessagePartDescription streamPart = new MessagePartDescription(partName, partNs); ExportMessagePart(message, streamPart, DataContractSerializerMessageContractImporter.StreamBodyTypeName, null/*xsdType*/, false/*isOptional*/, false/*isNillable*/, skipSchemaExport, !isRpc, wrapperNs, wrapperSequence, schemas); } void ExportFault(FaultDescription fault) { WsdlNS.Message faultMessage = new WsdlNS.Message(); faultMessage.Name = GetFaultMessageName(fault.Name); XmlQualifiedName elementName = ExportFaultElement(fault); this.contractContext.WsdlPortType.ServiceDescription.Messages.Add(faultMessage); AddMessagePart(faultMessage, "detail", elementName, null); // create a wsdl:fault to put inside the wsdl:portType/wsdl:operation WsdlNS.OperationFault operationFault = contractContext.GetOperationFault(fault); WsdlExporter.WSAddressingHelper.AddActionAttribute(fault.Action, operationFault, this.exporter.PolicyVersion); operationFault.Message = new XmlQualifiedName(faultMessage.Name, faultMessage.ServiceDescription.TargetNamespace); } XmlQualifiedName ExportFaultElement(FaultDescription fault) { XmlSchemaType xsdType; XmlQualifiedName typeName = ExportType(fault.DetailType, fault.Name, operation.Name, out xsdType); XmlQualifiedName elementName; if (XmlName.IsNullOrEmpty(fault.ElementName)) { elementName = DataContractExporter.GetRootElementName(fault.DetailType); if (elementName == null) throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SFxFaultTypeAnonymous, operation.Name, fault.DetailType.FullName))); } else elementName = new XmlQualifiedName(fault.ElementName.EncodedName, fault.Namespace); ExportGlobalElement(elementName.Name, elementName.Namespace, true/*isNillable*/, typeName, xsdType, this.exporter.GeneratedXmlSchemas); return elementName; } protected XsdDataContractExporter DataContractExporter { get { object dataContractExporter; if (!exporter.State.TryGetValue(typeof(XsdDataContractExporter), out dataContractExporter)) { dataContractExporter = new XsdDataContractExporter(this.exporter.GeneratedXmlSchemas); exporter.State.Add(typeof(XsdDataContractExporter), dataContractExporter); } return (XsdDataContractExporter)dataContractExporter; } } protected XmlQualifiedName ExportType(Type type, string partName, string operationName, out XmlSchemaType xsdType) { xsdType = null; if (type == null) throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SFxExportMustHaveType, operationName, partName))); if (type == typeof(void)) return null; DataContractExporter.Export(type); XmlQualifiedName typeName = DataContractExporter.GetSchemaTypeName(type); if (IsNullOrEmpty(typeName)) xsdType = DataContractExporter.GetSchemaType(type); return typeName; } protected XmlSchemaSet SchemaSet { get { return exporter.GeneratedXmlSchemas; } } static protected WsdlNS.MessagePart AddMessagePart(WsdlNS.Message message, string partName, XmlQualifiedName elementName, XmlQualifiedName typeName) { if (message.Parts[partName] != null) { if (IsNullOrEmpty(elementName)) throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentException(SR.GetString(SR.SFxPartNameMustBeUniqueInRpc, partName))); int i = 1; while (message.Parts[partName + i] != null) { if (i == Int32.MaxValue) throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentException(SR.GetString(SR.SFxTooManyPartsWithSameName, partName))); i++; } partName = partName + i.ToString(CultureInfo.InvariantCulture); } WsdlNS.MessagePart part = new WsdlNS.MessagePart(); part.Name = partName; part.Element = elementName; part.Type = typeName; message.Parts.Add(part); EnsureXsdImport(IsNullOrEmpty(elementName) ? typeName.Namespace : elementName.Namespace, message.ServiceDescription); return part; } static void EnsureXsdImport(string ns, WsdlNS.ServiceDescription wsdl) { string refNs = wsdl.TargetNamespace; if (!refNs.EndsWith("/", StringComparison.Ordinal)) refNs = refNs + "/Imports"; else refNs += "Imports"; if (refNs == ns) refNs = wsdl.TargetNamespace; XmlSchema xsd = GetContainedSchema(wsdl, refNs); if (xsd != null) { foreach (object include in xsd.Includes) { XmlSchemaImport import = include as XmlSchemaImport; if (import != null && SchemaHelper.NamespacesEqual(import.Namespace, ns)) return; } } else { xsd = new XmlSchema(); xsd.TargetNamespace = refNs; wsdl.Types.Schemas.Add(xsd); } XmlSchemaImport imp = new XmlSchemaImport(); if (ns != null && ns.Length > 0) imp.Namespace = ns; xsd.Includes.Add(imp); } static XmlSchema GetContainedSchema(WsdlNS.ServiceDescription wsdl, string ns) { foreach (XmlSchema xsd in wsdl.Types.Schemas) if (SchemaHelper.NamespacesEqual(xsd.TargetNamespace, ns)) return xsd; return null; } static protected bool IsNullOrEmpty(XmlQualifiedName qname) { return qname == null || qname.IsEmpty; } protected void ExportGlobalElement(string elementName, string elementNs, bool isNillable, XmlQualifiedName typeName, XmlSchemaType xsdType, XmlSchemaSet schemaSet) { #if DEBUG Fx.Assert(NamingHelper.IsValidNCName(elementName), "Name value has to be a valid NCName."); if (xsdType == null) Fx.Assert(NamingHelper.IsValidNCName(typeName.Name), "Name value has to be a valid NCName."); #endif XmlSchemaElement element = new XmlSchemaElement(); element.Name = elementName; if (xsdType != null) element.SchemaType = xsdType; else element.SchemaTypeName = typeName; element.IsNillable = isNillable; AddElementToSchema(element, elementNs, schemaSet); } void ExportLocalElement(string wrapperNs, string elementName, string elementNs, XmlQualifiedName typeName, XmlSchemaType xsdType, bool multiple, bool isOptional, bool isNillable, XmlSchemaSequence sequence, XmlSchemaSet schemaSet) { #if DEBUG Fx.Assert(NamingHelper.IsValidNCName(elementName), "Name value has to be a valid NCName."); if (xsdType == null) Fx.Assert(NamingHelper.IsValidNCName(typeName.Name), "Name value has to be a valid NCName."); #endif XmlSchema schema = SchemaHelper.GetSchema(wrapperNs, schemaSet); XmlSchemaElement element = new XmlSchemaElement(); if (elementNs == wrapperNs) { element.Name = elementName; if (xsdType != null) element.SchemaType = xsdType; else { element.SchemaTypeName = typeName; SchemaHelper.AddImportToSchema(element.SchemaTypeName.Namespace, schema); } SchemaHelper.AddElementForm(element, schema); element.IsNillable = isNillable; } else { element.RefName = new XmlQualifiedName(elementName, elementNs); SchemaHelper.AddImportToSchema(elementNs, schema); ExportGlobalElement(elementName, elementNs, isNillable, typeName, xsdType, schemaSet); } if (multiple) element.MaxOccurs = Decimal.MaxValue; if (isOptional) element.MinOccurs = 0; sequence.Items.Add(element); } static readonly XmlSchemaSequence emptySequence = new XmlSchemaSequence(); protected XmlSchemaSequence ExportWrappedPart(WsdlNS.Message message, string elementName, string elementNs, XmlSchemaSet schemaSet, bool skipSchemaExport) { #if DEBUG Fx.Assert(NamingHelper.IsValidNCName(elementName), "Name value has to be a valid NCName."); #endif AddMessagePart(message, "parameters", new XmlQualifiedName(elementName, elementNs), XmlQualifiedName.Empty); if (skipSchemaExport) return emptySequence; //return empty to denote it is wrapped part XmlSchemaElement wrapperGlobalElement = new XmlSchemaElement(); wrapperGlobalElement.Name = elementName; XmlSchemaComplexType wrapperType = new XmlSchemaComplexType(); wrapperGlobalElement.SchemaType = wrapperType; // generating an anonymous type for wrapper XmlSchemaSequence rootSequence = new XmlSchemaSequence(); wrapperType.Particle = rootSequence; AddElementToSchema(wrapperGlobalElement, elementNs, schemaSet); return rootSequence; } protected bool CreateMessage(MessageDescription message, int messageIndex, out WsdlNS.Message wsdlMessage) { wsdlMessage = null; bool isNewMessage = true; if (ExportedMessages.WsdlMessages.ContainsKey(new MessageDescriptionDictionaryKey(contractContext.Contract, message))) throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentException(SR.GetString(SR.MultipleCallsToExportContractWithSameContract))); TypedMessageKey typedMessageKey = null; OperationMessageKey messageKey = null; if (message.IsTypedMessage) { typedMessageKey = new TypedMessageKey(message.MessageType, operation.DeclaringContract.Namespace, this.GetExtensionData()); if (ExportedMessages.TypedMessages.TryGetValue(typedMessageKey, out wsdlMessage)) isNewMessage = false; } else if (operation.OperationMethod != null) { messageKey = new OperationMessageKey(operation, messageIndex); if (ExportedMessages.ParameterMessages.TryGetValue(messageKey, out wsdlMessage)) isNewMessage = false; } WsdlNS.ServiceDescription wsdl = contractContext.WsdlPortType.ServiceDescription; if (isNewMessage) { wsdlMessage = new WsdlNS.Message(); wsdlMessage.Name = GetMessageName(message); wsdl.Messages.Add(wsdlMessage); if (message.IsTypedMessage) ExportedMessages.TypedMessages.Add(typedMessageKey, wsdlMessage); else if (messageKey != null) ExportedMessages.ParameterMessages.Add(messageKey, wsdlMessage); } //Add Name to OperationMessage WsdlNS.OperationMessage wsdlOperationMessage = contractContext.GetOperationMessage(message); wsdlOperationMessage.Message = new XmlQualifiedName(wsdlMessage.Name, wsdlMessage.ServiceDescription.TargetNamespace); this.ExportedMessages.WsdlMessages.Add(new MessageDescriptionDictionaryKey(contractContext.Contract, message), wsdlMessage); return isNewMessage; } protected bool CreateHeaderMessage(MessageDescription message, out WsdlNS.Message wsdlMessage) { wsdlMessage = null; if (ExportedMessages.WsdlHeaderMessages.ContainsKey(new MessageDescriptionDictionaryKey(contractContext.Contract, message))) throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentException(SR.GetString(SR.MultipleCallsToExportContractWithSameContract))); TypedMessageKey typedMessageKey = null; if (message.IsTypedMessage) { typedMessageKey = new TypedMessageKey(message.MessageType, operation.DeclaringContract.Namespace, GetExtensionData()); if (ExportedMessages.TypedHeaderMessages.TryGetValue(typedMessageKey, out wsdlMessage)) { this.ExportedMessages.WsdlHeaderMessages.Add(new MessageDescriptionDictionaryKey(contractContext.Contract, message), wsdlMessage); return false; } } string messageName = GetHeaderMessageName(message); wsdlMessage = new WsdlNS.Message(); wsdlMessage.Name = messageName; contractContext.WsdlPortType.ServiceDescription.Messages.Add(wsdlMessage); if (message.IsTypedMessage) ExportedMessages.TypedHeaderMessages.Add(typedMessageKey, wsdlMessage); this.ExportedMessages.WsdlHeaderMessages.Add(new MessageDescriptionDictionaryKey(contractContext.Contract, message), wsdlMessage); return true; } string GetMessageName(MessageDescription messageDescription) { string messageNameBase = XmlName.IsNullOrEmpty(messageDescription.MessageName) ? null : messageDescription.MessageName.EncodedName; //If there wasn't one in the Message Description we create one. if (string.IsNullOrEmpty(messageNameBase)) { string portTypeName = contractContext.WsdlPortType.Name; string operationName = contractContext.GetOperation(operation).Name; string callbackString = operation.IsServerInitiated() ? "Callback" : string.Empty; // Microsoft: composing names have potential problem of generating name that looks like an encoded name, consider avoiding '_' if (messageDescription.Direction == MessageDirection.Input) messageNameBase = string.Format(System.Globalization.CultureInfo.InvariantCulture, "{0}_{1}_Input{2}Message", portTypeName, operationName, callbackString); else messageNameBase = string.Format(System.Globalization.CultureInfo.InvariantCulture, "{0}_{1}_Output{2}Message", portTypeName, operationName, callbackString); } WsdlNS.ServiceDescription wsdl = contractContext.WsdlPortType.ServiceDescription; return GetUniqueMessageName(wsdl, messageNameBase); } string GetHeaderMessageName(MessageDescription messageDescription) { WsdlNS.Message wsdlBodyMessage = this.ExportedMessages.WsdlMessages[new MessageDescriptionDictionaryKey(this.contractContext.Contract, messageDescription)]; string messageNameBase = string.Format(System.Globalization.CultureInfo.InvariantCulture, "{0}_Headers", wsdlBodyMessage.Name); WsdlNS.ServiceDescription wsdl = contractContext.WsdlPortType.ServiceDescription; return GetUniqueMessageName(wsdl, messageNameBase); } protected string GetFaultMessageName(string faultName) { string portTypeName = contractContext.WsdlPortType.Name; string operationName = contractContext.GetOperation(operation).Name; // Microsoft: composing names have potential problem of generating name that looks like an encoded name, consider avoiding '_' string faultNameBase = String.Format(System.Globalization.CultureInfo.InvariantCulture, "{0}_{1}_{2}_FaultMessage", portTypeName, operationName, faultName); WsdlNS.ServiceDescription wsdl = contractContext.WsdlPortType.ServiceDescription; return GetUniqueMessageName(wsdl, faultNameBase); } static bool DoesMessageNameExist(string messageName, object wsdlObject) { return ((WsdlNS.ServiceDescription)wsdlObject).Messages[messageName] != null; } string GetUniqueMessageName(WsdlNS.ServiceDescription wsdl, string messageNameBase) { return NamingHelper.GetUniqueName(messageNameBase, DoesMessageNameExist, wsdl); } protected void ExportMessagePart(WsdlNS.Message message, MessagePartDescription part, XmlQualifiedName typeName, XmlSchemaType xsdType, bool isOptional, bool isNillable, bool skipSchemaExport, bool generateElement, string wrapperNs, XmlSchemaSequence wrapperSequence, XmlSchemaSet schemaSet) { if (IsNullOrEmpty(typeName) && xsdType == null) return; #if DEBUG if (xsdType == null) Fx.Assert(NamingHelper.IsValidNCName(typeName.Name), "Name value has to be a valid NCName."); #endif string elementName = part.Name; string partName = string.IsNullOrEmpty(part.UniquePartName) ? elementName : part.UniquePartName; WsdlNS.MessagePart wsdlPart = null; if (generateElement) { if (wrapperSequence != null) { if (!skipSchemaExport) ExportLocalElement(wrapperNs, partName, part.Namespace, typeName, xsdType, part.Multiple, isOptional, isNillable, wrapperSequence, schemaSet); } else { if (!skipSchemaExport) ExportGlobalElement(elementName, part.Namespace, isNillable, typeName, xsdType, schemaSet); wsdlPart = AddMessagePart(message, partName, new XmlQualifiedName(elementName, part.Namespace), XmlQualifiedName.Empty); } } else { if (String.IsNullOrEmpty(typeName.Name)) throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SFxAnonymousTypeNotSupported, message.Name, partName))); wsdlPart = AddMessagePart(message, partName, XmlQualifiedName.Empty, typeName); } if (wsdlPart != null) { part.UniquePartName = wsdlPart.Name; } } protected void AddParameterOrder(MessageDescription message) { if (operation == null) return; WsdlNS.Operation wsdlOperation = contractContext.GetOperation(operation); if (wsdlOperation != null) { if (wsdlOperation.ParameterOrder == null) { wsdlOperation.ParameterOrder = new string[GetParameterCount()]; } if (wsdlOperation.ParameterOrder.Length == 0) return; foreach (MessagePartDescription part in message.Body.Parts) { ParameterInfo paramInfo = part.AdditionalAttributesProvider as ParameterInfo; if (paramInfo != null && paramInfo.Position >= 0) wsdlOperation.ParameterOrder[paramInfo.Position] = part.Name; } } } int GetParameterCount() { int count = -1; foreach (MessageDescription message in operation.Messages) { foreach (MessagePartDescription part in message.Body.Parts) { ParameterInfo paramInfo = part.AdditionalAttributesProvider as ParameterInfo; if (paramInfo == null) return 0; if (count < paramInfo.Position) count = paramInfo.Position; } } return count + 1; } protected virtual void Compile() { foreach (XmlSchema schema in SchemaSet.Schemas()) SchemaSet.Reprocess(schema); SchemaHelper.Compile(SchemaSet, exporter.Errors); } class MessageBindingExporter { WsdlEndpointConversionContext endpointContext; MessageExportContext exportedMessages; EnvelopeVersion soapVersion; WsdlExporter exporter; internal MessageBindingExporter(WsdlExporter exporter, WsdlEndpointConversionContext endpointContext) { this.endpointContext = endpointContext; this.exportedMessages = (MessageExportContext)exporter.State[typeof(MessageExportContext)]; this.soapVersion = SoapHelper.GetSoapVersion(endpointContext.WsdlBinding); this.exporter = exporter; } internal void ExportMessageBinding(OperationDescription operation, Type messageContractExporterType) { WsdlNS.OperationBinding wsdlOperationBinding = endpointContext.GetOperationBinding(operation); bool isRpc, isEncoded; if (!GetStyleAndUse(operation, messageContractExporterType, out isRpc, out isEncoded)) return; WsdlNS.SoapOperationBinding soapOperationBinding = SoapHelper.GetOrCreateSoapOperationBinding(endpointContext, operation, exporter); if (soapOperationBinding == null) return; soapOperationBinding.Style = isRpc ? WsdlNS.SoapBindingStyle.Rpc : WsdlNS.SoapBindingStyle.Document; if (isRpc) { WsdlNS.SoapBinding soapBinding = (WsdlNS.SoapBinding)endpointContext.WsdlBinding.Extensions.Find(typeof(WsdlNS.SoapBinding)); soapBinding.Style = soapOperationBinding.Style; } soapOperationBinding.SoapAction = operation.Messages[0].Action; foreach (MessageDescription message in operation.Messages) { WsdlNS.MessageBinding wsdlMessageBinding = endpointContext.GetMessageBinding(message); WsdlNS.Message headerMessage; if (this.exportedMessages.WsdlHeaderMessages.TryGetValue(new MessageDescriptionDictionaryKey(this.endpointContext.Endpoint.Contract, message), out headerMessage)) { XmlQualifiedName wsdlHeaderMessageName = new XmlQualifiedName(headerMessage.Name, headerMessage.ServiceDescription.TargetNamespace); foreach (MessageHeaderDescription header in message.Headers) { if (header.IsUnknownHeaderCollection) continue; ExportMessageHeaderBinding(header, wsdlHeaderMessageName, isEncoded, wsdlMessageBinding); } } ExportMessageBodyBinding(message, isRpc, isEncoded, wsdlMessageBinding); } foreach (FaultDescription fault in operation.Faults) { ExportFaultBinding(fault, isEncoded, wsdlOperationBinding); } } void ExportFaultBinding(FaultDescription fault, bool isEncoded, WsdlNS.OperationBinding operationBinding) { SoapHelper.CreateSoapFaultBinding(fault.Name, endpointContext, endpointContext.GetFaultBinding(fault), isEncoded); } void ExportMessageBodyBinding(MessageDescription messageDescription, bool isRpc, bool isEncoded, WsdlNS.MessageBinding messageBinding) { WsdlNS.SoapBodyBinding bodyBinding = SoapHelper.GetOrCreateSoapBodyBinding(endpointContext, messageBinding, exporter); if (bodyBinding == null) return; bodyBinding.Use = isEncoded ? WsdlNS.SoapBindingUse.Encoded : WsdlNS.SoapBindingUse.Literal; if (isRpc) { string ns; if (!ExportedMessages.WrapperNamespaces.TryGetValue(new MessageDescriptionDictionaryKey(endpointContext.ContractConversionContext.Contract, messageDescription), out ns)) ns = messageDescription.Body.WrapperNamespace; bodyBinding.Namespace = ns; } if (isEncoded) bodyBinding.Encoding = XmlSerializerOperationFormatter.GetEncoding(soapVersion); } void ExportMessageHeaderBinding(MessageHeaderDescription header, XmlQualifiedName messageName, bool isEncoded, WsdlNS.MessageBinding messageBinding) { #if DEBUG Fx.Assert(NamingHelper.IsValidNCName(messageName.Name), "Name value has to be a valid NCName."); #endif WsdlNS.SoapHeaderBinding headerBinding = SoapHelper.CreateSoapHeaderBinding(endpointContext, messageBinding); headerBinding.Part = string.IsNullOrEmpty(header.UniquePartName) ? header.Name : header.UniquePartName; headerBinding.Message = messageName; headerBinding.Use = isEncoded ? WsdlNS.SoapBindingUse.Encoded : WsdlNS.SoapBindingUse.Literal; if (isEncoded) headerBinding.Encoding = XmlSerializerOperationFormatter.GetEncoding(soapVersion); } static bool GetStyleAndUse(OperationDescription operation, Type messageContractExporterType, out bool isRpc, out bool isEncoded) { isRpc = isEncoded = false; if (messageContractExporterType == typeof(DataContractSerializerMessageContractExporter) || messageContractExporterType == null) { DataContractSerializerOperationBehavior dataContractSerializerBehavior = operation.Behaviors.Find(); if (dataContractSerializerBehavior != null) { isRpc = dataContractSerializerBehavior.DataContractFormatAttribute.Style == OperationFormatStyle.Rpc; isEncoded = false; return true; } if (messageContractExporterType == typeof(DataContractSerializerMessageContractExporter)) return false; } if (messageContractExporterType == typeof(XmlSerializerMessageContractExporter) || messageContractExporterType == null) { XmlSerializerOperationBehavior xmlSerializerBehavior = operation.Behaviors.Find(); if (xmlSerializerBehavior != null) { isRpc = xmlSerializerBehavior.XmlSerializerFormatAttribute.Style == OperationFormatStyle.Rpc; isEncoded = xmlSerializerBehavior.XmlSerializerFormatAttribute.IsEncoded; return true; } return false; } return false; } MessageExportContext ExportedMessages { get { return GetMessageExportContext(exporter); } } } protected class MessageExportContext { readonly internal Dictionary WsdlMessages = new Dictionary(); readonly internal Dictionary WsdlHeaderMessages = new Dictionary(); readonly internal Dictionary WrapperNamespaces = new Dictionary(); readonly internal Dictionary TypedMessages = new Dictionary(); readonly internal Dictionary TypedHeaderMessages = new Dictionary(); readonly internal Dictionary ParameterMessages = new Dictionary(); readonly internal Dictionary ElementTypes = new Dictionary(); } protected sealed class MessageDescriptionDictionaryKey { public readonly ContractDescription Contract; public readonly MessageDescription MessageDescription; public MessageDescriptionDictionaryKey(ContractDescription contract, MessageDescription MessageDescription) { this.Contract = contract; this.MessageDescription = MessageDescription; } public override bool Equals(object obj) { MessageDescriptionDictionaryKey key = obj as MessageDescriptionDictionaryKey; if (key != null && key.MessageDescription == this.MessageDescription && key.Contract == this.Contract) return true; return false; } public override int GetHashCode() { return this.Contract.GetHashCode() ^ this.MessageDescription.GetHashCode(); } } internal sealed class TypedMessageKey { Type type; string contractNS; object extensionData; public TypedMessageKey(Type type, string contractNS, object extensionData) { this.type = type; this.contractNS = contractNS; this.extensionData = extensionData; } public override bool Equals(object obj) { TypedMessageKey key = obj as TypedMessageKey; if (key != null && key.type == this.type && key.contractNS == this.contractNS && key.extensionData.Equals(this.extensionData)) return true; return false; } [SuppressMessage(FxCop.Category.Usage, "CA2303:FlagTypeGetHashCode", Justification = "The hashcode is not used for identity purposes for embedded types.")] public override int GetHashCode() { return type.GetHashCode(); } } internal sealed class OperationMessageKey { MethodInfo methodInfo; int messageIndex; ContractDescription declaringContract; public OperationMessageKey(OperationDescription operation, int messageIndex) { this.methodInfo = operation.OperationMethod; this.messageIndex = messageIndex; this.declaringContract = operation.DeclaringContract; } public override bool Equals(object obj) { OperationMessageKey key = obj as OperationMessageKey; if (key != null && key.methodInfo == this.methodInfo && key.messageIndex == this.messageIndex && key.declaringContract.Name == this.declaringContract.Name && key.declaringContract.Namespace == this.declaringContract.Namespace) return true; return false; } public override int GetHashCode() { return methodInfo.GetHashCode() ^ messageIndex; } } internal sealed class OperationElement { XmlSchemaElement element; OperationDescription operation; internal OperationElement(XmlSchemaElement element, OperationDescription operation) { this.element = element; this.operation = operation; } internal XmlSchemaElement Element { get { return element; } } internal OperationDescription Operation { get { return operation; } } } } class DataContractSerializerMessageContractExporter : MessageContractExporter { internal DataContractSerializerMessageContractExporter(WsdlExporter exporter, WsdlContractConversionContext context, OperationDescription operation, IOperationBehavior extension) : base(exporter, context, operation, extension) { } protected override void Compile() { XmlSchema wsdl = StockSchemas.CreateWsdl(); XmlSchema soap = StockSchemas.CreateSoap(); XmlSchema soapEncoding = StockSchemas.CreateSoapEncoding(); XmlSchema fakeXsdSchema = StockSchemas.CreateFakeXsdSchema(); SchemaSet.Add(wsdl); SchemaSet.Add(soap); SchemaSet.Add(soapEncoding); SchemaSet.Add(fakeXsdSchema); base.Compile(); SchemaSet.Remove(wsdl); SchemaSet.Remove(soap); SchemaSet.Remove(soapEncoding); SchemaSet.Remove(fakeXsdSchema); } protected override bool IsRpcStyle() { return ((DataContractSerializerOperationBehavior)extension).DataContractFormatAttribute.Style == OperationFormatStyle.Rpc; } protected override bool IsEncoded() { return false; } protected override object OnExportMessageContract() { return null; } protected override void ExportHeaders(int messageIndex, object state) { MessageDescription description = operation.Messages[messageIndex]; if (description.Headers.Count > 0) { WsdlNS.Message wsdlMessage; if (CreateHeaderMessage(description, out wsdlMessage)) { foreach (MessageHeaderDescription header in description.Headers) { if (header.IsUnknownHeaderCollection) continue; XmlSchemaType xsdType; bool isQueryable; Type dataContractType = DataContractSerializerOperationFormatter.GetSubstituteDataContractType(header.Type, out isQueryable); XmlQualifiedName typeName = ExportType(dataContractType, header.Name, operation.Name, out xsdType); ExportMessagePart(wsdlMessage, header, typeName, xsdType, true/*isOptional*/, IsTypeNullable(header.Type), false/*IsOperationInherited(operation)*/, true /*generateElement*/, null/*wrapperNamespace*/, null/*wrapperSequence*/, SchemaSet); } } } } static internal bool IsTypeNullable(Type type) { return !type.IsValueType || (type.IsGenericType && type.GetGenericTypeDefinition() == typeof(Nullable<>)); } protected override void ExportBody(int messageIndex, object state) { MessageDescription description = operation.Messages[messageIndex]; WsdlNS.Message wsdlMessage = this.ExportedMessages.WsdlMessages[new MessageDescriptionDictionaryKey(this.contractContext.Contract, description)]; DataContractFormatAttribute dataContractFormatAttribute = ((DataContractSerializerOperationBehavior)extension).DataContractFormatAttribute; XmlSchemaSequence wrapperSequence = null; bool isWrapped = description.Body.WrapperName != null; //bool isOperationInherited = IsOperationInherited(operation); if (dataContractFormatAttribute.Style == OperationFormatStyle.Document && isWrapped) wrapperSequence = ExportWrappedPart(wsdlMessage, description.Body.WrapperName, description.Body.WrapperNamespace, SchemaSet, false /*isOperationInherited*/); XmlSchemaType xsdType; if (OperationFormatter.IsValidReturnValue(description.Body.ReturnValue)) { bool isQueryable; Type dataContractType = DataContractSerializerOperationFormatter.GetSubstituteDataContractType(description.Body.ReturnValue.Type, out isQueryable); XmlQualifiedName typeName = ExportType(dataContractType, description.Body.ReturnValue.Name, operation.Name, out xsdType); ExportMessagePart(wsdlMessage, description.Body.ReturnValue, typeName, xsdType, true/*isOptional*/, IsTypeNullable(description.Body.ReturnValue.Type), false/*isOperationInherited*/, dataContractFormatAttribute.Style != OperationFormatStyle.Rpc, description.Body.WrapperNamespace, wrapperSequence, SchemaSet); } foreach (MessagePartDescription bodyPart in description.Body.Parts) { bool isQueryable; Type dataContractType = DataContractSerializerOperationFormatter.GetSubstituteDataContractType(bodyPart.Type, out isQueryable); XmlQualifiedName typeName = ExportType(dataContractType, bodyPart.Name, operation.Name, out xsdType); ExportMessagePart(wsdlMessage, bodyPart, typeName, xsdType, true/*isOptional*/, IsTypeNullable(bodyPart.Type), false/*isOperationInherited*/, dataContractFormatAttribute.Style != OperationFormatStyle.Rpc, description.Body.WrapperNamespace, wrapperSequence, SchemaSet); } if (dataContractFormatAttribute.Style == OperationFormatStyle.Rpc) { AddParameterOrder(description); } } protected override void ExportKnownTypes() { foreach (Type knownType in operation.KnownTypes) { DataContractExporter.Export(knownType); } } protected override object GetExtensionData() { return new ExtensionData(((DataContractSerializerOperationBehavior)extension).DataContractFormatAttribute); } class ExtensionData { DataContractFormatAttribute dcFormatAttr; internal ExtensionData(DataContractFormatAttribute dcFormatAttr) { this.dcFormatAttr = dcFormatAttr; } public override bool Equals(object obj) { if (object.ReferenceEquals(dcFormatAttr, obj)) return true; ExtensionData otherExtensionData = obj as ExtensionData; if (otherExtensionData == null) return false; return dcFormatAttr.Style == otherExtensionData.dcFormatAttr.Style; } public override int GetHashCode() { return 1; //This is never called } } } class XmlSerializerMessageContractExporter : MessageContractExporter { internal XmlSerializerMessageContractExporter(WsdlExporter exporter, WsdlContractConversionContext context, OperationDescription operation, IOperationBehavior extension) : base(exporter, context, operation, extension) { } protected override bool IsRpcStyle() { return ((XmlSerializerOperationBehavior)extension).XmlSerializerFormatAttribute.Style == OperationFormatStyle.Rpc; } protected override bool IsEncoded() { return ((XmlSerializerOperationBehavior)extension).XmlSerializerFormatAttribute.IsEncoded; } protected override object OnExportMessageContract() { object result = Reflector.ReflectOperation(operation); if (result == null) { // If result is null, that means that XmlSerializerFormatAttribute wasn't available in reflection, // so we need to get it from the XmlSerializerOperationBehavior instead XmlSerializerOperationBehavior serializerBehavior = this.extension as XmlSerializerOperationBehavior; if (serializerBehavior != null) { result = Reflector.ReflectOperation(operation, serializerBehavior.XmlSerializerFormatAttribute); } } return result; } protected override void ExportHeaders(int messageIndex, object state) { string portTypeName = contractContext.WsdlPortType.Name; string portTypeNs = contractContext.WsdlPortType.ServiceDescription.TargetNamespace; MessageDescription description = operation.Messages[messageIndex]; if (description.Headers.Count > 0) { XmlSerializerOperationBehavior.Reflector.OperationReflector operationReflector = (XmlSerializerOperationBehavior.Reflector.OperationReflector)state; XmlMembersMapping membersMapping = null; if (messageIndex == 0) { membersMapping = operationReflector.Request.HeadersMapping; } else { membersMapping = operationReflector.Reply.HeadersMapping; } if (membersMapping != null) { WsdlNS.Message wsdlMessage; if (CreateHeaderMessage(description, out wsdlMessage)) { ExportMembersMapping(membersMapping, wsdlMessage, false /*IsOperationInherited(operation)*/, operationReflector.IsEncoded, false/*isRpc*/, false/*isWrapped*/, true/*isHeader*/); } } } } protected override void ExportBody(int messageIndex, object state) { MessageDescription description = operation.Messages[messageIndex]; string portTypeName = contractContext.WsdlPortType.Name; string portTypeNs = contractContext.WsdlPortType.ServiceDescription.TargetNamespace; MessageDescriptionDictionaryKey key = new MessageDescriptionDictionaryKey(this.contractContext.Contract, description); WsdlNS.Message wsdlMessage = this.ExportedMessages.WsdlMessages[key]; XmlSerializerOperationBehavior.Reflector.OperationReflector operationReflector = (XmlSerializerOperationBehavior.Reflector.OperationReflector)state; XmlMembersMapping membersMapping = null; if (messageIndex == 0) { membersMapping = operationReflector.Request.BodyMapping; } else { membersMapping = operationReflector.Reply.BodyMapping; } if (membersMapping != null) { bool isDocWrapped = !operationReflector.IsRpc && description.Body.WrapperName != null; ExportMembersMapping(membersMapping, wsdlMessage, false /*IsOperationInherited(operation)*/, operationReflector.IsEncoded, operationReflector.IsRpc, isDocWrapped, false/*isHeader*/); if (operationReflector.IsRpc) { AddParameterOrder(operation.Messages[messageIndex]); this.ExportedMessages.WrapperNamespaces.Add(key, membersMapping.Namespace); } } } protected override void ExportFaults(object state) { XmlSerializerOperationBehavior.Reflector.OperationReflector operationReflector = (XmlSerializerOperationBehavior.Reflector.OperationReflector)state; if (operationReflector.Attribute.SupportFaults) { foreach (FaultDescription fault in operation.Faults) { ExportFault(fault, operationReflector); } Compile(); } else { base.ExportFaults(state); } } void ExportFault(FaultDescription fault, XmlSerializerOperationBehavior.Reflector.OperationReflector operationReflector) { WsdlNS.Message faultMessage = new WsdlNS.Message(); faultMessage.Name = GetFaultMessageName(fault.Name); XmlQualifiedName elementName = ExportFaultElement(fault, operationReflector); this.contractContext.WsdlPortType.ServiceDescription.Messages.Add(faultMessage); AddMessagePart(faultMessage, "detail", elementName, null); // create a wsdl:fault to put inside the wsdl:portType/wsdl:operation WsdlNS.OperationFault operationFault = contractContext.GetOperationFault(fault); WsdlExporter.WSAddressingHelper.AddActionAttribute(fault.Action, operationFault, this.exporter.PolicyVersion); operationFault.Message = new XmlQualifiedName(faultMessage.Name, faultMessage.ServiceDescription.TargetNamespace); } XmlQualifiedName ExportFaultElement(FaultDescription fault, XmlSerializerOperationBehavior.Reflector.OperationReflector operationReflector) { XmlQualifiedName elementName; XmlMembersMapping mapping = operationReflector.ImportFaultElement(fault, out elementName); if (operationReflector.IsEncoded) SoapExporter.ExportMembersMapping(mapping); else XmlExporter.ExportMembersMapping(mapping); return elementName; } protected override void ExportKnownTypes() { } protected override object GetExtensionData() { return new ExtensionData(((XmlSerializerOperationBehavior)this.extension).XmlSerializerFormatAttribute); } class ExtensionData { XmlSerializerFormatAttribute xsFormatAttr; internal ExtensionData(XmlSerializerFormatAttribute xsFormatAttr) { this.xsFormatAttr = xsFormatAttr; } public override bool Equals(object obj) { if (object.ReferenceEquals(xsFormatAttr, obj)) return true; ExtensionData otherExtensionData = obj as ExtensionData; if (otherExtensionData == null) return false; return xsFormatAttr.Style == otherExtensionData.xsFormatAttr.Style && xsFormatAttr.Use == otherExtensionData.xsFormatAttr.Use; } public override int GetHashCode() { return 1; //This is never called } } void ExportMembersMapping(XmlMembersMapping membersMapping, WsdlNS.Message message, bool skipSchemaExport, bool isEncoded, bool isRpc, bool isDocWrapped, bool isHeader) { if (!skipSchemaExport) { if (isEncoded) SoapExporter.ExportMembersMapping(membersMapping); else XmlExporter.ExportMembersMapping(membersMapping, !isRpc); } if (isDocWrapped) { if (isHeader) { Fx.Assert("Header cannot be Document Wrapped"); throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(String.Format(CultureInfo.InvariantCulture, "Header cannot be Document Wrapped"))); } AddMessagePart(message, "parameters", new XmlQualifiedName(membersMapping.XsdElementName, membersMapping.Namespace), XmlQualifiedName.Empty); return; } bool generateElement = !isRpc && !isEncoded; for (int i = 0; i < membersMapping.Count; i++) { XmlMemberMapping member = membersMapping[i]; string partName = (isHeader || generateElement) ? NamingHelper.XmlName(member.MemberName) : member.XsdElementName; if (generateElement) AddMessagePart(message, partName, new XmlQualifiedName(member.XsdElementName, member.Namespace), XmlQualifiedName.Empty); else { if (string.IsNullOrEmpty(member.TypeName)) throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SFxAnonymousTypeNotSupported, message.Name, partName))); AddMessagePart(message, partName, XmlQualifiedName.Empty, new XmlQualifiedName(member.TypeName, member.TypeNamespace)); } } } XmlSerializerOperationBehavior.Reflector Reflector { get { object reflector; if (!exporter.State.TryGetValue(typeof(XmlSerializerOperationBehavior.Reflector), out reflector)) { reflector = new XmlSerializerOperationBehavior.Reflector(contractContext.Contract.Namespace, contractContext.Contract.ContractType); exporter.State.Add(typeof(XmlSerializerOperationBehavior.Reflector), reflector); } return (XmlSerializerOperationBehavior.Reflector)reflector; } } SoapSchemaExporter SoapExporter { get { object soapExporter; if (!exporter.State.TryGetValue(typeof(SoapSchemaExporter), out soapExporter)) { soapExporter = new SoapSchemaExporter(Schemas); exporter.State.Add(typeof(SoapSchemaExporter), soapExporter); } return (SoapSchemaExporter)soapExporter; } } XmlSchemaExporter XmlExporter { get { object xmlExporter; if (!exporter.State.TryGetValue(typeof(XmlSchemaExporter), out xmlExporter)) { xmlExporter = new XmlSchemaExporter(Schemas); exporter.State.Add(typeof(XmlSchemaExporter), xmlExporter); } return (XmlSchemaExporter)xmlExporter; } } XmlSchemas Schemas { get { object schemas; if (!exporter.State.TryGetValue(typeof(XmlSchemas), out schemas)) { schemas = new XmlSchemas(); foreach (XmlSchema schema in this.SchemaSet.Schemas()) if (!((XmlSchemas)schemas).Contains(schema.TargetNamespace)) ((XmlSchemas)schemas).Add(schema); exporter.State.Add(typeof(XmlSchemas), schemas); } return (XmlSchemas)schemas; } } protected override void Compile() { XmlSchema wsdl = StockSchemas.CreateWsdl(); XmlSchema soap = StockSchemas.CreateSoap(); XmlSchema soapEncoding = StockSchemas.CreateSoapEncoding(); XmlSchema fakeXsdSchema = StockSchemas.CreateFakeXsdSchema(); MoveSchemas(); SchemaSet.Add(wsdl); SchemaSet.Add(soap); SchemaSet.Add(soapEncoding); SchemaSet.Add(fakeXsdSchema); base.Compile(); SchemaSet.Remove(wsdl); SchemaSet.Remove(soap); SchemaSet.Remove(soapEncoding); SchemaSet.Remove(fakeXsdSchema); } void MoveSchemas() { XmlSchemas schemas = this.Schemas; XmlSchemaSet schemaSet = this.SchemaSet; if (schemas != null) { schemas.Compile( delegate(object sender, ValidationEventArgs args) { SchemaHelper.HandleSchemaValidationError(sender, args, exporter.Errors); }, false/*fullCompile*/ ); foreach (XmlSchema srcSchema in schemas) { if (!schemaSet.Contains(srcSchema)) { schemaSet.Add(srcSchema); schemaSet.Reprocess(srcSchema); } } } } } static class StockSchemas { internal static XmlSchema CreateWsdl() { StringReader reader = new StringReader(wsdl); return XmlSchema.Read(new XmlTextReader(reader) { DtdProcessing = DtdProcessing.Prohibit }, null); } internal static XmlSchema CreateSoap() { StringReader reader = new StringReader(soap); return XmlSchema.Read(new XmlTextReader(reader) { DtdProcessing = DtdProcessing.Prohibit }, null); } internal static XmlSchema CreateSoapEncoding() { StringReader reader = new StringReader(soapEncoding); return XmlSchema.Read(new XmlTextReader(reader) { DtdProcessing = DtdProcessing.Prohibit }, null); } internal static XmlSchema CreateFakeSoapEncoding() { StringReader reader = new StringReader(fakeSoapEncoding); return XmlSchema.Read(new XmlTextReader(reader) { DtdProcessing = DtdProcessing.Prohibit }, null); } internal static XmlSchema CreateFakeXsdSchema() { StringReader reader = new StringReader(fakeXsd); return XmlSchema.Read(new XmlTextReader(reader) { DtdProcessing = DtdProcessing.Prohibit }, null); } internal static XmlSchema CreateFakeXmlSchema() { StringReader reader = new StringReader(fakeXmlSchema); return XmlSchema.Read(new XmlTextReader(reader) { DtdProcessing = DtdProcessing.Prohibit }, null); } internal static bool IsKnownSchema(string ns) { return ns == XmlSchema.Namespace || ns == "http://schemas.xmlsoap.org/wsdl/soap/" || ns == "http://schemas.xmlsoap.org/soap/encoding/"; } internal const string WsdlNamespace = "http://schemas.xmlsoap.org/wsdl/"; internal const string SoapNamespace = "http://schemas.xmlsoap.org/wsdl/soap/"; internal const string SoapEncodingNamespace = "http://schemas.xmlsoap.org/soap/encoding/"; const string wsdl = @" This type is extended by component types to allow them to be documented This type is extended by component types to allow attributes from other namespaces to be added. This type is extended by component types to allow elements from other namespaces to be added. Any top level optional element allowed to appear more then once - any child of definitions element except wsdl:types. Any extensibility element is allowed in any place. "; const string soap = @" 'encodingStyle' indicates any canonicalization conventions followed in the contents of the containing element. For example, the value 'http://schemas.xmlsoap.org/soap/encoding/' indicates the pattern described in SOAP specification "; const string soapEncoding = @" "; const string fakeXsd = @" "; const string fakeXmlSchema = @" "; const string fakeSoapEncoding = @" "; } }