9

私は、FaultException と FaultException<T> を使用して、アプリケーションでの最適な使用パターンを判断しようとしています。SOAP 1.1 クライアントや SOAP 1.2 クライアントなど、WCF だけでなく、WCF 以外のサービス コンシューマー/クライアントもサポートする必要があります。

参考までに: FaultExceptions を wsHttpBinding で使用すると SOAP 1.2 セマンティクスになりますが、FaultExceptions を basicHttpBinding で使用すると SOAP 1.1 セマンティクスになります。

次のコードを使用して、FaultException<FaultDetails> をスローしています。

  throw new FaultException<FaultDetails>(
      new FaultDetails("Throwing FaultException<FaultDetails>."),
      new FaultReason("Testing fault exceptions."),
      FaultCode.CreateSenderFaultCode(new FaultCode("MySubFaultCode"))
      );

FaultDetails クラスは、以下に示すように、文字列 "Message" プロパティを含む単純なテスト クラスです。

wsHttpBinding を使用する場合の応答は次のとおりです。

<?xml version="1.0" encoding="utf-16"?>
<Fault xmlns="http://www.w3.org/2003/05/soap-envelope">
<Code>
  <Value>Sender</Value>
  <Subcode>
    <Value>MySubFaultCode</Value>
  </Subcode>
</Code>
<Reason>
  <Text xml:lang="en-US">Testing fault exceptions.</Text>
</Reason>
<Detail>
  <FaultDetails xmlns="http://schemas.datacontract.org/2004/07/ClassLibrary" xmlns:i="http://www.w3.org/2001/XMLSchema-instance">
    <Message>Throwing FaultException&lt;FaultDetails&gt;.</Message>
  </FaultDetails>
</Detail>

これは、SOAP 1.2 仕様によると正しく見えます。メイン/ルートの「コード」は「送信者」で、「サブコード」は「MySubFaultCode」です。サービス コンシューマー/クライアントが WCF を使用している場合、クライアント側の FaultException も同じ構造を模倣し、faultException.Code.Name は「Sender」、faultException.Code.SubCode.Name は「MySubFaultCode」になります。

basicHttpBinding を使用する場合の応答は次のとおりです。

<?xml version="1.0" encoding="utf-16"?>
<s:Fault xmlns:s="http://schemas.xmlsoap.org/soap/envelope/">
  <faultcode>s:MySubFaultCode</faultcode>
  <faultstring xml:lang="en-US">Testing fault exceptions.</faultstring>
  <detail>
    <FaultDetails xmlns="http://schemas.datacontract.org/2004/07/ClassLibrary" xmlns:i="http://www.w3.org/2001/XMLSchema-instance">
      <Message>Throwing FaultException&lt;FaultDetails&gt;.</Message>
    </FaultDetails>
  </detail>
</s:Fault>

これは正しくありません。SOAP 1.1 の仕様を見て、FaultCode.CreateSenderFaultCode(new FaultCode("MySubFaultCode")) を使用すると、「faultcode」の値が「s:Client.MySubFaultCode」になると予想していました。また、WCF クライアントは正しくない構造を取得します。faultException.Code.Name は「Sender」ではなく「MySubFaultCode」であり、faultException.Code.SubCode は faultException.Code.SubCode.Name が「MySubFaultCode」ではなく null です。また、faultException.Code.IsSenderFault は false です。

FaultCode.CreateReceiverFaultCode(new FaultCode("MySubFaultCode")) を使用した場合の同様の問題:

  • SOAP 1.2 では期待どおりに動作します
  • 「s:Server.MySubFaultCode」の代わりに「s:MySubFaultCode」を生成し、SOAP 1.1 の faultException.Code.IsReceiverFault は false です。

この項目は、2006 年にhttp://forums.microsoft.com/MSDN/ShowPost.aspx?PostID=669420&SiteID=1に他の人によって投稿されたもので、誰も回答していません。まだ誰もこれに遭遇していないとは信じがたいです。

同様の問題を抱えている他の人がいます:http://forums.microsoft.com/msdn/ShowPost.aspx?PostID=3883110&SiteID=1&mode=1

Microsoft Connect のバグ: https://connect.microsoft.com/wcf/feedback/ViewFeedback.aspx?FeedbackID=367963

