最近在做一个调用webservice的客户端,之前做过,以为很容易,但这过程中遇到的一些问题让我发现自己懂的还是太少。
webservice的服务端是需要证书认证的,证书已经发给我了,cer格式的,使用java的keytool工具提取证书后,在调用webservice时执行以下语句:
System.setProperty("javax.net.ssl.trustStore", "xxxx.truststore");
告诉服务端我客户端是有证书的,目前来看没问题。
接下来,我用axis2的wsdl2java生成了客户端的代码,结果一执行就出现如下错误:
org.apache.axis2.AxisFault: [ISS.0088.9125] SOAP request does not conform to the SOAP message model
在soap的开发文档中查到这个错,是因为请求的soap message格式错误,文档地址:http://documentation.softwareag.com/webmethods/wmsuite7/Developer/Guides/7-1-1_SOAP_Developers_Guide.pdf(89页)
这时对方要求我把soap request message发过去,我对webservice只是停留在用的阶段,会用axis,xfire之类的工具写服务端和客户端,但是对soap就是一知半解了,上网找资料,问同事,折腾了半天,总算把request message找出来了。同事给的方法是用tcp monitor之类的工具,但是webservice是https格式的,不能用,后来在生成的客户端代码里把SOAPEnvelope.toString()打出来才得到,如下:
<?xml version='1.0' encoding='utf-8'?>
<soapenv:Envelope xmlns:soapenv="http://schemas.xmlsoap.org/soap/envelope/">
<soapenv:Body>
<ns1:UPLGenerate xmlns:ns1="http://www.alcatel-lucent.com/webService/WS_UPL">
<OperationCode>1 </OperationCode>
<Plant>2 </Plant>
<QuoteNumber>3 </QuoteNumber>
<UserID>4 </UserID>
<IncludePriceType>5 </IncludePriceType>
</ns1:UPLGenerate>
</soapenv:Body>
</soapenv:Envelope>
其实这个问题的原因是axis2客户端默认的transfer-encoding是chunked,所以在soap 请求报文的body首尾会有两个数字,可以用tcpmon测试一下,.net或者其他一些server是不支持这种模式的,代码中设置一下就可以了,如下:
stub._getServiceClient().getOptions().setProperty(HTTPConstants.CHUNKED, false);
周末研究出来的,之前咨询了公司的一个senior,帮我查,也给了我一些建议,非常感谢他!
现在执行应该没问题了吧,我倒是想,结果又有问题了
Exception in thread "main" org.apache.axis2.AxisFault: org.apache.axis2.databinding.ADBException: Unexpected subelement xxxxResponse
at org.apache.axis2.AxisFault.makeFault(AxisFault.java:430)
这个问题折腾了我三天,在网上查到很多,有的说要用wstx-asl-3.2.1.jar等等,都试了,不通,至少我这边不是因为这个错,很急,一点头绪没有,后来偶然在stub里把org.apache.axiom.soap.SOAPEnvelope _returnEnv打印出来了,就是_returnEnv.toString(),发现和用soapUI返回的结果不一样,因为这个webservice是https的,所以不能用tcpmon,这下明白了,因为返回response里的一下元素不是一直有的,但是在wsdl定义里却没有定义,即没写minOccurs="0",结果用这个wsdl生成的客户端代码如果没取到那些没返回的元素,自然会报错,在wsdl里加上minOccurs="0",重新生成客户端代码,测试,通过。
最后一步,部署到websphere上后,执行webservice调用出错:
java.net.SocketException: Unconnected sockets not implemented
是证书的问题,花了好几天时间,后来又反复,是因为缺少这个证书的父证书,总之如果在websphere里无法通过host和port得到所有证书,包括从直接使用的证书到证书颁发机构的证书,只有通过导入.cer文件的方法了,问题的根源就在这里:证书!!!
遇到的问题很烦人,但是收获也不小,这短短几天,让我对webservice的认识加深很多,很惭愧的说,以前只知道用axis等写服务端,然后用wsdl生成客户端,不知道用wsdl还可以生成服务端的。还好有这次机会,我进步了,大家共勉。
-