2

我们正在连接到一个 Web 服务,我们得到的错误消息没有反序列化(根本没有),而且我可以制作的任何版本的类都不会正确反序列化。我们无法控制服务器端的事情。服务器不允许发现,因此将 ?WSDL 添加到端点 URL 的末尾会导致错误,而不是 WSDL。

[Fiddler][1] 显示返回的故障消息如下所示:

<?xml version="1.0" encoding="UTF-8"?>
<soapenv:Envelope xmlns:eGov="http://eGov.gov" xmlns:soapenv="http://schemas.xmlsoap.org/soap/envelope/">
  <soapenv:Body>
    <soapenv:Fault>
    <faultcode>Client</faultcode>
    <faultstring/>
      <detail>
        <eGov:eGov2Exception>
          <eGov:ErrorClassification>SOME_ERROR</eGov:ErrorClassification>
          <eGov:ErrorCode>SOME_ERROR_CODE</eGov:ErrorCode>
          <eGov:ErrorMessage>Your request was unsuccessful. blah blah blah.</eGov:ErrorMessage>
        </eGov:eGov2Exception>
      </detail>
    </soapenv:Fault>
  </soapenv:Body>
</soapenv:Envelope>

然而,当我们尝试捕获它时,我们创建的任何类(尝试 xsd.exe、svcutil 和其他包括我们从头编写的代码)都无法反序列化它:

catch (FaultException<eGov2ExceptionType> exp)
  {
     // Never stops here. 
  }
catch (FaultException<AllOtherAttemptedClasses> exp)
  {
     // Never stops here. 
  }
catch (SoapException se)
  {
     // Never stops here. 
  }
catch (FaultException exp)
  {
     //Always gets caught here. 
  }

只有基本的 FaultException catch 会被调用,这意味着我们丢失了正在发送的 FaultMessage 的内容。我编写的一些类将序列化非常接近上面的示例,但无法反序列化它,所以我们怀疑存在命名空间问题。

问题:

1 - 你会怎么写这个?

2 - 这是 WCF 的常见错误/问题吗?

[1]: http://www.fiddler2.com/fiddler2/提琴手

4

3 回答 3

4

我们最终做的是放弃尝试捕获故障并将其传递回 SOAP 通道。相反,我们创建了一个自定义异常,并连接了一个MessageInspector来监视错误并将其作为异常抛出。

(已清理)代码的相关部分:

   public void AfterReceiveReply(ref Message reply, object correlationState)
    {
        if (reply.IsFault)
        {
            XmlDictionaryReader xdr = reply.GetReaderAtBodyContents();
            XNode xn = XDocument.ReadFrom(xdr);
            string s = xn.ToString();
            XDocument xd = XDocument.Parse(s);
            XNamespace nsSoap = "http://schemas.xmlsoap.org/soap/envelope/";
            XNamespace ns = "http://eGov.gov";
            XElement xErrorClass = xd.Element(nsSoap + "Fault").Element("detail").Element(ns + "eGov2Exception").Element(ns + "RequestErrorClassification");
            XElement xErrorCode = xd.Element(nsSoap + "Fault").Element("detail").Element(ns + "eGov2Exception").Element(ns + "RequestErrorCode");
            XElement xErrorMessage = xd.Element(nsSoap + "Fault").Element("detail").Element(ns + "eGov2Exception").Element(ns + "RequestErrorMessage");

            throw new eGovException(xErrorClass.Value, xErrorCode.Value, xErrorMessage.Value);
        }
    }

然后主应用程序使用:

catch (eGovException ex)
{
// 在这里处理异常。
}

尝试更正名称空间浪费了太多时间。谢谢回答。

于 2009-09-25T18:03:06.543 回答
1

如果序列化完全失败,处理此问题的一种方法是使用 OoperationContract 使用Message类型作为输入和输出。这样,您可以在 Iffault == true 时手动解析 XML,或者如果没有发生错误,则使用 GetBody() 获取常规内容。

于 2009-09-18T08:02:10.770 回答
1

首先,如果您无法从制作此 Web 服务的人那里获得 WSDL,那么他们就没有必要拥有基于 SOAP 的 Web 服务。无论是否使用“?WSDL”,正确的 WSDL 都可以解决您的问题。

其次,请张贴eGov2ExceptionType班级的代码。我怀疑它没有http://eGov.gov设置命名空间。

于 2009-09-19T20:19:10.390 回答