File: System\Web\Services\Protocols\Soap12ServerProtocol.cs
Project: ndp\cdf\src\NetFx20\System.Web.Services\System.Web.Services.csproj (System.Web.Services)
//------------------------------------------------------------------------------
// <copyright file="SoapServerProtocol.cs" company="Microsoft">
//     Copyright (c) Microsoft Corporation.  All rights reserved.
// </copyright>                                                                
//------------------------------------------------------------------------------
 
namespace System.Web.Services.Protocols {
    using System;
    using System.Collections;
    using System.IO;
    using System.Net;
    using System.Reflection;
    using System.Text;
    using System.Xml;
    using System.Xml.Schema;
    using System.Xml.Serialization;
    using System.Web.Services.Configuration;
    using System.Web.Services.Description;
    using System.Web.Services.Diagnostics;
 
    internal class Soap12ServerProtocolHelper : SoapServerProtocolHelper {
        internal Soap12ServerProtocolHelper(SoapServerProtocol protocol) : base(protocol) {
        }
 
        internal Soap12ServerProtocolHelper(SoapServerProtocol protocol, string requestNamespace) : base(protocol, requestNamespace) {
        }
 
        internal override SoapProtocolVersion Version { 
            get { return SoapProtocolVersion.Soap12; }
        }
        internal override WebServiceProtocols Protocol {
            get { return WebServiceProtocols.HttpSoap12; }
        }
        internal override string EnvelopeNs { 
            get { return Soap12.Namespace; } 
        }
        internal override string EncodingNs { 
            get { return Soap12.Encoding; } 
        }
        internal override string HttpContentType { 
            get { return ContentType.ApplicationSoap; } 
        }
        
        internal override SoapServerMethod RouteRequest() {
            string action = ContentType.GetAction(ServerProtocol.Request.ContentType);
 
            SoapServerMethod method = null;
            bool duplicateAction = false, duplicateRequestElement = false;
 
            TraceMethod caller = Tracing.On ? new TraceMethod(this, "RouteRequest") : null;
 
            if (action != null && action.Length > 0) {
                action = HttpUtility.UrlDecode(action);
                if (Tracing.On) Tracing.Enter("RouteRequest", caller, new TraceMethod(ServerType, "GetMethod", action), Tracing.Details(ServerProtocol.Request));
                method = ServerType.GetMethod(action);
                if (Tracing.On) Tracing.Exit("RouteRequest", caller);
                if (method != null) {
                    if (ServerType.GetDuplicateMethod(action) != null) {
                        method = null;
                        duplicateAction = true;
                    }
                }
            }
 
            XmlQualifiedName requestElement = XmlQualifiedName.Empty;
            if (method == null) {
                // try request element
                requestElement = GetRequestElement();
                if (Tracing.On) Tracing.Enter("RouteRequest", caller, new TraceMethod(ServerType, "GetMethod", requestElement), Tracing.Details(ServerProtocol.Request));
                method = ServerType.GetMethod(requestElement);
                if (Tracing.On) Tracing.Exit("RouteRequest", caller);
                if (method != null) {
                    if (ServerType.GetDuplicateMethod(requestElement) != null) {
                        method = null;
                        duplicateRequestElement = true;
                    }
                }
            }
 
            if (method == null) {
                // try to figure out what happened...
                if (action == null || action.Length == 0) {
                    // they didn't send a soap action and we couldn't route on request element
                    // require soap action for future requests:
                    throw new SoapException(Res.GetString(Res.UnableToHandleRequestActionRequired0), Soap12FaultCodes.SenderFaultCode);
                }
                else if (duplicateAction) {
                    // what went wrong with the request element?
                    if (duplicateRequestElement) {
                        // server's fault -- nothing the client could have done to prevent this
                        throw new SoapException(Res.GetString(Res.UnableToHandleRequest0), Soap12FaultCodes.ReceiverFaultCode);
                    }
                    else {
                        // probably client's fault -- we didn't recognize the request element they sent
                        throw new SoapException(Res.GetString(Res.TheRequestElementXmlnsWasNotRecognized2, requestElement.Name, requestElement.Namespace), Soap12FaultCodes.SenderFaultCode);
                    }
                }
                else {
                    // neither action nor request element worked out for us. since they sent an action,
                    // we'll suggest they do a better job next time
                    throw new SoapException(Res.GetString(Res.UnableToHandleRequestActionNotRecognized1, action), Soap12FaultCodes.SenderFaultCode);
                }
            }
 
            return method;
        }
 
