C# 解析WCF ProtocolException时System.ServiceModel.CommunicationException

C# 解析WCF ProtocolException时System.ServiceModel.CommunicationException,c#,wcf,soap,onvif,communicationexception,C#,Wcf,Soap,Onvif,Communicationexception,我尝试使用本页推荐的代码,如下所示: static FaultException ParseProtocolExecption(ProtocolException ex) { try { System.IO.Stream stream = (ex.InnerException as WebException).Response.GetResponseStream(); System.Xml.XmlReader xmr = System.Xml.XmlReader.Crea

我尝试使用本页推荐的代码,如下所示:

static FaultException ParseProtocolExecption(ProtocolException ex)
{
 try
 {
     System.IO.Stream stream = (ex.InnerException as WebException).Response.GetResponseStream();
     System.Xml.XmlReader xmr = System.Xml.XmlReader.Create(stream);
     Message message = Message.CreateMessage(xmr, (int)stream.Length, MessageVersion.Soap12);
     MessageFault mf = MessageFault.CreateFault(message, (int)stream.Length);
     FaultException fe = new FaultException(mf);
     message.Close();
     return fe;
 }
 catch (Exception)
 {
     return new FaultException(ex.Message);
 }
}
<?xml version="1.0" encoding="UTF-8"?>
<SOAP-ENV:Envelope xmlns:SOAP-ENV="http://www.w3.org/2003/05/soap-envelope" xmlns:SOAP-ENC="http://www.w3.org/2003/05/soap-encoding" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:wsa5="http://www.w3.org/2005/08/addressing" xmlns:c14n="http://www.w3.org/2001/10/xml-exc-c14n#" xmlns:wsu="http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-utility-1.0.xsd" xmlns:ds="http://www.w3.org/2000/09/xmldsig#" xmlns:wsse="http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-secext-1.0.xsd" xmlns:xmime5="http://www.w3.org/2005/05/xmlmime" xmlns:xop="http://www.w3.org/2004/08/xop/include" xmlns:tptz="http://www.onvif.org/ver20/ptz/wsdl" xmlns:tds2="http://www.onvif.org/ver10/schema" xmlns:tds3="http://docs.oasis-open.org/wsn/b-2" xmlns:tds4="http://docs.oasis-open.org/wsrf/bf-2" xmlns:tds5="http://docs.oasis-open.org/wsn/t-1" xmlns:timg="http://www.onvif.org/ver20/imaging/wsdl" xmlns:trt="http://www.onvif.org/ver10/media/wsdl" xmlns:tan="http://www.onvif.org/ver20/analytics/wsdl" xmlns:tds6="http://www.canon.com/ns/networkcamera/onvif/va/schema" xmlns:tmd="http://www.onvif.org/ver10/deviceIO/wsdl" xmlns:tev="http://www.onvif.org/ver10/events/wsdl" xmlns:tds9="http://docs.oasis-open.org/wsrf/r-2" xmlns:tds="http://www.onvif.org/ver10/device/wsdl" xmlns:ter="http://www.onvif.org/ver10/error">
  <SOAP-ENV:Header></SOAP-ENV:Header>
  <SOAP-ENV:Body>
    <SOAP-ENV:Fault>
      <SOAP-ENV:Code>
        <SOAP-ENV:Value>SOAP-ENV:Sender</SOAP-ENV:Value>
        <SOAP-ENV:Subcode>
          <SOAP-ENV:Value>ter:NotAuthorized</SOAP-ENV:Value>
        </SOAP-ENV:Subcode>
      </SOAP-ENV:Code>
      <SOAP-ENV:Reason>
        <SOAP-ENV:Text xml:lang="en">Sender Not Authorized</SOAP-ENV:Text>
      </SOAP-ENV:Reason>
      <SOAP-ENV:Node>http://www.w3.org/2003/05/soap-envelope/node/ultimateReceiver</SOAP-ENV:Node>
      <SOAP-ENV:Role>http://www.w3.org/2003/05/soap-envelope/role/ultimateReceiver</SOAP-ENV:Role>
      <SOAP-ENV:Detail>The action requested requires authorization and the sender is not authorized</SOAP-ENV:Detail>
    </SOAP-ENV:Fault>
  </SOAP-ENV:Body>
