Zend_XmlRpc_Server.xml 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475
  1. <sect1 id="zend.xmlrpc.server">
  2. <title>Zend_XmlRpc_Server</title>
  3. <sect2 id="zend.xmlrpc.server.introduction">
  4. <title>介绍</title>
  5. <para>
  6. Zend_XmlRpc_Server 依照 <ulink url="http://www.xmlrpc.com/spec">www.xmlrpc.com 上的规格描述</ulink>
  7. 实现了一个全功能 XML-RPC 服务器。同时,它还实现了允许批量传输(boxcarring)的 system.multicall()
  8. 方法。
  9. </para>
  10. </sect2>
  11. <sect2 id="zend.xmlrpc.server.usage">
  12. <title>基本使用</title>
  13. <para>
  14. 最基本的使用实例:
  15. </para>
  16. <programlisting role="php"><![CDATA[<?php
  17. require_once 'Zend/XmlRpc/Server.php';
  18. require_once 'My/Service/Class.php';
  19. $server = new Zend_XmlRpc_Server();
  20. $server->setClass('My_Service_Class');
  21. echo $server->handle();]]>
  22. </programlisting>
  23. </sect2>
  24. <sect2 id="zend.xmlrpc.server.structure">
  25. <title>服务器结构</title>
  26. <para>
  27. Zend_XmlRpc_Server 是由各种不同组件组成,包括服务器自身用于接收请求或进行相应的组件,
  28. 以及错误对象。
  29. </para>
  30. <para>
  31. 为了启动 Zend_XmlRpc_Server,开发人员必须通过 <code>setClass()</code>
  32. 和 <code>addFunction()</code> 方法附加一个或者多个类或函数到服务器。
  33. </para>
  34. <para>
  35. 完成附加操作后,则可以向 <code>Zend_XmlRpc_Server::handle()</code>
  36. 传递 <code>Zend_XmlRpc_Request</code> 对象,如果没有提供它将实例化一个
  37. <code>Zend_XmlRpc_Request_Http</code> 对象用于从 <code>php://input</code>
  38. 获取请求。
  39. </para>
  40. <para>
  41. 这时 <code>Zend_XmlRpc_Server::handle()</code> 将基于请求尝试调度合适的处理程序。
  42. 它将返回基于 <code>Zend_XmlRpc_Response</code> 的对象或者 <code>Zend_XmlRpc_Server_Fault</code>
  43. 对象。这些对象都有 <code>__toString()</code> 用来创建合法的,允许直接返回的 XML-RPC 的
  44. XML 响应文本。
  45. </para>
  46. </sect2>
  47. <sect2 id="zend.xmlrpc.server.conventions">
  48. <title>转换</title>
  49. <para>
  50. Zend_XmlRpc_Server 允许开发人员附加函数和类方法作为 XML-RPC 方法来调度。通过
  51. Zend_Server_Reflection 自省所有附加的方法,使用函数和方法的文档部分(docblocks)
  52. 决定方法的帮助文档和方法标识。
  53. </para>
  54. <para>
  55. XML-RPC 类型不可能与 PHP 类型一一对应。然而程序仍然利用 @param 和 @return
  56. 的值列表尽可能的猜测最合适的类型。一些 XML-RPC 类型没有合适的 PHP 中的匹配,
  57. 所以应当在 phpdoc 中对 XML-RPC 类型给予提示。这些包括:
  58. </para>
  59. <itemizedlist>
  60. <listitem><para>dateTime.iso8601,字符串格式
  61. YYYYMMDDTHH:mm:ss</para></listitem>
  62. <listitem><para>base64, base64 编码数据</para></listitem>
  63. <listitem><para>struct, 结构匹配的数组</para></listitem>
  64. </itemizedlist>
  65. <para>
  66. 下面的例子演示了如何增加这样的提示:
  67. </para>
  68. <programlisting role="php"><![CDATA[<?php
  69. /**
  70. * 这是一个演示函数
  71. *
  72. * @param base64 $val1 Base64-encoded data
  73. * @param dateTime.iso8601 $val2 An ISO date
  74. * @param struct $val3 An associative array
  75. * @return struct
  76. */
  77. function myFunc($val1, $val2, $val3)
  78. {
  79. }]]>
  80. </programlisting>
  81. <para>
  82. PhpDocumentor 不会验证参数和返回值的类型,所以这对 API 文档没有任何影响。
  83. 而且当服务器验证调用方法所提供的参数时,就必须提供提示。
  84. </para>
  85. <para>
  86. 最有效的办法是未参数和返回值指定多个类型;XML-RPC 文档甚至建议 system.methodSignature
  87. 应当返回一个数组,含有所有可能的方法标识(例如,参数和返回值的所有可能的组合)。
  88. 你可以像通常那样在 PhpDocumentor 中使用“|”符号做到这点:
  89. </para>
  90. <programlisting role="php"><![CDATA[<?php
  91. /**
  92. * 这是一个演示函数
  93. *
  94. * @param string|base64 $val1 String or base64-encoded data
  95. * @param string|dateTime.iso8601 $val2 String or an ISO date
  96. * @param array|struct $val3 Normal indexed array or an associative array
  97. * @return boolean|struct
  98. */
  99. function myFunc($val1, $val2, $val3)
  100. {
  101. }]]>
  102. </programlisting>
  103. <para>
  104. 需要注意的是,允许多个标识可能干扰使用服务的开发人员;一般来说,一个 XML-RPC
  105. 方法只应该有一个标识。
  106. </para>
  107. </sect2>
  108. <sect2 id="zend.xmlrpc.server.namespaces">
  109. <title>使用命名空间</title>
  110. <para>
  111. XML-RPC 有名字空间的概念;通常,它允许使用“.”分割 XML-RPC 方法到各个命名空间。
  112. 这有助于防止不同类提供的不同服务方法之间的命名冲突。例如,XML-RPC
  113. 服务在命名空间“system”中保留了一些方法:
  114. </para>
  115. <itemizedlist>
  116. <listitem><para>system.listMethods</para></listitem>
  117. <listitem><para>system.methodHelp</para></listitem>
  118. <listitem><para>system.methodSignature</para></listitem>
  119. </itemizedlist>
  120. <para>
  121. 从内部来讲,这些方法映射到 Zend_XmlRpc_Server 的同名方法上。
  122. </para>
  123. <para>
  124. 如果想要为服务方法增加命名空间,只要在附加一个函数或类时,添加命名空间到适当的方法上:
  125. </para>
  126. <programlisting role="php"><![CDATA[<?php
  127. // My_Service_Class 类的公共方法将被映射为 myservice.METHODNAME
  128. $server->setClass('My_Service_Class', 'myservice');
  129. // 函数“somefunc”将被映射为 funcs.somefunc
  130. $server->addFunction('somefunc', 'funcs');]]>
  131. </programlisting>
  132. </sect2>
  133. <sect2 id="zend.xmlrpc.server.request">
  134. <title>自定义请求对象</title>
  135. <para>
  136. 多数情况下,只需要使用 Zend_XmlRpc_Server 默认提供的请求类型
  137. Zend_XmlRpc_Request_Http。然而,仍然有可能在 CLI、GUI 或者其他环境使用
  138. XML-RPC,亦或需要记录每个请求。需要从 Zend_XmlRpc_Request
  139. 继承,创建自定义的对象来完成这个工作。最重要的是记得确保实现 getMethod()
  140. 和 getParams() 方法,这样 XML-RPC 服务器就可以获取必要的信息进行调度。
  141. </para>
  142. </sect2>
  143. <sect2 id="zend.xmlrpc.server.response">
  144. <title>自定义响应对象</title>
  145. <para>
  146. 和请求对象一样,Zend_XmlRpc_Server 可以返回自定义响应对象;默认情况下,
  147. Zend_XmlRpc_Response_Http 对象被返回,这个对象包含一个合适的用于 XML-RPC 的
  148. HTTP Content-Type 头。使用自定义对象的情况可能是需要记录响应,或需要发送响应到
  149. STDOUT。
  150. </para>
  151. <para>
  152. 在调用 handle() 之前使用 Zend_XmlRpc_Server::setResponseClass()
  153. 指定自定义的响应类。
  154. </para>
  155. </sect2>
  156. <sect2 id="zend.xmlrpc.server.fault">
  157. <title>处理错误产生的异常</title>
  158. <para>
  159. Zend_XmlRpc_Server 会捕获调度方法产生的一场,同时生成 XML-RPC 失败响应。
  160. 然而,默认情况下,异常消息和代码不出现在失败响应中。有意设计成如此形式主要是为了保护代码;
  161. 许多异常会暴露超出预期的大量关于代码和开发环境的信息(常见的例子如数据库抽象层或访问层一场)。
  162. </para>
  163. <para>
  164. 异常类也可以加入白名单中作为失败响应。要做到这点,只需使用
  165. Zend_XmlRpc_Server_Fault::attachFaultException() 添加一个异常类到白名单即可:
  166. </para>
  167. <programlisting role="php"><![CDATA[<?php
  168. Zend_XmlRpc_Server_Fault::attachFaultException('My_Project_Exception');]]>
  169. </programlisting>
  170. <para>
  171. 如果你使用的是一个继承于其他项目的异常,可以一次将整个类树加入白名单。
  172. Zend_XmlRpc_Server_Exceptions 总是在白名单中,用于报告指定的内部错误(如未定义的方法等)。
  173. </para>
  174. <para>
  175. 一个未在白名单中指定的异常将会返回错误代码“404”,错误消息“Unknown error”(未知错误)。
  176. </para>
  177. </sect2>
  178. <sect2 id="zend.xmlrpc.server.caching">
  179. <title>在请求之间缓存服务器定义</title>
  180. <para>
  181. 附加多个类到 XML-RPC 服务器实例可能消耗大量的资源;每一个类必须使用反射
  182. API 进行自省(通过 Zend_Server_Reflection),生成一个包含有所有可用方法的列表用于服务器调用。
  183. </para>
  184. <para>
  185. 为了减少这种性能污点,可以使用 Zend_XmlRpc_Server_Cache 来缓存请求之间的服务器定义。
  186. 联合 __autoload() 使用,可以极大的提高性能。
  187. </para>
  188. <para>
  189. 实例如下:
  190. </para>
  191. <programlisting role="php"><![CDATA[<?php
  192. require_once 'Zend/Loader.php';
  193. require_once 'Zend/XmlRpc/Server.php';
  194. require_once 'Zend/XmlRpc/Server/Cache.php';
  195. function __autoload($class)
  196. {
  197. Zend_Loader::loadClass($class);
  198. }
  199. $cacheFile = dirname(__FILE__) . '/xmlrpc.cache';
  200. $server = new Zend_XmlRpc_Server();
  201. if (!Zend_XmlRpc_Server_Cache::get($cacheFile, $server)) {
  202. require_once 'My/Services/Glue.php';
  203. require_once 'My/Services/Paste.php';
  204. require_once 'My/Services/Tape.php';
  205. $server->setClass('My_Services_Glue', 'glue'); // glue. namespace
  206. $server->setClass('My_Services_Paste', 'paste'); // paste. namespace
  207. $server->setClass('My_Services_Tape', 'tape'); // tape. namespace
  208. Zend_XmlRpc_Server_Cache::save($cacheFile, $server);
  209. }
  210. echo $server->handle();]]>
  211. </programlisting>
  212. <para>
  213. 上面的例子尝试从当前脚本所在目录下的 xmlrpc.cache 文件获取服务器定义。
  214. 如果失败的话就加载需要的服务类,附加他们到服务器实例,并尝试使用服务器定义创建新的缓存文件。
  215. </para>
  216. </sect2>
  217. <sect2 id="zend.xmlrpc.server.use">
  218. <title>使用实例</title>
  219. <para>
  220. 下面有一些有用的例子,向开发人员展示了可能用到的各种情况。
  221. 每一个实例都是建立在前一个实例基础上的扩充。
  222. </para>
  223. <sect3 id="zend.xmlrpc.server.use.case1">
  224. <title>基本使用</title>
  225. <para>
  226. 下面的例子演示了附加一个函数作为 XML-RPC 的调度方法,并用其处理请求。
  227. </para>
  228. <programlisting role="php"><![CDATA[<?php
  229. require_once 'Zend/XmlRpc/Server.php';
  230. /**
  231. * 返回 MD5 值
  232. *
  233. * @param string $value Value to md5sum
  234. * @return string MD5 sum of value
  235. */
  236. function md5Value($value)
  237. {
  238. return md5($value);
  239. }
  240. $server = new Zend_XmlRpc_Server();
  241. $server->addFunction('md5Value');
  242. echo $server->handle();]]>
  243. </programlisting>
  244. </sect3>
  245. <sect3 id="zend.xmlrpc.server.use.case2">
  246. <title>附加一个类</title>
  247. <para>
  248. 下面的例子演示了附加一个类的公共方法作为 XML-RPC 的调度方法。
  249. </para>
  250. <programlisting role="php"><![CDATA[<?php
  251. require_once 'Zend/XmlRpc/Server.php';
  252. require_once 'Services/Comb.php';
  253. $server = new Zend_XmlRpc_Server();
  254. $server->setClass('Services_Comb');
  255. echo $server->handle();]]>
  256. </programlisting>
  257. </sect3>
  258. <sect3 id="zend.xmlrpc.server.use.case3">
  259. <title>利用命名空间附加多个类</title>
  260. <para>
  261. 下面的例子演示了附加多个类,每个类都有自己的命名空间。
  262. </para>
  263. <programlisting role="php"><![CDATA[<?php
  264. require_once 'Zend/XmlRpc/Server.php';
  265. require_once 'Services/Comb.php';
  266. require_once 'Services/Brush.php';
  267. require_once 'Services/Pick.php';
  268. $server = new Zend_XmlRpc_Server();
  269. $server->setClass('Services_Comb', 'comb'); // methods called as comb.*
  270. $server->setClass('Services_Brush', 'brush'); // methods called as brush.*
  271. $server->setClass('Services_Pick', 'pick'); // methods called as pick.*
  272. echo $server->handle();]]>
  273. </programlisting>
  274. </sect3>
  275. <sect3 id="zend.xmlrpc.server.use.case4">
  276. <title>指定异常作为合法的失败响应</title>
  277. <para>
  278. 下面的例子允许任何 Services_Exception 的派生类作为失败响应返回其代码和消息。
  279. </para>
  280. <programlisting role="php"><![CDATA[<?php
  281. require_once 'Zend/XmlRpc/Server.php';
  282. require_once 'Zend/XmlRpc/Server/Fault.php';
  283. require_once 'Services/Exception.php';
  284. require_once 'Services/Comb.php';
  285. require_once 'Services/Brush.php';
  286. require_once 'Services/Pick.php';
  287. // 允许 Services_Exceptions 作为响应失败输出
  288. Zend_XmlRpc_Server_Fault::attachFaultException('Services_Exception');
  289. $server = new Zend_XmlRpc_Server();
  290. $server->setClass('Services_Comb', 'comb'); // methods called as comb.*
  291. $server->setClass('Services_Brush', 'brush'); // methods called as brush.*
  292. $server->setClass('Services_Pick', 'pick'); // methods called as pick.*
  293. echo $server->handle();]]>
  294. </programlisting>
  295. </sect3>
  296. <sect3 id="zend.xmlrpc.server.use.case5">
  297. <title>设置自定义请求对象</title>
  298. <para>
  299. 下面的例子演示了实例化自定义请求对象并将其传递到服务器进行处理。
  300. </para>
  301. <programlisting role="php"><![CDATA[<?php
  302. require_once 'Zend/XmlRpc/Server.php';
  303. require_once 'Zend/XmlRpc/Server/Fault.php';
  304. require_once 'Services/Request.php';
  305. require_once 'Services/Exception.php';
  306. require_once 'Services/Comb.php';
  307. require_once 'Services/Brush.php';
  308. require_once 'Services/Pick.php';
  309. // 允许 Services_Exceptions 作为响应失败输出
  310. Zend_XmlRpc_Server_Fault::attachFaultException('Services_Exception');
  311. $server = new Zend_XmlRpc_Server();
  312. $server->setClass('Services_Comb', 'comb'); // methods called as comb.*
  313. $server->setClass('Services_Brush', 'brush'); // methods called as brush.*
  314. $server->setClass('Services_Pick', 'pick'); // methods called as pick.*
  315. // 创建请求对象
  316. $request = new Services_Request();
  317. echo $server->handle($request);]]>
  318. </programlisting>
  319. </sect3>
  320. <sect3 id="zend.xmlrpc.server.use.case6">
  321. <title>设置自定义响应对象</title>
  322. <para>
  323. 下面的例子演示了指定自定义类作为响应返回。
  324. </para>
  325. <programlisting role="php"><![CDATA[<?php
  326. require_once 'Zend/XmlRpc/Server.php';
  327. require_once 'Zend/XmlRpc/Server/Fault.php';
  328. require_once 'Services/Request.php';
  329. require_once 'Services/Response.php';
  330. require_once 'Services/Exception.php';
  331. require_once 'Services/Comb.php';
  332. require_once 'Services/Brush.php';
  333. require_once 'Services/Pick.php';
  334. // 允许 Services_Exceptions 作为响应失败输出
  335. Zend_XmlRpc_Server_Fault::attachFaultException('Services_Exception');
  336. $server = new Zend_XmlRpc_Server();
  337. $server->setClass('Services_Comb', 'comb'); // methods called as comb.*
  338. $server->setClass('Services_Brush', 'brush'); // methods called as brush.*
  339. $server->setClass('Services_Pick', 'pick'); // methods called as pick.*
  340. // 创建请求对象
  341. $request = new Services_Request();
  342. // 设置自定义响应
  343. $server->setResponseClass('Services_Response');
  344. echo $server->handle($request);]]>
  345. </programlisting>
  346. </sect3>
  347. <sect3 id="zend.xmlrpc.server.use.case7">
  348. <title>在请求之间缓存服务器定义</title>
  349. <para>
  350. 下面的例子演示了在请求之间缓存服务器定义。
  351. </para>
  352. <programlisting role="php"><![CDATA[<?php
  353. require_once 'Zend/XmlRpc/Server.php';
  354. require_once 'Zend/XmlRpc/Server/Fault.php';
  355. require_once 'Zend/XmlRpc/Server/Cache.php';
  356. require_once 'Services/Request.php';
  357. require_once 'Services/Response.php';
  358. require_once 'Services/Exception.php';
  359. require_once 'Services/Comb.php';
  360. require_once 'Services/Brush.php';
  361. require_once 'Services/Pick.php';
  362. // 指定一个缓存文件
  363. $cacheFile = dirname(__FILE__) . '/xmlrpc.cache';
  364. // 允许 Services_Exceptions 作为响应失败输出
  365. Zend_XmlRpc_Server_Fault::attachFaultException('Services_Exception');
  366. $server = new Zend_XmlRpc_Server();
  367. // 尝试从缓存中获取服务器定义
  368. if (!Zend_XmlRpc_Server_Cache::get($cacheFile, $server)) {
  369. $server->setClass('Services_Comb', 'comb'); // methods called as comb.*
  370. $server->setClass('Services_Brush', 'brush'); // methods called as brush.*
  371. $server->setClass('Services_Pick', 'pick'); // methods called as pick.*
  372. // 保存缓存
  373. Zend_XmlRpc_Server_Cache::save($cacheFile, $server));
  374. }
  375. // 创建请求对象
  376. $request = new Services_Request();
  377. // 设置自定义响应
  378. $server->setResponseClass('Services_Response');
  379. echo $server->handle($request);]]>
  380. </programlisting>
  381. </sect3>
  382. </sect2>
  383. </sect1>
  384. <!--
  385. vim:se ts=4 sw=4 et:
  386. -->