        internal override void WriteFault(XmlWriter writer, SoapException soapException, HttpStatusCode statusCode) {
            if (statusCode != HttpStatusCode.InternalServerError)
                return;
            if (soapException == null)
                return;
            writer.WriteStartDocument();
            writer.WriteStartElement(Soap.Prefix, Soap.Element.Envelope, Soap12.Namespace);
            writer.WriteAttributeString("xmlns", Soap.Prefix, null, Soap12.Namespace);
            writer.WriteAttributeString("xmlns", "xsi", null, XmlSchema.InstanceNamespace);
            writer.WriteAttributeString("xmlns", "xsd", null, XmlSchema.Namespace);
            if (ServerProtocol.ServerMethod != null)
                SoapHeaderHandling.WriteHeaders(writer, ServerProtocol.ServerMethod.outHeaderSerializer, ServerProtocol.Message.Headers, ServerProtocol.ServerMethod.outHeaderMappings, SoapHeaderDirection.Fault, ServerProtocol.ServerMethod.use == SoapBindingUse.Encoded, ServerType.serviceNamespace, ServerType.serviceDefaultIsEncoded, Soap12.Namespace);
            else
                SoapHeaderHandling.WriteUnknownHeaders(writer, ServerProtocol.Message.Headers, Soap12.Namespace);
 
            writer.WriteStartElement(Soap.Element.Body, Soap12.Namespace);
            
            writer.WriteStartElement(Soap.Element.Fault, Soap12.Namespace);
            writer.WriteStartElement(Soap12.Element.FaultCode, Soap12.Namespace);
            WriteFaultCodeValue(writer, TranslateFaultCode(soapException.Code), soapException.SubCode);
            writer.WriteEndElement(); // </faultcode>
            
            writer.WriteStartElement(Soap12.Element.FaultReason, Soap12.Namespace);
            writer.WriteStartElement(Soap12.Element.FaultReasonText, Soap12.Namespace);
            writer.WriteAttributeString("xml", "lang", Soap.XmlNamespace, Res.GetString(Res.XmlLang));
            writer.WriteString(ServerProtocol.GenerateFaultString(soapException));
            writer.WriteEndElement(); // </Text>
            writer.WriteEndElement(); // </Reason>
            
            // Only write an actor element if the actor was specified (it's optional for end-points)
            string actor = soapException.Actor;
            if (actor.Length > 0)
                writer.WriteElementString(Soap12.Element.FaultNode, Soap12.Namespace, actor);
 
            string role = soapException.Role;
            if (role.Length > 0)
                writer.WriteElementString(Soap12.Element.FaultRole, Soap12.Namespace, role);
 
            // Only write a FaultDetail element if exception is related to the body (not a header)
            if (!(soapException is SoapHeaderException)) {
                if (soapException.Detail == null) {
                    writer.WriteStartElement(Soap12.Element.FaultDetail, Soap12.Namespace);
                    writer.WriteEndElement();
                }
                else {
                    soapException.Detail.WriteTo(writer);
                }
            }
            writer.WriteEndElement();
            
            writer.WriteEndElement();
            writer.WriteEndElement();
            writer.Flush();
        }
 
        private static void WriteFaultCodeValue(XmlWriter writer, XmlQualifiedName code, SoapFaultSubCode subcode) {
            if (code == null) return;
            writer.WriteStartElement(Soap12.Element.FaultCodeValue, Soap12.Namespace);
            if (code.Namespace != null && code.Namespace.Length > 0 && writer.LookupPrefix(code.Namespace) == null)
                writer.WriteAttributeString("xmlns", "q0", null, code.Namespace);
            writer.WriteQualifiedName(code.Name, code.Namespace);
            writer.WriteEndElement(); // </value>
            if (subcode != null) {
                writer.WriteStartElement(Soap12.Element.FaultSubcode, Soap12.Namespace);
                WriteFaultCodeValue(writer, subcode.Code, subcode.SubCode);
                writer.WriteEndElement(); // </subcode>
            }
        }
 
        private static XmlQualifiedName TranslateFaultCode(XmlQualifiedName code) {
            // note that we're allowing user-defined codes at the top-level here which technically
            // is not allowed by the soap 1.2 spec.
            
            if (code.Namespace == Soap.Namespace) {
                if (code.Name == Soap.Code.Server)
                    return Soap12FaultCodes.ReceiverFaultCode;
                else if (code.Name == Soap.Code.Client)
                    return Soap12FaultCodes.SenderFaultCode;
                else if (code.Name == Soap.Code.MustUnderstand)
                    return Soap12FaultCodes.MustUnderstandFaultCode;
                else if (code.Name == Soap.Code.VersionMismatch)
                    return Soap12FaultCodes.VersionMismatchFaultCode;
            }
            return code;
        }
        
    }
}