CXF: org.apache.cxf.interceptor.Fault: неверное количество аргументов при вызове

Я реализую простой клиент, используя apache cxf и плагин codegen maven (оба выпуска 3.1.11).

Вот мой пример wsdl:

    <?xml version="1.0" encoding="UTF-8"?>
<wsdl:definitions targetNamespace="urn:webServiceTest" xmlns:s0="urn:webServiceTest" xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/" xmlns:soapenc="http://schemas.xmlsoap.org/soap/encoding/" xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
 <wsdl:types>
  <xsd:schema elementFormDefault="qualified" targetNamespace="urn:webServiceTest">
   <xsd:element name="OpenTk" type="s0:InputMapping1"/>
   <xsd:complexType name="InputMapping1">
    <xsd:sequence>
     <xsd:element minOccurs="0" name="ID_NOTIFICA" type="xsd:string"/>
     <xsd:element minOccurs="0" name="ID_MESSAGGIO" type="xsd:string"/>
     <xsd:element minOccurs="0" name="DATA_ID_NOTIFICA" type="xsd:dateTime"/>
     <xsd:element minOccurs="0" name="DATA_ID_MESSAGGIO" type="xsd:dateTime"/>
     <xsd:element minOccurs="0" name="ID_RISORSA" type="xsd:string"/>
    </xsd:sequence>
   </xsd:complexType>
   <xsd:element name="APERTURATKResponse" type="s0:OutputMapping1"/>
   <xsd:complexType name="OutputMapping1">
    <xsd:sequence>
     <xsd:element minOccurs="0" name="CODICE" type="xsd:string"/>
     <xsd:element minOccurs="0" name="DESCRIZIONE" type="xsd:string"/>
     <xsd:element minOccurs="0" name="NOTE" nillable="true" type="xsd:string"/>
    </xsd:sequence>
   </xsd:complexType>

   <xsd:element name="AuthenticationInfo" type="s0:AuthenticationInfo"/>
   <xsd:complexType name="AuthenticationInfo">
    <xsd:sequence>
     <xsd:element name="userName" type="xsd:string"/>
     <xsd:element name="password" type="xsd:string"/>
     <xsd:element minOccurs="0" name="authentication" type="xsd:string"/>
     <xsd:element minOccurs="0" name="locale" type="xsd:string"/>
     <xsd:element minOccurs="0" name="timeZone" type="xsd:string"/>
    </xsd:sequence>
   </xsd:complexType>
  </xsd:schema>
 </wsdl:types>
   <wsdl:message name="OpenTkSoapOut">
      <wsdl:part element="s0:APERTURATKResponse" name="parameters">
      </wsdl:part>
   </wsdl:message>
   <wsdl:message name="OpenTkSoapIn">
      <wsdl:part element="s0:OpenTk" name="parameters">
      </wsdl:part>
   </wsdl:message>
   <wsdl:message name="ARAuthenticate">
      <wsdl:part element="s0:AuthenticationInfo" name="parameters">
      </wsdl:part>
   </wsdl:message>
   <wsdl:portType name="New_Port_0PortType">
      <wsdl:operation name="OpenTk">
         <wsdl:input message="s0:OpenTkSoapIn">
       </wsdl:input>
         <wsdl:output message="s0:OpenTkSoapOut">
       </wsdl:output>
      </wsdl:operation>
   </wsdl:portType>
   <wsdl:binding name="New_Port_0SoapBinding" type="s0:New_Port_0PortType">
      <soap:binding style="document" transport="http://schemas.xmlsoap.org/soap/http"/>
      <wsdl:operation name="OpenTk">
         <soap:operation soapAction="urn:webServiceTest/OpenTk" style="document"/>
         <wsdl:input>
            <soap:header message="s0:ARAuthenticate" part="parameters" use="literal">
            </soap:header>
            <soap:body use="literal"/>
         </wsdl:input>
         <wsdl:output>
            <soap:body use="literal"/>
         </wsdl:output>
      </wsdl:operation>
   </wsdl:binding>
   <wsdl:service name="WebServicesFWCFFService">
      <wsdl:port binding="s0:New_Port_0SoapBinding" name="New_Port_0Soap">
         <soap:address location="http://localhost:9090/testService"/>
      </wsdl:port>
   </wsdl:service>
</wsdl:definitions>

Мне нужно изменить конечную точку клиента, поэтому я решил использовать функцию cxf.frontend. Я создал простой клиент и использовал API для установки свойства адреса. Вот код:

QName SERVICE = new QName("urn:webServiceTest", "WebServicesFWCFFService");

URL url = this.getClass().getClassLoader().getResource("testWsdl.wsdl").toURI().toURL();
            NewPort0PortType portType = new WebServicesFWCFFService(url, SERVICE).getNewPort0Soap();
            Client client = ClientProxy.getClient(portType);
            client.getRequestContext().put(Message.ENDPOINT_ADDRESS,address);

Затем внутри своих модульных тестов я пытаюсь вызвать сервис (сервис, созданный плагином codegen), выполнив:

ClientTest client = new ClientTest();
        Holder<String> code= new Holder<>();
        Holder<String> description= new Holder<>();
        Holder<String> notes= new Holder<>();
        client.portType.openTk("12","",
                null,null,"1",code, description, notes);
        assertNotNull(notes.value);
        assertNotNull(code.value);

но я получаю ошибку мыла org.apache.cxf.binding.soap.SoapFault: неверное число аргументов при вызове public void it.test.cxf.impl.New_Port_0SoapImpl.openTk.

Я обнаружил, что проблема, кажется, находится внутри org.apache.cxf.jaxws.interceptors.WrapperClassOutInterceptor и, в частности, когда этот перехватчик создает класс WrapperHelper с использованием метода getHelperWrapper. Затем использует эту оболочку для создания объекта с использованием ObjectFactory, созданного плагином codegen (createWrapperObject(objs);). Это приводит к новому объекту InputMappin1 со всеми пустыми полями.

Может кто-нибудь объяснить мне это поведение и как я могу избежать этого? Нужно ли реализовывать какой-то пользовательский перехватчик или мне не хватает какой-то конфигурации?

1 ответ

Я знаю, что это старый вопрос, но вот мои 2 цента.

Как вы сказали, это ошибка в cxf. Я получил ту же ошибку, открывая службу SOAP с JBoss, используя аннотации JAX-WS в исходном коде (мой подход - последний контракт, я начинаю с аннотированного класса java и позволяю cxf генерировать службу с ее WSDL).

Чтобы решить эту проблему, я указал стиль привязки SOAP как DOCUMENT, а стиль параметров как WRAPPED. Я использовал аннотацию JAX WS для класса службы:

      @WebService
@SOAPBinding(
        style = SOAPBinding.Style.DOCUMENT, 
        parameterStyle = SOAPBinding.ParameterStyle.WRAPPED)
public class MySoapService

Подробнее о привязках SOAP можно прочитать здесь . В любом случае, это изменяет структуру XML-сообщений, но cxf справится со всеми деталями за вас.

Другие вопросы по тегам