Zend_XmlRpc_Server.xml 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432
  1. <!-- EN-Revision: 13868 -->
  2. <sect1 id="zend.xmlrpc.server">
  3. <title>Zend_XmlRpc_Server</title>
  4. <sect2 id="zend.xmlrpc.server.introduction">
  5. <title>Introduction</title>
  6. <para><classname>Zend_XmlRpc_Server</classname> fournit un serveur XML-RPC qui suit les spécifications <ulink
  7. url="http://www.xmlrpc.com/spec">dictées par www.xmlrpc.com</ulink>. Il fournit aussi la méthode
  8. <code>system.multicall()</code>, permettant le traitement de requêtes multiples.</para>
  9. </sect2>
  10. <sect2 id="zend.xmlrpc.server.usage">
  11. <title>Usage de base</title>
  12. <para>Voici un exemple d'utilisation basique :</para>
  13. <programlisting role="php"><![CDATA[
  14. $server = new Zend_XmlRpc_Server();
  15. $server->setClass('My_Service_Class');
  16. echo $server->handle();
  17. ]]></programlisting>
  18. </sect2>
  19. <sect2 id="zend.xmlrpc.server.structure">
  20. <title>Structures du serveur</title>
  21. <para><classname>Zend_XmlRpc_Server</classname> se décompose en un objet serveur (lui-même), un objet requête, réponse, et
  22. des objets d'erreurs.</para>
  23. <para>Pour démarrer un serveur <classname>Zend_XmlRpc_Server</classname>, vous devez attacher une ou plusieurs classes ou
  24. fonctions au serveur, grâce à <code>setClass()</code> et <code>addFunction()</code>.</para>
  25. <para>Lorsque c'est fait, vous pouvez passer un objet <classname>Zend_XmlRpc_Request</classname> à
  26. <classname>Zend_XmlRpc_Server::handle()</classname>, sinon par défaut il utilisera un objet
  27. <classname>Zend_XmlRpc_Request_Http</classname> qui récupérera la requête depuis <code>php://input</code>.</para>
  28. <para><classname>Zend_XmlRpc_Server::handle()</classname> va alors essayer de traiter la requête. Cette méthode retournera
  29. un objet <classname>Zend_XmlRpc_Response</classname> ou <classname>Zend_XmlRpc_Server_Fault</classname>. Tous deux possèdent une
  30. méthode <code>__toString()</code> qui crée une réponse XML valide XML-RPC.</para>
  31. </sect2>
  32. <sect2 id="zend.xmlrpc.server.conventions">
  33. <title>Conventions</title>
  34. <para><classname>Zend_XmlRpc_Server</classname> permet d'attacher des classes et/ou des fonctions au serveur XML-RPC.
  35. Grâce à <classname>Zend_Server_Reflection</classname>, l'introspection va utiliser les blocs de commentaires pour
  36. déterminer les types d'arguments et de réponse de la fonction/classe.</para>
  37. <para>Les types XML-RPC n'ont pas forcément de correspondance native vers un type PHP. Le code fera de son mieux
  38. pour deviner le type de données approprié, en se basant sur les valeurs listées dans les balises @param et
  39. @return. Certains types XML-RPC n'ont par contre pas d'équivalent PHP direct, ils devront alors être spécifiés
  40. manuellement sous forme de balises phpdoc :</para>
  41. <itemizedlist>
  42. <listitem>
  43. <para>dateTime.iso8601, une chaîne formatée comme YYYYMMDDTHH:mm:ss</para>
  44. </listitem>
  45. <listitem>
  46. <para>base64, données encodées en base64</para>
  47. </listitem>
  48. <listitem>
  49. <para>struct, tableau associatif</para>
  50. </listitem>
  51. </itemizedlist>
  52. <para>Voici un exemple d'utilisation de type particulier:</para>
  53. <programlisting role="php"><![CDATA[
  54. /**
  55. * This is a sample function
  56. *
  57. * @param base64 $val1 Base64-encoded data
  58. * @param dateTime.iso8601 $val2 An ISO date
  59. * @param struct $val3 An associative array
  60. * @return struct
  61. */
  62. function myFunc($val1, $val2, $val3)
  63. {}
  64. ]]></programlisting>
  65. <para>PhpDocumentor ne vérifie (valide) pas les types des paramètres, mais les types sont obligatoires pour que
  66. le serveur puisse lui, valider les paramètres passés aux appels des méthodes.</para>
  67. <para>Il est parfaitement valide de spécifier plusieurs types pour les paramètres et les retours de méthodes. La
  68. spécification XML-RPC suggère que system.methodSignature retourne un tableau des possibilités au regard des
  69. paramètres d'entrée de la méthode, et de son type de sortie. Ceci ce fait grâce au caractère '|' de
  70. PhpDocumentor</para>
  71. <programlisting role="php"><![CDATA[
  72. /**
  73. * This is a sample function
  74. *
  75. * @param string|base64 $val1 String or base64-encoded data
  76. * @param string|dateTime.iso8601 $val2 String or an ISO date
  77. * @param array|struct $val3 Normal indexed array or an associative array
  78. * @return boolean|struct
  79. */
  80. function myFunc($val1, $val2, $val3)
  81. {}
  82. ]]></programlisting>
  83. <para>Attention toutefois, une signature multiple peut prêter à confusion au regard des personnes utilisant
  84. votre service. En général une fonction ne devrait posséder qu'une seule signature.</para>
  85. </sect2>
  86. <sect2 id="zend.xmlrpc.server.namespaces">
  87. <title>Utiliser des espaces de noms (Namespaces)</title>
  88. <para>XML-RPC accepte le concept d'espace de noms, ce qui permet de grouper les méthodes XML-RPC. Ceci aide à
  89. prévenir les collisions de noms (deux fonctions avec le même nom), de différentes classes. Par exemple le
  90. serveur XML-RPC sert des méthodes dans l'espace "system" :</para>
  91. <itemizedlist>
  92. <listitem>
  93. <para>system.listMethods</para>
  94. </listitem>
  95. <listitem>
  96. <para>system.methodHelp</para>
  97. </listitem>
  98. <listitem>
  99. <para>system.methodSignature</para>
  100. </listitem>
  101. </itemizedlist>
  102. <para>En interne la correspondance est faite avec les méthodes du même nom, de
  103. <classname>Zend_XmlRpc_Server</classname>.</para>
  104. <para>Si vous voulez ajouter un espace de noms aux méthodes que vous servez, procédez alors comme suit :</para>
  105. <programlisting role="php"><![CDATA[
  106. // Toutes les méthodes publiques de My_Service_Class seront accessibles
  107. // via myservice.METHODNAME
  108. $server->setClass('My_Service_Class', 'myservice');
  109. // la fonction 'somefunc' sera accessible via funcs.somefunc
  110. $server->addFunction('somefunc', 'funcs');
  111. ]]></programlisting>
  112. </sect2>
  113. <sect2 id="zend.xmlrpc.server.request">
  114. <title>Requêtes personnalisées</title>
  115. <para>La plupart du temps, vous utiliserez l'objet de requête par défaut <classname>Zend_XmlRpc_Request_Http</classname>,
  116. sans vous en occuper. En revanche si vous avez un besoin spécifique, comme par exemple journaliser la requête,
  117. traiter une requête CLI, GUI, ou autre environnement, vous devrez alors créer un objet étendant
  118. <classname>Zend_XmlRpc_Request</classname>. Implémentez les méthodes <code>getMethod()</code> et <code>getParams()</code>
  119. afin que le serveur puisse analyser ces informations pour traiter la requête.</para>
  120. </sect2>
  121. <sect2 id="zend.xmlrpc.server.response">
  122. <title>Réponses personnalisées</title>
  123. <para>Comme avec les objets de requête, <classname>Zend_XmlRpc_Server</classname> peut retourner des objets de réponse
  124. personnalisés. Par défaut il s'agit d'objets <classname>Zend_XmlRpc_Response_Http</classname> qui envoient un en-tête HTTP
  125. Content-Type HTTP pour XML-RPC. Vous pourriez utiliser des objets de réponse personnalisés pour par exemple
  126. renvoyer les réponses vers STDOUT, ou les journaliser.</para>
  127. <para>Pour utiliser une classe de réponse personnalisée, utilisez
  128. <classname>Zend_XmlRpc_Server::setResponseClass()</classname> avant d'appeler <code>handle()</code>.</para>
  129. </sect2>
  130. <sect2 id="zend.xmlrpc.server.fault">
  131. <title>Gérer les exceptions grâce aux erreurs (Faults)</title>
  132. <para><classname>Zend_XmlRpc_Server</classname> attrape les Exceptions générées par vos classes/fonctions, et génère une
  133. réponse XML-RPC "fault" lorsqu'une exception a été rencontrée. Par défaut, les message et code des exceptions ne
  134. sont pas attachés dans la réponse XML-RPC. Ceci est du au fait que de telles exceptions peuvent en dire trop, au
  135. regard de la sécurité de votre application.</para>
  136. <para>Des classes d'exception peuvent cependant être mises en liste blanche, et donc utilisées pour les réponses
  137. d'erreur ("fault"). Utilisez simplement <classname>Zend_XmlRpc_Server_Fault::attachFaultException()</classname> en lui
  138. passant une classe d'exception :</para>
  139. <programlisting role="php"><![CDATA[
  140. Zend_XmlRpc_Server_Fault::attachFaultException('My_Project_Exception');
  141. ]]></programlisting>
  142. <para>Si vous héritez correctement vos exceptions, vous pouvez alors passer en liste blanche l'exception de plus
  143. bas niveau, et ainsi accepter plusieurs types d'exceptions qui en hériteront. Évidemment, les
  144. Zend_XmlRpc_Server_Exceptions sont elles automatiquement mises en liste blanche, afin de pouvoir traiter les
  145. requêtes vers des méthodes inexistantes, ou toute autre erreur "générique".</para>
  146. <para>Toute exception rencontrée, mais non mise en liste blanche, donnera naissance à une réponse d'erreur avec
  147. le code "404" et le message "Unknown error".</para>
  148. </sect2>
  149. <sect2 id="zend.xmlrpc.server.caching">
  150. <title>Cacher la définition du serveur entre les requêtes</title>
  151. <para>Attacher beaucoup de classes au serveur XML-RPC peut consommer beaucoup de ressources, car l'introspection
  152. de chaque classe/fonction est mise en place.</para>
  153. <para>Pour améliorer les performances, <classname>Zend_XmlRpc_Server_Cache</classname> peut être utilisé pour mettre en
  154. cache la définition d'un serveur. Combiné à <code>__autoload()</code>, ceci améliore grandement les
  155. performances.</para>
  156. <para>Un exemple d'utilisation :</para>
  157. <programlisting role="php"><![CDATA[
  158. function __autoload($class)
  159. {
  160. Zend_Loader::loadClass($class);
  161. }
  162. $cacheFile = dirname(__FILE__) . '/xmlrpc.cache';
  163. $server = new Zend_XmlRpc_Server();
  164. if (!Zend_XmlRpc_Server_Cache::get($cacheFile, $server)) {
  165. require_once 'My/Services/Glue.php';
  166. require_once 'My/Services/Paste.php';
  167. require_once 'My/Services/Tape.php';
  168. $server->setClass('My_Services_Glue', 'glue');
  169. // espace de noms glue
  170. $server->setClass('My_Services_Paste', 'paste');
  171. // espace de noms paste
  172. $server->setClass('My_Services_Tape', 'tape');
  173. // espace de noms tape
  174. Zend_XmlRpc_Server_Cache::save($cacheFile, $server);
  175. }
  176. echo $server->handle();
  177. ]]></programlisting>
  178. <para>L'exemple ci dessus essaye de récupérer la définition du serveur via le fichier
  179. <filename>xmlrpc.cache</filename>. Si ceci échoue, alors les classes nécessaires au service sont chargées,
  180. attachées au serveur, et une tentative de création de cache est lancée.</para>
  181. </sect2>
  182. <sect2 id="zend.xmlrpc.server.use">
  183. <title>Exemples d'utilisation</title>
  184. <para>Voici quelques exemples qui démontrent les diverses options disponibles pour un serveur XML-RPC.</para>
  185. <sect3 id="zend.xmlrpc.server.use.case1">
  186. <title>Utilisation basique</title>
  187. <para>L'exemple ci dessous attache une fonction au service XML-RPC.</para>
  188. <programlisting role="php"><![CDATA[
  189. /**
  190. * Retourne le hash MD5 d'une valeur
  191. *
  192. * @param string $value Valeur à hasher
  193. * @return string Hash MD5 de la valeur
  194. */
  195. function md5Value($value)
  196. {
  197. return md5($value);
  198. }
  199. $server = new Zend_XmlRpc_Server();
  200. $server->addFunction('md5Value');
  201. echo $server->handle();
  202. ]]></programlisting>
  203. </sect3>
  204. <sect3 id="zend.xmlrpc.server.use.case2">
  205. <title>Attacher une classe</title>
  206. <para>L'exemple ci dessous montre comment attacher les méthodes publiques d'une classe en tant que méthodes
  207. XML-RPC.</para>
  208. <programlisting role="php"><![CDATA[
  209. $server = new Zend_XmlRpc_Server();
  210. $server->setClass('Services_Comb');
  211. echo $server->handle();
  212. ]]></programlisting>
  213. </sect3>
  214. <sect3 id="zend.xmlrpc.server.use.case3">
  215. <title>Attacher plusieurs classes grâce aux espaces de noms</title>
  216. <para>L'exemple ci dessous montre comment attacher plusieurs classes grâce aux espaces de noms.</para>
  217. <programlisting role="php"><![CDATA[
  218. require_once 'Services/Comb.php';
  219. require_once 'Services/Brush.php';
  220. require_once 'Services/Pick.php';
  221. $server = new Zend_XmlRpc_Server();
  222. $server->setClass('Services_Comb', 'comb');
  223. // méthodes appelées sous la forme comb.*
  224. $server->setClass('Services_Brush', 'brush');
  225. // méthodes appelées sous la forme brush.*
  226. $server->setClass('Services_Pick', 'pick');
  227. // méthodes appelées sous la forme pick.*
  228. echo $server->handle();
  229. ]]></programlisting>
  230. </sect3>
  231. <sect3 id="zend.xmlrpc.server.use.case4">
  232. <title>Spécifier les exceptions à utiliser en cas d'erreurs dans les réponses XML-RPC</title>
  233. <para>L'exemple ci dessous montre comment spécifier les exceptions à utiliser en cas d'erreurs dans les
  234. réponses XML-RPC.</para>
  235. <programlisting role="php"><![CDATA[
  236. require_once 'Services/Exception.php';
  237. require_once 'Services/Comb.php';
  238. require_once 'Services/Brush.php';
  239. require_once 'Services/Pick.php';
  240. // Utilise les Services_Exception pour les erreurs
  241. Zend_XmlRpc_Server_Fault::attachFaultException('Services_Exception');
  242. $server = new Zend_XmlRpc_Server();
  243. $server->setClass('Services_Comb', 'comb');
  244. // méthodes appelées sous la forme comb.*
  245. $server->setClass('Services_Brush', 'brush');
  246. // méthodes appelées sous la forme brush.*
  247. $server->setClass('Services_Pick', 'pick');
  248. // méthodes appelées sous la forme pick.*
  249. echo $server->handle();
  250. ]]></programlisting>
  251. </sect3>
  252. <sect3 id="zend.xmlrpc.server.use.case5">
  253. <title>Utiliser un objet de requête personnalisé</title>
  254. <para>L'exemple suivant montre comment utiliser un objet de requête personnalisé.</para>
  255. <programlisting role="php"><![CDATA[
  256. require_once 'Services/Request.php';
  257. require_once 'Services/Exception.php';
  258. require_once 'Services/Comb.php';
  259. require_once 'Services/Brush.php';
  260. require_once 'Services/Pick.php';
  261. // Utilise les Services_Exception pour les erreurs
  262. Zend_XmlRpc_Server_Fault::attachFaultException('Services_Exception');
  263. $server = new Zend_XmlRpc_Server();
  264. $server->setClass('Services_Comb', 'comb');
  265. // méthodes appelées sous la forme comb.*
  266. $server->setClass('Services_Brush', 'brush');
  267. // méthodes appelées sous la forme brush.*
  268. $server->setClass('Services_Pick', 'pick');
  269. // méthodes appelées sous la forme pick.*
  270. // Crée un objet de requête
  271. $request = new Services_Request();
  272. echo $server->handle($request);
  273. ]]></programlisting>
  274. </sect3>
  275. <sect3 id="zend.xmlrpc.server.use.case6">
  276. <title>Utiliser un objet de réponse personnalisé</title>
  277. <para>L'exemple suivant montre comment utiliser un objet de réponse personnalisé.</para>
  278. <programlisting role="php"><![CDATA[
  279. require_once 'Services/Request.php';
  280. require_once 'Services/Response.php';
  281. require_once 'Services/Exception.php';
  282. require_once 'Services/Comb.php';
  283. require_once 'Services/Brush.php';
  284. require_once 'Services/Pick.php';
  285. // Utilise les Services_Exception pour les erreurs
  286. Zend_XmlRpc_Server_Fault::attachFaultException('Services_Exception');
  287. $server = new Zend_XmlRpc_Server();
  288. $server->setClass('Services_Comb', 'comb');
  289. // méthodes appelées sous la forme comb.*
  290. $server->setClass('Services_Brush', 'brush');
  291. // méthodes appelées sous la forme brush.*
  292. $server->setClass('Services_Pick', 'pick');
  293. // méthodes appelées sous la forme pick.*
  294. // Crée un objet de requête
  295. $request = new Services_Request();
  296. // Utilise une réponse personnalisée
  297. $server->setResponseClass('Services_Response');
  298. echo $server->handle($request);
  299. ]]></programlisting>
  300. </sect3>
  301. <sect3 id="zend.xmlrpc.server.use.case7">
  302. <title>Cache entre les requêtes</title>
  303. <para>Les exemples suivants montrent comment gérer une politique de cache inter-requêtes.</para>
  304. <programlisting role="php"><![CDATA[
  305. require_once 'Services/Request.php';
  306. require_once 'Services/Response.php';
  307. require_once 'Services/Exception.php';
  308. require_once 'Services/Comb.php';
  309. require_once 'Services/Brush.php';
  310. require_once 'Services/Pick.php';
  311. // Specifier un fichier de cache
  312. $cacheFile = dirname(__FILE__) . '/xmlrpc.cache';
  313. // Utilise les Services_Exception pour les erreurs
  314. Zend_XmlRpc_Server_Fault::attachFaultException('Services_Exception');
  315. $server = new Zend_XmlRpc_Server();
  316. // Essaye de récupérer la définition du serveur via le cache
  317. if (!Zend_XmlRpc_Server_Cache::get($cacheFile, $server)) {
  318. $server->setClass('Services_Comb', 'comb');
  319. // méthodes appelées sous la forme comb.*
  320. $server->setClass('Services_Brush', 'brush');
  321. // méthodes appelées sous la forme brush.*
  322. $server->setClass('Services_Pick', 'pick');
  323. // méthodes appelées sous la forme pick.*
  324. // Sauve le cache
  325. Zend_XmlRpc_Server_Cache::save($cacheFile, $server));
  326. }
  327. // Crée un objet de requête
  328. $request = new Services_Request();
  329. // Utilise une réponse personnalisée
  330. $server->setResponseClass('Services_Response');
  331. echo $server->handle($request);
  332. ]]></programlisting>
  333. </sect3>
  334. </sect2>
  335. </sect1>