</SOAP-ENV:Envelope>
我正在使用VS 2012和使用WCF的.NET 4.5。当应用程序收到400个错误请求并将ProtocolException传递给ParseProtocolException时,它会在这一行抛出一个异常:

Message message = Message.CreateMessage(xmr, (int)stream.Length, MessageVersion.Soap12);
System.Xml.XmlReader xmr = System.Xml.XmlReader.Create(stream);
对于System.ServiceModel.CommunicationException:“缓冲XML内容所需的大小超过了缓冲区配额。”

stream.Length=2704字节,不是很大。我在这个网站上尝试了建议的解决方案。但是,即使MaxFaultSize=1 Mb,它也会得到相同的错误

而不是这一行:

Message message = Message.CreateMessage(xmr, (int)stream.Length, MessageVersion.Soap12);
System.Xml.XmlReader xmr = System.Xml.XmlReader.Create(stream);
我试过这个:

xmr = XmlDictionaryReader.CreateTextReader(stream, XmlDictionaryReaderQuotas.Max);
将所有配额设置为其最大值(Int32.MaxValue);但是,我在CreateMessage调用中仍然得到相同的错误

System.Net.WebException中的响应流示例如下所示:

static FaultException ParseProtocolExecption(ProtocolException ex)
{
 try
 {
     System.IO.Stream stream = (ex.InnerException as WebException).Response.GetResponseStream();
     System.Xml.XmlReader xmr = System.Xml.XmlReader.Create(stream);
     Message message = Message.CreateMessage(xmr, (int)stream.Length, MessageVersion.Soap12);
     MessageFault mf = MessageFault.CreateFault(message, (int)stream.Length);
     FaultException fe = new FaultException(mf);
     message.Close();
     return fe;
 }
 catch (Exception)
 {
     return new FaultException(ex.Message);
 }
}
<?xml version="1.0" encoding="UTF-8"?>
<SOAP-ENV:Envelope xmlns:SOAP-ENV="http://www.w3.org/2003/05/soap-envelope" xmlns:SOAP-ENC="http://www.w3.org/2003/05/soap-encoding" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:wsa5="http://www.w3.org/2005/08/addressing" xmlns:c14n="http://www.w3.org/2001/10/xml-exc-c14n#" xmlns:wsu="http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-utility-1.0.xsd" xmlns:ds="http://www.w3.org/2000/09/xmldsig#" xmlns:wsse="http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-secext-1.0.xsd" xmlns:xmime5="http://www.w3.org/2005/05/xmlmime" xmlns:xop="http://www.w3.org/2004/08/xop/include" xmlns:tptz="http://www.onvif.org/ver20/ptz/wsdl" xmlns:tds2="http://www.onvif.org/ver10/schema" xmlns:tds3="http://docs.oasis-open.org/wsn/b-2" xmlns:tds4="http://docs.oasis-open.org/wsrf/bf-2" xmlns:tds5="http://docs.oasis-open.org/wsn/t-1" xmlns:timg="http://www.onvif.org/ver20/imaging/wsdl" xmlns:trt="http://www.onvif.org/ver10/media/wsdl" xmlns:tan="http://www.onvif.org/ver20/analytics/wsdl" xmlns:tds6="http://www.canon.com/ns/networkcamera/onvif/va/schema" xmlns:tmd="http://www.onvif.org/ver10/deviceIO/wsdl" xmlns:tev="http://www.onvif.org/ver10/events/wsdl" xmlns:tds9="http://docs.oasis-open.org/wsrf/r-2" xmlns:tds="http://www.onvif.org/ver10/device/wsdl" xmlns:ter="http://www.onvif.org/ver10/error">
  <SOAP-ENV:Header></SOAP-ENV:Header>
  <SOAP-ENV:Body>
    <SOAP-ENV:Fault>
      <SOAP-ENV:Code>
        <SOAP-ENV:Value>SOAP-ENV:Sender</SOAP-ENV:Value>
        <SOAP-ENV:Subcode>
          <SOAP-ENV:Value>ter:NotAuthorized</SOAP-ENV:Value>
        </SOAP-ENV:Subcode>
      </SOAP-ENV:Code>
      <SOAP-ENV:Reason>
        <SOAP-ENV:Text xml:lang="en">Sender Not Authorized</SOAP-ENV:Text>
      </SOAP-ENV:Reason>
      <SOAP-ENV:Node>http://www.w3.org/2003/05/soap-envelope/node/ultimateReceiver</SOAP-ENV:Node>
      <SOAP-ENV:Role>http://www.w3.org/2003/05/soap-envelope/role/ultimateReceiver</SOAP-ENV:Role>
      <SOAP-ENV:Detail>The action requested requires authorization and the sender is not authorized</SOAP-ENV:Detail>
    </SOAP-ENV:Fault>
  </SOAP-ENV:Body>
