сгенерированный do tnet -svcutil прокси WCF (и даже пользовательский канал <>) возвращают ноль для правильного ответа SOAP - PullRequest
3 голосов
/ 21 февраля 2020

Прокси, сгенерированный с помощью do tnet -svcutil 2.0.1 (dotnet-svcutil --sync --outputDir . http://XXX/?WSDL) и System.ServiceModel. * 4.7.0, код вызова и? WSDL ниже. Прокси просто не может десериализовать действительный ответ и просто возвращает ноль. Пробовал оба. NET Core 3.0 и 3.1 на Windows 10 и macOS Catalina, тот же нулевой результат. Запрос и ответ Fiddler вместе с? WSDL от сервера (сервер вне моего контроля).

С прокси я использую обходной путь @ shmao set_mode (https://github.com/dotnet/wcf/issues/2219), чтобы избежать исключение «сценарии JScript / CSharp не поддерживается». Кроме того, я должен удалить атрибуты Namespace="", чтобы заставить работать даже часть запроса. Я добавил EventListeners и выгрузил все в Verbose из всех источников событий, без предупреждений / ошибок, только с нулевым значением.

Я также попробовал подход MessageContract / DataContract на основе каналов, в конечном итоге тот же нулевой результат (!), Я не могу использовать какой-либо. NET Базовый код на основе WCF для десериализации данного ответа.

Любое решение или даже частичное решение для десериализации данного ответа с использованием. NET Core 3.1 WCF, в идеале будет рассматриваться с do tnet -svcutil. Раскрытие предупреждения / ошибки или даже получение доступа к строке ответа вручную все равно будет улучшением по сравнению с подходом, не основанным на строке WCF / HttpRequest.

      WSWebServiceSoapPortClient proxy;
      try {
        proxy = new WSWebServiceSoapPortClient(new BasicHttpBinding(),
          new EndpointAddress("http://XXX"));

        await proxy.OpenAsync();
      } catch (Exception e) {
        Console.WriteLine(e.Message);
        return;
      }

      if (proxy.State == System.ServiceModel.CommunicationState.Faulted) {
        System.Console.WriteLine("Unable to connect to the proxy.");
        return;
      }

      var one = new WSUserLoginRequest1(new WSUserLoginRequest() {
        userName = "XXX",
        userPassword = "XXX",
      });
      WSUserLoginResponse1 wsUserLoginResponse = null;

      try {
        wsUserLoginResponse = await proxy.WSUserLoginAsync(one);   // returns null
      } catch (Exception e) {
        Console.WriteLine(e.ToString());
        return;
      }

Соответствующий WSDL с сервера

<?xml version="1.0" encoding="UTF-8"?>
<wsdl:definitions xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/" 
  xmlns:http="http://schemas.xmlsoap.org/wsdl/http/" 
  xmlns:mime="http://schemas.xmlsoap.org/wsdl/mime/" 
  xmlns:s="http://www.w3.org/2001/XMLSchema" 
  xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/" 
  xmlns:soap12="http://schemas.xmlsoap.org/wsdl/soap12/" 
  xmlns:soapenc="http://schemas.xmlsoap.org/soap/encoding/" 
  xmlns:tm="http://microsoft.com/wsdl/mime/textMatching/" 
  xmlns:tns="WSWebService" name="WSWebService" targetNamespace="WSWebService">
  <wsdl:types>
    <schema xmlns="http://www.w3.org/2001/XMLSchema" targetNamespace="WSWebService">

      ...

      <s:complexType name="WSUserLoginRequest">
        <s:sequence>
          <s:element name="userName" type="s:string" />
          <s:element name="userPassword" type="s:string" />
        </s:sequence>
      </s:complexType>
      <s:complexType name="WSUserLoginResponse">
        <s:sequence>
          <s:element name="userToken" type="s:string" />
          <s:element name="wsdlVersion" type="s:string" minOccurs="1" maxOccurs="1" default="2.0.0.0" />
          <s:element name="result" type="s:int" />
          <s:element name="resultString" type="s:string" />
        </s:sequence>
      </s:complexType>

      ...

      <wsdl:message name="WSUserLoginSoapIn">
        <wsdl:part name="parameters" type="tns:WSUserLoginRequest" />
      </wsdl:message>
      <wsdl:message name="WSUserLoginSoapOut">
        <wsdl:part name="parameters" type="tns:WSUserLoginResponse" />
      </wsdl:message>

      ...

      <wsdl:operation name="WSUserLogin">
        <wsdl:documentation>Authenticate user using provided username and password.</wsdl:documentation>
        <wsdl:input message="tns:WSUserLoginSoapIn" />
        <wsdl:output message="tns:WSUserLoginSoapOut" />
      </wsdl:operation>

request response

(РЕДАКТИРОВАТЬ)

Определение класса для WSUserLoginResponse1, сгенерированное dotnet-svcutil 2.0.1:

[System.Diagnostics.DebuggerStepThroughAttribute()]
[System.CodeDom.Compiler.GeneratedCodeAttribute("Microsoft.Tools.ServiceModel.Svcutil", "2.0.1")]
[System.ServiceModel.MessageContractAttribute(WrapperName = "WSUserLoginResponse", WrapperNamespace = "WSWebService", IsWrapped = true)]
public partial class WSUserLoginResponse1 {

  [System.ServiceModel.MessageBodyMemberAttribute(Namespace = "", Order = 0)]
  public WSUserLoginResponse parameters;

  public WSUserLoginResponse1() {
  }

  public WSUserLoginResponse1(WSUserLoginResponse parameters) {
    this.parameters = parameters;
  }
}

(РЕДАКТИРОВАТЬ 2) WSUserLoginResponse от do tnet -svcutil как предложено.

/// <remarks/>
[System.CodeDom.Compiler.GeneratedCodeAttribute("Microsoft.Tools.ServiceModel.Svcutil", "2.0.1")]
[System.Diagnostics.DebuggerStepThroughAttribute()]
[System.Xml.Serialization.XmlTypeAttribute(Namespace = "WSWebService")]
public partial class WSUserLoginResponse {

  private string userTokenField;

  private string wsdlVersionField;

  private int resultField;

  private string resultStringField;

  public WSUserLoginResponse() {
    this.wsdlVersionField = "2.0.0.0";
  }

  /// <remarks/>
  [System.Xml.Serialization.XmlElementAttribute(Form = System.Xml.Schema.XmlSchemaForm.Unqualified, Order = 0)]
  public string userToken {
    get {
      return this.userTokenField;
    }
    set {
      this.userTokenField = value;
    }
  }

  /// <remarks/>
  [System.Xml.Serialization.XmlElementAttribute(Form = System.Xml.Schema.XmlSchemaForm.Unqualified, Order = 1)]
  public string wsdlVersion {
    get {
      return this.wsdlVersionField;
    }
    set {
      this.wsdlVersionField = value;
    }
  }

  /// <remarks/>
  [System.Xml.Serialization.XmlElementAttribute(Form = System.Xml.Schema.XmlSchemaForm.Unqualified, Order = 2)]
  public int result {
    get {
      return this.resultField;
    }
    set {
      this.resultField = value;
    }
  }

  /// <remarks/>
  [System.Xml.Serialization.XmlElementAttribute(Form = System.Xml.Schema.XmlSchemaForm.Unqualified, Order = 3)]
  public string resultString {
    get {
      return this.resultStringField;
    }
    set {
      this.resultStringField = value;
    }
  }
}

(EDIT 3) WSUserLoginResponse от wsdl.exe как предложено. Нет WSUserLoginResponse1

    [System.CodeDom.Compiler.GeneratedCodeAttribute("System.Xml", "4.6.1087.0")]
    [System.SerializableAttribute()]
    [System.Diagnostics.DebuggerStepThroughAttribute()]
    [System.ComponentModel.DesignerCategoryAttribute("code")]
    [System.Xml.Serialization.XmlTypeAttribute(Namespace="WSWebService")]
    public partial class WSUserLoginResponse : object, System.ComponentModel.INotifyPropertyChanged {

        private string userTokenField;

        private string wsdlVersionField;

        private int resultField;

        private string resultStringField;

        public WSUserLoginResponse() {
            this.wsdlVersionField = "2.0.0.0";
        }

        /// <remarks/>
        [System.Xml.Serialization.XmlElementAttribute(Form=System.Xml.Schema.XmlSchemaForm.Unqualified, Order=0)]
        public string userToken {
            get {
                return this.userTokenField;
            }
            set {
                this.userTokenField = value;
                this.RaisePropertyChanged("userToken");
            }
        }

        /// <remarks/>
        [System.Xml.Serialization.XmlElementAttribute(Form=System.Xml.Schema.XmlSchemaForm.Unqualified, Order=1)]
        public string wsdlVersion {
            get {
                return this.wsdlVersionField;
            }
            set {
                this.wsdlVersionField = value;
                this.RaisePropertyChanged("wsdlVersion");
            }
        }

        /// <remarks/>
        [System.Xml.Serialization.XmlElementAttribute(Form=System.Xml.Schema.XmlSchemaForm.Unqualified, Order=2)]
        public int result {
            get {
                return this.resultField;
            }
            set {
                this.resultField = value;
                this.RaisePropertyChanged("result");
            }
        }

        /// <remarks/>
        [System.Xml.Serialization.XmlElementAttribute(Form=System.Xml.Schema.XmlSchemaForm.Unqualified, Order=3)]
        public string resultString {
            get {
                return this.resultStringField;
            }
            set {
                this.resultStringField = value;
                this.RaisePropertyChanged("resultString");
            }
        }

        public event System.ComponentModel.PropertyChangedEventHandler PropertyChanged;

        protected void RaisePropertyChanged(string propertyName) {
            System.ComponentModel.PropertyChangedEventHandler propertyChanged = this.PropertyChanged;
            if ((propertyChanged != null)) {
                propertyChanged(this, new System.ComponentModel.PropertyChangedEventArgs(propertyName));
            }
        }
    }

(РЕДАКТИРОВАТЬ 4) Автономный WSDL.

<?xml version="1.0" encoding="UTF-8"?>
<wsdl:definitions xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/" 
  xmlns:http="http://schemas.xmlsoap.org/wsdl/http/" 
  xmlns:mime="http://schemas.xmlsoap.org/wsdl/mime/" 
  xmlns:s="http://www.w3.org/2001/XMLSchema" 
  xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/" 
  xmlns:soap12="http://schemas.xmlsoap.org/wsdl/soap12/" 
  xmlns:soapenc="http://schemas.xmlsoap.org/soap/encoding/" 
  xmlns:tm="http://microsoft.com/wsdl/mime/textMatching/" 
  xmlns:tns="WSWebService" name="WSWebService" targetNamespace="WSWebService">
  <wsdl:types>
    <schema xmlns="http://www.w3.org/2001/XMLSchema" targetNamespace="WSWebService">
      <s:complexType name="WSUserLoginRequest">
        <s:sequence>
          <s:element name="userName" type="s:string" />
          <s:element name="userPassword" type="s:string" />
        </s:sequence>
      </s:complexType>
      <s:complexType name="WSUserLoginResponse">
        <s:sequence>
          <s:element name="userToken" type="s:string" />
          <s:element name="wsdlVersion" type="s:string" minOccurs="1" maxOccurs="1" default="2.0.0.0" />
          <s:element name="result" type="s:int" />
          <s:element name="resultString" type="s:string" />
        </s:sequence>
      </s:complexType>
    </schema>
  </wsdl:types>
  <wsdl:message name="WSUserLoginSoapIn">
    <wsdl:part name="parameters" type="tns:WSUserLoginRequest" />
  </wsdl:message>
  <wsdl:message name="WSUserLoginSoapOut">
    <wsdl:part name="parameters" type="tns:WSUserLoginResponse" />
  </wsdl:message>
  <wsdl:portType name="WSWebServiceSoapPort">
    <wsdl:operation name="WSUserLogin">
      <wsdl:documentation>Documentation</wsdl:documentation>
      <wsdl:input message="tns:WSUserLoginSoapIn" />
      <wsdl:output message="tns:WSUserLoginSoapOut" />
    </wsdl:operation>
  </wsdl:portType>
  <wsdl:binding name="WSWebServiceSoapBinding" type="tns:WSWebServiceSoapPort">
    <soap:binding style="rpc" transport="http://schemas.xmlsoap.org/soap/http" />
    <wsdl:operation name="WSUserLogin">
      <soap:operation soapAction="WSUserLogin" style="rpc" />
      <wsdl:input>
        <soap:body use="literal" namespace="WSWebService" />
      </wsdl:input>
      <wsdl:output>
        <soap:body use="literal" namespace="WSWebService" />
      </wsdl:output>
    </wsdl:operation>
  </wsdl:binding>
  <wsdl:service name="WSWebService">
    <wsdl:port name="WSWebServiceSoapPort" binding="tns:WSWebServiceSoapBinding">
      <soap:address location="https://x.x.x.x:x" />
    </wsdl:port>
  </wsdl:service>
</wsdl:definitions>

1 Ответ

2 голосов
/ 01 марта 2020

Я использовал ваш файл wsdl для генерации кода c#, используя dotnet-svcutil 2.0.1, как вы это сделали.

Я высмеял конечную точку SOAP, используя Сервис SoapUI mocking , запустил его, и затем запустил ваш код на моей машине (подключаясь к поддельной конечной точке на моем локальном хосте).

Я рано получил исключение при вызове proxy.OpenAsync. Сообщение об исключении, которое я получил, было:

Верхний XML элемент 'параметры' из пространства имен '' ссылается на различные типы WSUserLoginRequest и WSUserLoginResponse. Используйте атрибуты XML, чтобы указать другое XML имя или пространство имен для элемента или типов.

Итак, я перешел к WSUserLoginRequest1 внутри сгенерированного кода и добавил значение к Namespace parameters, как объяснено здесь :

[System.Diagnostics.DebuggerStepThroughAttribute()]
[System.CodeDom.Compiler.GeneratedCodeAttribute("Microsoft.Tools.ServiceModel.Svcutil", "2.0.1")]
[System.ServiceModel.MessageContractAttribute(WrapperName="WSUserLogin", WrapperNamespace="WSWebService", IsWrapped=true)]
public partial class WSUserLoginRequest1
{
    // replaced Namespace empty string value with my endpoint url
    [System.ServiceModel.MessageBodyMemberAttribute(Namespace="http://localhost:8181/WSUserLogin", Order=0)]
    public WSUserLoginRequest parameters;

    public WSUserLoginRequest1()
    {
    }

    // ...
}

После этого изменения оно просто сработало, и я получил ответ при вызове proxy.WSUserLoginAsync.

I wi sh Я знал, как добавить это значение Namespace, передав значения в параметр /namespace, равный dotnet-svcutil, но следуя этим сообщениям SO ( 1 , 2 , хотя они относятся к svcutil, а не dotnet-svcutil) только к добавленным пространствам имен для других мест в сгенерированном файле.

Кроме того, как вы заметили, использование wsdl.exe не генерирует ненужные WSUserLoginResponse1, в то время как svcutil-dotnet делает. Кажется, это известная проблема, см., Например, здесь: svcutil сгенерировал ненужные классы-оболочки .

Я надеюсь, что она будет иметь какое-то значение для вашей проблемы.

...