近期在搞一个项目,甲方只给了一个WSDL文件,让我们实现响应接口。
于是研究了WSDL和SOAP通信相关知识,获益甚深,把我的理解写下来,希望对PHP新手有帮助。
1.基本概念
soap是简单对象访问协议的缩写,是一种可以基于HTTP协议的访问方式。客户端发送请求,然后调用带参数的服务端函数得到服务端的数据;服务端编写处理函数并响应客户端。
wsdl是网络服务者动态语言的缩写,这里面定义了双方通信时包含的东西。客户端把wsdl文件给服务端,服务端分析wsdl并写出里面的函数,这样两者无论是什么平台、什么语言都可以通信。
我对soap的理解就是类似于POST请求的一种传递参数的方式,只是要求格式比POST更严格。
我认为wsdl也仅仅是一种xml标记文本,跟html文本没有什么区别。
但是两者结合起来就很头疼了。
我是第一次接触soap和wsdl,这两者同时出现,我把他们俩混在一起搞不清。
参看了很多文档才清楚这两者到底是什么。
soap结合wsdl,其实就是把soap通信方式限制的更死,死到你服务端里只能按照wsdl里面规定的函数来写,并严格限制参数和返回结果。
2.我走的弯路
一开始我拿到这个wsdl文档,打开来看密密麻麻的,一共将近200行。
我没搞清是什么东西,连里面的标记符都没有搞懂,就开始在网上搜php与wsdl有关的博客。
搜到一个SoapDiscovery.class.php文件,说可以创建wsdl文件。
我就开始盲目的想创建wsdl文件,并使这个wsdl文件和甲方的一样。
我错误的认为客户端发送wsdl文件给我,我来解析里面是什么,然后我处理后再次封装成wsdl文件发送给客户端。
其实,这个wsdl根本就不用创建,它甚至都不是重点。
这个wsdl文件是双方通信前就规定好的,双方都按照这个wsdl里面的规定访问函数或返回函数等内容。
所以重点不是解析、构建wsdl文件,而是根据这个wsdl写好服务端的函数和返回值。
3.解决问题阶段一
理解wsdl的几个网站:http://staff.ustc.edu.cn/~shizhu/DotNet/WSDLxj.htm 【Web Service描述语言 WSDL 详解】
https://www.ibm.com/developerworks/cn/education/webservices/ws-dewsdl/ws-dewsdl.html 【描述 Web 服务:WSDL】
认真看完上面两个网站,基本就知道wsdl里面讲的是什么了。
我在网上下载了很多关于php与wsdl的实例,
感觉这个例子很好:http://www.cnblogs.com/VipBin/archive/2011/12/07/2279927.html 【在PHP中利用wsdl创建标准webservice】
有了以上的基础。
我开始理解wsdl工作原理了。
客户端client就是发送请求的,服务端service就是接收请求的,再来一个class类处理函数(这里我叫它Robot类),就行了。
上面怎么写客户端与服务端用到了SoapClient,SoapServer两个类。
怎么写Robot类的函数用到了wsdl文件里规定的函数,也就是wsdl知识。
4.解决问题阶段二
通过上面的例子,我可以成功运行。但已加上我Robot类就直接http 500错误,没有任何提示信息,调试及其麻烦。
还好网上有两款很好的软件。
一款是让wsdl文件直观化,直接看到里面定义函数的规则,叫XMLSpy软件。
另一款是SoapUI,不需要写客户端,可以直接加入wsdl文件,访问服务端。
从这两个软件中,我认识到wsdl文件虽然很乱,但只有几个是重点。
一个是wsdl里的soap:address,要定义好访问的服务端的地址。
一个是wsdl里的portType的operation,每一个operation的name就是Robot类中必须写的函数名,
最后一个是wsdl里operation的out,也就是每个函数的返回值的规定。
通过SoapUI,我可以清晰的看到客户端发送和服务端返回的到底是什么内容。
其实里面没有任何wsdl的标记,都是正规的soap标记,所以wsdl根本就没有在通信过程用到,只在通信两端用到。
我们就不需要管wsdl了,只要能实现里面的函数就行了。
5.解决问题阶段三
到了这个阶段,客户端与服务端的函数访问什么的都有了,唯一有问题的就是来回的参数格式问题。
这时不会报http500错误,而是200 OK,但页面不会显示任何东西。因为参数是有问题的。
先说怎么知道参数的格式的。
1 //下面两个看懂,输出就是你要写的类 2 var_dump($client->__getFunctions());//打印暴露的方法 3 print("<br/>"); 4 var_dump($client->__getTypes());//打印对应方法的参数和参数类型 5 print("<br/>");
然后打印出来的参数是下面这样的:
array(7) { [0]=> string(97) "struct standPointInfo { string POINTDES; string STANDPOINT; string STOR_TYPE; string WH_NO; }" [1]=> string(57) "struct webServiceResult { string INFO; string STATUS; }" [2]=> string(79) "struct taskResultInfo { string EXEC_STATE; string PICTURE; string TASK_NO; }" [3]=> string(265) "struct inventoryResultInfo { string CCDD; string CCLX; string CKH; string CW; string GC; string KCLX; double KCSL; string PC; double PDCY; string PDSJ; double PDSL; string RWH; string TSKCBH; string TSKCLX; string WLBH; string WZSFM; string ZHXM; }" [4]=> string(52) "struct standPointInfoArray { standPointInfo item; }" [5]=> string(62) "struct inventoryResultInfoArray { inventoryResultInfo item; }" [6]=> string(37) "struct Exception { string message; }" }
看到上面的参数格式,然后找到合适的php数据类型就行了。
里面有struct类型,但php没有,不用怕,直接当array来用就行了,比如构建参数standPointInfoArray :
1 $standPointInfoArray = array( 2 array( 3 ‘POINTDES‘ => ‘hujun‘, 4 ‘STANDPOINT‘ => ‘standPoint‘, 5 ‘STOR_TYPE‘ => ‘storType‘, 6 ‘WH_NO‘ => ‘whNo1‘, 7 ), 8 array( 9 ‘POINTDES‘ => ‘hujun‘, 10 ‘STANDPOINT‘ => ‘standPoint‘, 11 ‘STOR_TYPE‘ => ‘storType‘, 12 ‘WH_NO‘ => ‘whNo2‘, 13 ), 14 );
这个参数在wsdl里定义为二维数组,所以我用php也构建了一个二维数组,键名不能改要与wsdl名字一模一样而其必须加上,不然没法传参数。
然后把这个参数扔到客户端的访问函数参数里就行了,不用转什么stdClass和json之类的,多余,直接用array就行了。
客户端参数搞定了,下面看看服务端形参怎么搞。
下面是我得到的wsdl规定的函数:
array(3) { [0]=> string(76) "webServiceResult receiveStandPointInfo(standPointInfoArray $StandPointInfos)" [1]=> string(82) "webServiceResult receiveInventoryResult(inventoryResultInfoArray $InventoryResult)" [2]=> string(66) "webServiceResult receiveTaskResult(taskResultInfo $TaskResultInfo)" }
以receiveStandPointInfo函数为例。
在Robot类里必须写这个函数,而且函数名必须一模一样,不能有一点改变。
然后看形参是standPointInfoArray $StandPointInfos,这个standPointInfoArray类型在PHP很难自定义,但PHP的好处就是可以不用写类型。
所以直接在形成里写变量名就行了,把前面的类型去掉。
函数名如下:
public function receiveStandPointInfo($standPointInfoArray)
这样形参就好了,$standPointInfoArray可以被客户端赋值。
这个时候是最折磨人的,因为你没法知道这个$standPointInfoArray是什么东西。服务端不给打印信息。
只能先猜测为数组,然后用数组的方式调用,直接报错。
Fatal error: Cannot use object of type stdClass as array
上面讲的很清楚,这是个stdClass的类型。
要是一维数组还好访问,直接用$standPointInfo->POINTDES就可以访问了。
但是刚才客户端传递的是二维数组呀,这怎么访问呢?
所以必须知道$standPointInfoArray到底是什么东西。
在网上找到了stdClass的打印信息示例如下:
stdClass Object ( [item] => Array ( [0] => stdClass Object ( [date] => 2008-07-17T01:23:06Z [directory] => 1 [downloadCount] => 0 ) [1] => stdClass Object ( [date] => 2009-11-03T23:03:15Z [directory] => 2 [downloadCount] => 5 ) ) )
这下明朗了,原理这个键名叫item,是Soap协议传输多个复杂类型规定的。
所以可以使用$standPointInfoArray->item[0]->data来访问日期等等。
到这里,参数传递就完成了。
返回的参数构造与发送的参数一样,这里就不写了。
6.源代码
客户端的client.php
1 <?php 2 $client = new SoapClient("robot_origin.wsdl", array(‘trace‘=>true)); 3 try { 4 $parms = array( 5 ‘EXEC_STATE‘ => "hujun", 6 ‘PICTURE‘ => ‘pictures‘, 7 ‘TASK_NO‘ => ‘task_nos‘, 8 ); 9 $result = $client->receiveTaskResult($parms); 10 var_dump($result); 11 print("<br/>=======================<br/>"); 12 13 $standPointInfo = array( 14 ‘POINTDES‘ => ‘hujun‘, 15 ‘STANDPOINT‘ => ‘standPoint‘, 16 ‘STOR_TYPE‘ => ‘storType‘, 17 ‘WH_NO‘ => ‘whNo‘, 18 ); 19 $standPointInfoArray = array( 20 array( 21 ‘POINTDES‘ => ‘hujun‘, 22 ‘STANDPOINT‘ => ‘standPoint‘, 23 ‘STOR_TYPE‘ => ‘storType‘, 24 ‘WH_NO‘ => ‘whNo1‘, 25 ), 26 array( 27 ‘POINTDES‘ => ‘hujun‘, 28 ‘STANDPOINT‘ => ‘standPoint‘, 29 ‘STOR_TYPE‘ => ‘storType‘, 30 ‘WH_NO‘ => ‘whNo2‘, 31 ), 32 ); 33 $result = $client->receiveStandPointInfo($standPointInfoArray); 34 var_dump($result); 35 print("<br/>=======================<br/>"); 36 37 //下面两个看懂,输出就是你要写的类 38 var_dump($client->__getFunctions());//打印暴露的方法 39 print("<br/>"); 40 var_dump($client->__getTypes());//打印对应方法的参数和参数类型 41 print("<br/>"); 42 echo("\nDumping request headers:\n"); 43 var_dump($client->__getLastRequestHeaders()); 44 echo "<br>"; 45 echo("\nDumping request:\n"); 46 var_dump($client->__getLastRequest()); 47 echo "<br>"; 48 echo("\nDumping response headers:\n"); 49 var_dump($client->__getLastResponseHeaders()); 50 echo "<br>"; 51 echo("\nDumping response:\n"); 52 var_dump($client->__getLastResponse()); 53 } 54 catch (SoapFault $f){ 55 echo "Error Message: {$f->getMessage()}"; 56 } 57 ?>
服务端的service.php
1 <?php 2 include("robot.class.php"); 3 ini_set(‘soap.wsdl_cache_enabled‘,‘0‘); //关闭WSDL缓存 4 $objSoapServer = new SoapServer("robot_origin.wsdl");//person.wsdl是刚创建的wsdl文件 5 6 $objSoapServer->setClass("Robot");//注册person类的所有方法 7 $objSoapServer->handle();//处理请求
服务端处理类Robot.class.php
1 <?php 2 class Robot{ 3 public function receiveInventoryResult($inventoryResultInfoArray){ 4 $webServiceResult = array( 5 ‘INFO‘ => ‘info‘, 6 ‘STATUS‘ => ‘status‘, 7 ); 8 9 $webServiceResult[‘INFO‘] = $inventoryResultInfoArray->item[0]->KCSL; 10 11 return $webServiceResult; 12 } 13 public function receiveStandPointInfo($standPointInfoArray){ 14 $webServiceResult = array( 15 ‘INFO‘ => ‘info‘, 16 ‘STATUS‘ => ‘status‘, 17 ); 18 if($standPointInfoArray->item[0]->POINTDES){ 19 $webServiceResult[‘INFO‘] = ‘we can change it‘; 20 } 21 return $webServiceResult; 22 } 23 24 public function receiveTaskResult($taskResultInfo){ 25 $EXEC_STATE = $taskResultInfo->EXEC_STATE; 26 $PICTURE = $taskResultInfo->PICTURE; 27 $TASK_NO = $taskResultInfo->TASK_NO; 28 29 $webServiceResult = array( 30 ‘INFO‘ => ‘info‘, 31 ‘STATUS‘ => $PICTURE, 32 ); 33 $webServiceResult[‘INFO‘] = $EXEC_STATE; 34 return $webServiceResult; 35 } 36 37 public function Exception($exception){ 38 $this->exception = $exception; 39 return $this->exception; 40 } 41 }
一直是电脑在用的wsdl文件
<?xml version=‘1.0‘ encoding=‘utf-8‘?><wsdl:definitions name="RobotWebServiceImplService" targetNamespace="http://robot.server.webService/" xmlns:ns1="http://schemas.xmlsoap.org/soap/http" xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/" xmlns:tns="http://robot.server.webService/" xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/" xmlns:xsd="http://www.w3.org/2001/XMLSchema"> <wsdl:types> <!------schema才是关键,注意类型的寻址,下面是xs开头的,意思是这些类型到xmlns:xs=里面找------------> <xs:schema attributeFormDefault="unqualified" elementFormDefault="unqualified" targetNamespace="http://robot.server.webService/" xmlns:tns="http://robot.server.webService/" xmlns:xs="http://www.w3.org/2001/XMLSchema"> //------------------------------------------------------------------------------------------------------------ <!--设计一个复杂的数据类型standPointInfo--> <xs:complexType name="standPointInfo"> <!--sequence是顺序限制--> <!--<minOccurs> 指示器可规定某个元素能够出现的最小次数--> <xs:sequence> <xs:element minOccurs="0" name="POINTDES" type="xs:string"></xs:element> <xs:element minOccurs="0" name="STANDPOINT" type="xs:string"></xs:element> <xs:element minOccurs="0" name="STOR_TYPE" type="xs:string"></xs:element> <xs:element minOccurs="0" name="WH_NO" type="xs:string"></xs:element> </xs:sequence> </xs:complexType> //-------------------------------1.1.1 output <xs:complexType name="webServiceResult"> <xs:sequence> <xs:element minOccurs="0" name="INFO" type="xs:string"></xs:element> <xs:element minOccurs="0" name="STATUS" type="xs:string"></xs:element> </xs:sequence> </xs:complexType> //--------------------------------不需要 <xs:complexType name="taskResultInfo"> <xs:sequence> <xs:element minOccurs="0" name="EXEC_STATE" type="xs:string"></xs:element> <xs:element minOccurs="0" name="PICTURE" type="xs:string"></xs:element> <xs:element minOccurs="0" name="TASK_NO" type="xs:string"></xs:element> </xs:sequence> </xs:complexType> //----------------------------------1.2.1 input <xs:complexType name="inventoryResultInfo"> <xs:sequence> <xs:element minOccurs="0" name="CCDD" type="xs:string"></xs:element> <xs:element minOccurs="0" name="CCLX" type="xs:string"></xs:element> <xs:element minOccurs="0" name="CKH" type="xs:string"></xs:element> <xs:element minOccurs="0" name="CW" type="xs:string"></xs:element> <xs:element minOccurs="0" name="GC" type="xs:string"></xs:element> <xs:element minOccurs="0" name="KCLX" type="xs:string"></xs:element> <xs:element minOccurs="0" name="KCSL" type="xs:double"></xs:element> <xs:element minOccurs="0" name="PC" type="xs:string"></xs:element> <xs:element minOccurs="0" name="PDCY" type="xs:double"></xs:element> <xs:element minOccurs="0" name="PDSJ" type="xs:string"></xs:element> <xs:element minOccurs="0" name="PDSL" type="xs:double"></xs:element> <xs:element minOccurs="0" name="RWH" type="xs:string"></xs:element> <xs:element minOccurs="0" name="TSKCBH" type="xs:string"></xs:element> <xs:element minOccurs="0" name="TSKCLX" type="xs:string"></xs:element> <xs:element minOccurs="0" name="WLBH" type="xs:string"></xs:element> <xs:element minOccurs="0" name="WZSFM" type="xs:string"></xs:element> <xs:element minOccurs="0" name="ZHXM" type="xs:string"></xs:element> </xs:sequence> </xs:complexType> //------------------------------------arrary <xs:complexType final="#all" name="standPointInfoArray"> <xs:sequence> <xs:element maxOccurs="unbounded" minOccurs="0" name="item" nillable="true" type="tns:standPointInfo"></xs:element> </xs:sequence> </xs:complexType> //-------------------------------------arrary <xs:complexType final="#all" name="inventoryResultInfoArray"> <xs:sequence> <xs:element maxOccurs="unbounded" minOccurs="0" name="item" nillable="true" type="tns:inventoryResultInfo"></xs:element> </xs:sequence> </xs:complexType> //--------------------------------------异常 <xs:element name="Exception" type="tns:Exception"></xs:element> <xs:complexType name="Exception"> <xs:sequence> <xs:element minOccurs="0" name="message" type="xs:string"></xs:element> </xs:sequence> </xs:complexType> </xs:schema> </wsdl:types> //---------------------------------------message是operation的参数 <!--<message> 元素将数据(数据类型在 <types> 元素中进行定义)分组成一个用于逻辑网络传输的特征符,并将数据绑定到一个名称上--> <!--上面的name用于operation引用,下面的name是声明的实例--> <wsdl:message name="receiveInventoryResult"> <wsdl:part name="InventoryResult" type="tns:inventoryResultInfoArray"> </wsdl:part> </wsdl:message> <wsdl:message name="receiveTaskResult"> <wsdl:part name="TaskResultInfo" type="tns:taskResultInfo"> </wsdl:part> </wsdl:message> <wsdl:message name="receiveStandPointInfoResponse"> <wsdl:part name="return" type="tns:webServiceResult"> </wsdl:part> </wsdl:message> <wsdl:message name="receiveInventoryResultResponse"> <wsdl:part name="return" type="tns:webServiceResult"> </wsdl:part> </wsdl:message> <wsdl:message name="receiveTaskResultResponse"> <wsdl:part name="return" type="tns:webServiceResult"> </wsdl:part> </wsdl:message> <wsdl:message name="receiveStandPointInfo"> <wsdl:part name="StandPointInfos" type="tns:standPointInfoArray"> </wsdl:part> </wsdl:message> <wsdl:message name="Exception"> <wsdl:part element="tns:Exception" name="Exception"> </wsdl:part> </wsdl:message> //-------------------------------------------------------------- <!---相当于 Java 中的接口--> <!---它将对 <message> 元素的引用分组成逻辑操作,一个进程可以对另一个进程执行这些操作,并将它们绑定到一个名称--> <wsdl:portType name="RobotWebService"> //--------------------------------------------------------------- <!---<input> 元素声明客户机向 Web 服务请求传输的需求。<output> 声明 Web 服务响应的内容。<fault> 元素描述当 Web 服务设法响应客户机的请求时所发生的任何消息级异常--> <wsdl:operation name="receiveStandPointInfo"> <wsdl:input message="tns:receiveStandPointInfo" name="receiveStandPointInfo"></wsdl:input> <wsdl:output message="tns:receiveStandPointInfoResponse" name="receiveStandPointInfoResponse"></wsdl:output> <wsdl:fault message="tns:Exception" name="Exception"></wsdl:fault> </wsdl:operation> //----------------------------- <wsdl:operation name="receiveTaskResult"> <wsdl:input message="tns:receiveTaskResult" name="receiveTaskResult"></wsdl:input> <wsdl:output message="tns:receiveTaskResultResponse" name="receiveTaskResultResponse"></wsdl:output> <wsdl:fault message="tns:Exception" name="Exception"></wsdl:fault> </wsdl:operation> //------------------------------ <wsdl:operation name="receiveInventoryResult"> <wsdl:input message="tns:receiveInventoryResult" name="receiveInventoryResult"> </wsdl:input> <wsdl:output message="tns:receiveInventoryResultResponse" name="receiveInventoryResultResponse"> </wsdl:output> <wsdl:fault message="tns:Exception" name="Exception"> </wsdl:fault> </wsdl:operation> </wsdl:portType> //---------------------------------------------------------------- <!--到这里,上面的定义什么的都是抽象的,binding将这些抽象的钩接点与 Web 协议束缚起来--> <!--描述绑定的元素嵌套在 <binding> 元素的这些子元素之中,这是为了将消息传递协议的详细内容链接到目标 portType 中所提到的通则上。--> <wsdl:binding name="RobotWebServiceImplServiceSoapBinding" type="tns:RobotWebService"> <!--创建 SOAP 绑定--> <soap:binding style="rpc" transport="http://schemas.xmlsoap.org/soap/http"></soap:binding> <wsdl:operation name="receiveStandPointInfo"> <!--soapAction,在 SOAP 1.2 中,不赞成使用这个头,而赞成将请求中的请求目的声明为一个参数,称作“action”。因而通常就让这个头空着。--> <soap:operation soapAction="" style="rpc"></soap:operation> <!--输入值中,除了出错数据应当包含在 SOAP <Fault> 元素中之外,其它所有消息都将包含在常规的 SOAP <Body> 元素中--> <wsdl:input name="receiveStandPointInfo"><soap:body namespace="http://robot.server.webService/" use="literal"></soap:body></wsdl:input> <wsdl:output name="receiveStandPointInfoResponse"><soap:body namespace="http://robot.server.webService/" use="literal"></soap:body></wsdl:output> <wsdl:fault name="Exception"><soap:fault name="Exception" use="literal"></soap:fault></wsdl:fault> </wsdl:operation> //--- <wsdl:operation name="receiveInventoryResult"> <soap:operation soapAction="" style="rpc"></soap:operation> <wsdl:input name="receiveInventoryResult"> <soap:body namespace="http://robot.server.webService/" use="literal"></soap:body> </wsdl:input> <wsdl:output name="receiveInventoryResultResponse"> <soap:body namespace="http://robot.server.webService/" use="literal"></soap:body> </wsdl:output> <wsdl:fault name="Exception"> <soap:fault name="Exception" use="literal"></soap:fault> </wsdl:fault> </wsdl:operation> //-- <!--literal 编码的优点是没有对正在传输的 XML 文档作任何限制。它的编码依赖于模式,因此是完全可扩展的。--> <wsdl:operation name="receiveTaskResult"> <soap:operation soapAction="" style="rpc"></soap:operation> <wsdl:input name="receiveTaskResult"> <soap:body namespace="http://robot.server.webService/" use="literal"></soap:body> </wsdl:input> <wsdl:output name="receiveTaskResultResponse"> <soap:body namespace="http://robot.server.webService/" use="literal"></soap:body> </wsdl:output> <wsdl:fault name="Exception"> <soap:fault name="Exception" use="literal"></soap:fault> </wsdl:fault> </wsdl:operation> //-- </wsdl:binding> //----------------------------------------------------------服务器地址 <!--将某个具体的绑定与网络上的一个或多个进程相关联,这些进程可以根据绑定所实现的 portType 来处理请求--> <!--文档样式的消息传递表示 SOAP <Body> 元素的内容是任意的 XML 文档。--> <!--尽管可以在请求-响应类型的通信方案中使用文档样式的消息传递,但是在异步通信中使用它非常理想,因为这个自包含的 XML 文档可以放入队列等待处理。--> <wsdl:service name="RobotWebServiceImplService"> <wsdl:port binding="tns:RobotWebServiceImplServiceSoapBinding" name="RobotWebServiceImplPort"> <soap:address location="http://soap.cn/service.php"></soap:address> </wsdl:port> </wsdl:service> </wsdl:definitions>