我正在使用C#从.NET 4.5调用SOAP Web服务,但我不明白如何捕捉未类型化的细节FaultException。
如果Web服务遇到错误,我将收到如下消息:
<S:Envelope xmlns:S="http://schemas.xmlsoap.org/soap/envelope/">
<S:Body>
<S:Fault xmlns:ns4="http://www.w3.org/2003/05/soap-envelope">
<faultcode>GSX.SYS.003</faultcode>
<faultstring>Multiple error messages exist. Please check the detail section.</faultstring>
<detail>
<operationId>HqrCoRMbtKczWFH2WMuFJGe</operationId>
<errors>
<error>
<code>ENT.UPL.005</code>
<message>User ID is required for authentication.</message>
</error>
<error>
<code>ENT.UPL.005</code>
<message>Password is required for authentication.</message>
</error>
<error>
<code>ENT.UPL.005</code>
<message>Sold-To is required for authentication.</message>
</error>
</errors>
</detail>
</S:Fault>
</S:Body>
</S:Envelope>
Run Code Online (Sandbox Code Playgroud)
为了捕获错误,我将调用包装在try / catch块中并捕获了FaultException:
try
{
// Do something.
}
catch (FaultException faultException)
{
var messageFault = faultException.CreateMessageFault();
// ???
throw;
}
Run Code Online (Sandbox Code Playgroud)
问题就在于这// ???行:如果由于FaultException未messageFault.GetDetail<T>()键入细节而无法使用,我该如何访问的细节?
对于代码和我没有问题的原因,我可以通过messageFault.Code.Name和来获得它们messageFault.Reason.Translations.Single().Text。
我能够炮制的是这样的:
var stringWriter = new StringWriter();
var xmlTextWriter = new XmlTextWriter(stringWriter);
var messageFault = faultException.CreateMessageFault();
messageFault.WriteTo(xmlTextWriter, EnvelopeVersion.Soap12);
var stringValue = Convert.ToString(stringWriter);
var nameTable = new NameTable();
var xmlNamespaceManager = new XmlNamespaceManager(nameTable);
xmlNamespaceManager.AddNamespace("soap", "http://www.w3.org/2003/05/soap-envelope");
var xmlDocument = XDocument.Parse(stringValue);
var operationId =
xmlDocument
.XPathSelectElement("/soap:Fault/soap:Detail/operationId", xmlNamespaceManager)
.Value;
var errors =
xmlDocument
.XPathSelectElements("/soap:Fault/soap:Detail/errors/error", xmlNamespaceManager)
.Select(element => new
{
Code = element.XPathSelectElement("code").Value,
Message = element.XPathSelectElement("message").Value
})
.ToArray();
Run Code Online (Sandbox Code Playgroud)
但是除了空引用和编码问题之外,这是一个怪物。
还有必须是一个更理智的方式来获得详细信息。
您可以读取 XML 类型的详细信息,例如。XElement或XmlElement并使用 XPath、Linq to XML 对其进行处理,或者仅将其视为字符串。
在类似的情况下,我使用以下代码来获取公认的简单详细信息元素的内容:
var detail = fault.GetDetail<XElement>();
return detail.Value;
Run Code Online (Sandbox Code Playgroud)