</SOAP-ENV:Envelope>

SOAP-ENV:发送方
三:未授权
发件人未经授权
http://www.w3.org/2003/05/soap-envelope/node/ultimateReceiver
http://www.w3.org/2003/05/soap-envelope/role/ultimateReceiver
请求的操作需要授权,而发件人未经授权
使用我在F#Interactive中编写的异步web爬虫,我发现一些名称空间url无法解析。我更正了错误的页面,然后再次运行爬虫程序来计算名称空间页面的长度。总字节数为715965字节,远小于XmlDictionaryReaderQuotas中所有配额的Int32.MaxValue。也许XmlDictionaryReader有一个bug,或者它返回的错误不是真正的问题

通过删除SOAP-ENV:Body中没有实际使用的名称空间定义(即,仅保留子代码元素中使用的xmlns:ter),我最终实现了消息创建。但是,当然,这并不能真正解决问题,因为服务正在生成SOAP错误;而且我无法更改服务实现(它是第三方设备-Onvif摄像头)


而且,我不能再增加配额了;那么,如何处理此异常呢?

解决方案是捕获CommunicationException,然后对流中不需要名称空间解析的XML进行另一种解析:

    public static FaultException ParseProtocolException(System.ServiceModel.ProtocolException ex) {
        var stream = (ex.InnerException as System.Net.WebException).Response.GetResponseStream();
        try {
            var xmr = XmlReader.Create(stream);
            var message = Message.CreateMessage(xmr, (int)stream.Length, MessageVersion.Soap12);
            var mf = MessageFault.CreateFault(message, (int)stream.Length);
            message.Close();
            return new FaultException(mf);
        } catch (CommunicationException) {    // If CreateMessage has a problem parsing the XML,
            // then this error will be thrown.  Most likely, there is an unresolvable namespace reference.
            // Do an alternate parse
            stream.Seek(0, System.IO.SeekOrigin.Begin);
            var soapFault = GetSoapFault(stream);
            return new FaultException(soapFault.Reason);
        }
    }
catch将流重置为开头,然后使用以下内容使用XmlReader获取SOAP错误的详细信息:

    private struct SoapFault {
        public string Subcode;
        public string Reason;
        public string Detail;
    }
    private static string GetTextChild(XmlReader xmr, string childName) {
      return xmr.ReadToDescendant(childName) ? 
          xmr.ReadString() : System.String.Empty;
    }
    private static SoapFault GetSoapFault(System.IO.Stream s) {
        var xr = XmlReader.Create(s);
        var fault = new SoapFault();
        if (xr.ReadToFollowing("SOAP-ENV:Subcode")) {
            fault.Subcode = GetTextChild(xr, "SOAP-ENV:Value");
            if (xr.ReadToFollowing("SOAP-ENV:Reason")) {
                fault.Reason = GetTextChild(xr, "SOAP-ENV:Text");
                if (xr.ReadToFollowing("SOAP-ENV:Detail"))
                    fault.Detail = GetTextChild(xr, "SOAP-ENV:Text");
            }
        }
        return fault;
    }