障害がどのように機能するかの説明: http://blogs.msdn.com/drnick/archive/2006/12/19/creating-faults-part-3.aspx

私は何か間違ったことをしていますか、それとも本当に WCF のバグですか?

4

2 に答える 2

8

これは私の現在の回避策です:

    /// <summary>
    /// Replacement for the static methods on FaultCode to generate Sender and Receiver fault codes due
    /// to what seems like bugs in the implementation for basicHttpBinding (SOAP 1.1). wsHttpBinding 
    /// (SOAP 1.2) seems to work just fine.
    /// 
    /// The subCode parameter for FaultCode.CreateReceiverFaultCode and FaultCode.CreateSenderFaultCode
    /// seem to take over the main 'faultcode' value in the SOAP 1.1 response, whereas in SOAP 1.2 the
    /// subCode is correctly put under the 'Code->SubCode->Value' value in the XML response.
    /// 
    /// This workaround is to create the FaultCode with Sender/Receiver (SOAP 1.2 terms, but gets
    /// translated by WCF depending on the binding) and an agnostic namespace found by using reflector
    /// on the FaultCode class. When that NS is passed in WCF seems to be able to generate the proper
    /// response with SOAP 1.1 (Client/Server) and SOAP 1.2 (Sender/Receiver) fault codes automatically.
    /// 
    /// This means that it is not possible to create a FaultCode that works in both bindings with
    /// subcodes.
    /// </summary>
    /// <remarks>
    /// See http://stackoverflow.com/questions/65008/net-wcf-faults-generating-incorrect-soap-11-faultcode-values
    /// for more details.
    /// </remarks>
    public static class FaultCodeFactory
    {
        private const string _ns = "http://schemas.microsoft.com/ws/2005/05/envelope/none";

        /// <summary>
        /// Creates a sender fault code.
        /// </summary>
        /// <returns>A FaultCode object.</returns>
        /// <remarks>Does not support subcodes due to a WCF bug.</remarks>
        public static FaultCode CreateSenderFaultCode()
        {
            return new FaultCode("Sender", _ns);
        }

        /// <summary>
        /// Creates a receiver fault code.
        /// </summary>
        /// <returns>A FaultCode object.</returns>
        /// <remarks>Does not support subcodes due to a WCF bug.</remarks>
        public static FaultCode CreateReceiverFaultCode()
        {
            return new FaultCode("Receiver", _ns);
        }
    }

残念ながら、SOAP1.1または1.2クライアントを壊さずにサブコードを使用する方法がわかりません。

Code.SubCode構文を使用すると、SOAP 1.1互換のフォールトコード値を作成できますが、SOAP1.2が機能しなくなります。

.NETで適切なサブコードサポートを使用すると(静的なFaultCodeメソッドまたはオーバーロードの1つを介して)、SOAP 1.1は機能しなくなりますが、SOAP1.2では機能します。

于 2008-09-16T04:29:38.797 に答える
7

マイクロソフトからの応答:

http://msdn.microsoft.com/en-us/library/ms789039.aspxで説明されているように、カスタム エラー コードの Soap 1.1 仕様で概説されている 2 つの方法があります。

(1)あなたが説明するように「ドット」表記を使用する

(2) まったく新しいフォルトコードの定義

残念ながら、WS-I Basic Profile 仕様では使用が推奨されていないため、「ドット」表記は避ける必要があります。基本的に、これは、Soap 1.1 を使用する場合、Soap 1.2 のフォルト サブコードに相当するものがないことを意味します。

そのため、障害を生成するときは、バインディングで定義されている MessageVersion を認識し、それに応じて障害コードを生成する必要があります。

「送信者」と「受信者」は Soap 1.1 の有効な障害コードではなく、障害サブコードに相当するものがないため、Soap 1.1 のカスタム障害コードを生成するときに CreateSenderFaultCode メソッドと CreateReceiverFaultCode メソッドを使用しないでください。

代わりに、独自の名前空間と名前を使用して、独自の faultcode を定義する必要があります。

FaultCode customFaultCode = new FaultCode(localName, faultNamespace);

于 2008-10-15T22:49:20.167 に答える