Zend_Http_Client-Adapters.xml 35 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!-- Reviewed: no -->
  3. <!-- EN-Revision: 19419 -->
  4. <sect1 id="zend.http.client.adapters">
  5. <title>Zend_Http_Client - 接続アダプタ</title>
  6. <sect2 id="zend.http.client.adapters.overview">
  7. <title>概要</title>
  8. <para>
  9. <classname>Zend_Http_Client</classname> は、接続アダプタとして設計されています。
  10. 接続アダプタは実際にサーバへの接続を行うオブジェクトで、
  11. リクエストの書き込みやレスポンスの読み込みも行います。
  12. この接続アダプタは置き換えることができます。
  13. つまり、デフォルトの接続アダプタを継承して自分の好みにあうように変更できます。
  14. <acronym>HTTP</acronym> クライアントクラス全体を書き換える必要はありません。
  15. 同じインターフェイスを実装しているだけでいいのです。
  16. </para>
  17. <para>
  18. 現在、<classname>Zend_Http_Client</classname> クラスは四つの組み込み接続アダプタを提供しています。
  19. <itemizedlist>
  20. <listitem>
  21. <para>
  22. <classname>Zend_Http_Client_Adapter_Socket</classname> (デフォルト)
  23. </para>
  24. </listitem>
  25. <listitem>
  26. <para>
  27. <classname>Zend_Http_Client_Adapter_Proxy</classname>
  28. </para>
  29. </listitem>
  30. <listitem>
  31. <para>
  32. <classname>Zend_Http_Client_Adapter_Curl</classname>
  33. </para>
  34. </listitem>
  35. <listitem>
  36. <para>
  37. <classname>Zend_Http_Client_Adapter_Test</classname>
  38. </para>
  39. </listitem>
  40. </itemizedlist>
  41. </para>
  42. <para>
  43. <classname>Zend_Http_Client</classname> オブジェクトの接続アダプタを指定するには、
  44. 設定オプション 'adapter' を使用します。
  45. クライアントオブジェクトのインスタンスを作成する際に、オプション
  46. 'adapter' にアダプタの名前 (たとえば 'Zend_Http_Client_Adapter_Socket' など)
  47. を指定できます。あるいは、アダプタオブジェクトの変数
  48. (たとえば <code>new Zend_Http_Client_Adapter_Test</code> など) を指定することもできます。
  49. <classname>Zend_Http_Client->setConfig()</classname> メソッドを使用し、
  50. アダプタを後で設定することも可能です。
  51. </para>
  52. </sect2>
  53. <sect2 id="zend.http.client.adapters.socket">
  54. <title>ソケットアダプタ</title>
  55. <para>
  56. デフォルトの接続アダプタは <classname>Zend_Http_Client_Adapter_Socket</classname>
  57. です。明示的に接続アダプタを指定しない場合は、これが使用されます。
  58. Socket アダプタは <acronym>PHP</acronym> の組み込み関数 fsockopen()
  59. を使用しており、特別な拡張モジュールやコンパイルオプションは必要ありません。
  60. </para>
  61. <para>
  62. ソケットアダプタには、追加の設定オプションを指定できます。これは
  63. <classname>Zend_Http_Client->setConfig()</classname> で指定するか、
  64. あるいはクライアントのコンストラクタに渡します。
  65. <table id="zend.http.client.adapter.socket.configuration.table">
  66. <title>Zend_Http_Client_Adapter_Socket の設定パラメータ</title>
  67. <tgroup cols="4">
  68. <thead>
  69. <row>
  70. <entry>パラメータ</entry>
  71. <entry>説明</entry>
  72. <entry>予期する型</entry>
  73. <entry>デフォルト値</entry>
  74. </row>
  75. </thead>
  76. <tbody>
  77. <row>
  78. <entry>persistent</entry>
  79. <entry>持続的な <acronym>TCP</acronym> 接続を使用するかどうか</entry>
  80. <entry>boolean</entry>
  81. <entry>false</entry>
  82. </row>
  83. <row>
  84. <entry>ssltransport</entry>
  85. <entry><acronym>SSL</acronym> トランスポート層 (たとえば 'sslv2'、'tls')</entry>
  86. <entry>文字列</entry>
  87. <entry>ssl</entry>
  88. </row>
  89. <row>
  90. <entry>sslcert</entry>
  91. <entry><acronym>PEM</acronym> でエンコードした、<acronym>SSL</acronym> 証明書ファイルへのパス</entry>
  92. <entry>文字列</entry>
  93. <entry>null</entry>
  94. </row>
  95. <row>
  96. <entry>sslpassphrase</entry>
  97. <entry><acronym>SSL</acronym> 証明書ファイルのパスフレーズ</entry>
  98. <entry>文字列</entry>
  99. <entry>null</entry>
  100. </row>
  101. </tbody>
  102. </tgroup>
  103. </table>
  104. <note>
  105. <title>持続的な TCP 接続</title>
  106. <para>
  107. 持続的な <acronym>TCP</acronym> 接続を使用すると、<acronym>HTTP</acronym>
  108. リクエストの処理速度が向上する可能性があります。
  109. しかし、たいていの場合はその効果はごくわずかで、
  110. 接続先の <acronym>HTTP</acronym> サーバにかかる負荷が大きくなります。
  111. </para>
  112. <para>
  113. 持続的な <acronym>TCP</acronym> 接続を使用するのは、
  114. 同じサーバに頻繁に接続する場合で
  115. そのサーバが同時に多数の接続を処理できることがわかっている場合のみにすることを推奨します。
  116. いずれにせよ、このオプションを使用する前には
  117. クライアント側の速度だけでなくサーバ側の負荷についてもベンチマークをとるようにしましょう。
  118. </para>
  119. <para>
  120. さらに、持続的な接続を使用するときには
  121. <xref linkend="zend.http.client.configuration" />
  122. で説明した Keep-Alive <acronym>HTTP</acronym> リクエストも有効にしておくことを推奨します。
  123. そうしないと、持続的な接続の効果はほとんどといっていいほどなくなってしまいます。
  124. </para>
  125. </note>
  126. <note>
  127. <title>HTTPS SSL ストリームパラメータ</title>
  128. <para>
  129. <code>ssltransport, sslcert</code> および <code>sslpassphrase</code>
  130. は、<acronym>HTTPS</acronym> 接続で使用する <acronym>SSL</acronym> レイヤーにのみ関連するものです。
  131. </para>
  132. <para>
  133. たいていの場合はデフォルトの <acronym>SSL</acronym> 設定でうまく動作するでしょうが、
  134. 接続先のサーバが特別なクライアント設定を要求している場合は
  135. それにあわせた設定をする必要があるかもしれません。
  136. その場合は、
  137. <ulink url="http://www.php.net/manual/ja/transports.php#transports.inet">ここ</ulink>
  138. で <acronym>SSL</acronym> トランスポート層やオプションについての説明を参照ください。
  139. </para>
  140. </note>
  141. </para>
  142. <example id="zend.http.client.adapters.socket.example-1">
  143. <title>HTTPS トランスポート層の変更</title>
  144. <programlisting language="php"><![CDATA[
  145. // 設定パラメータを指定します
  146. $config = array(
  147. 'adapter' => 'Zend_Http_Client_Adapter_Socket',
  148. 'ssltransport' => 'tls'
  149. );
  150. // クライアントオブジェクトのインスタンスを作成します
  151. $client = new Zend_Http_Client('https://www.example.com', $config);
  152. // これ以降のリクエストは、TLS セキュア接続上で行われます
  153. $response = $client->request();
  154. ]]></programlisting>
  155. </example>
  156. <para>
  157. 上の例の結果は、次の <acronym>PHP</acronym> コマンドで <acronym>TCP</acronym>
  158. 接続をオープンした場合と同じになります。
  159. </para>
  160. <para>
  161. <methodname>fsockopen('tls://www.example.com', 443)</methodname>
  162. </para>
  163. <sect3 id="zend.http.client.adapters.socket.streamcontext">
  164. <title>ソケットアダプタのストリームコンテキストへのアクセスとカスタマイズ</title>
  165. <para>
  166. Zend Framework 1.9 以降、<classname>Zend_Http_Client_Adapter_Socket</classname>
  167. は、リモートサーバへの接続に使用している
  168. <ulink url="http://www.php.net/manual/ja/stream.contexts.php">ストリームコンテキスト</ulink>
  169. に直接アクセスできるようになります。これを使用すると、
  170. <acronym>TCP</acronym> ストリーム (<acronym>HTTPS</acronym> 接続の場合は <acronym>SSL</acronym> ラッパー)
  171. 固有のオプションやパラメータを渡せるようになります。
  172. </para>
  173. <para>
  174. ストリームコンテキストにアクセスするには、
  175. <classname>Zend_Http_Client_Adapter_Socket</classname>
  176. のこれらのメソッドを使用します。
  177. <itemizedlist>
  178. <listitem>
  179. <para>
  180. <firstterm><methodname>setStreamContext($context)</methodname></firstterm>
  181. アダプタが使用するストリームコンテキストを設定します。
  182. <acronym>PHP</acronym> の
  183. <ulink url="http://php.net/manual/ja/function.stream-context-create.php"><methodname>stream_context_create()</methodname></ulink>
  184. 関数で作成したストリームコンテキストリソースか、
  185. この関数に渡すのと同じ形式のストリームコンテキストオプションの配列のいずれかを指定できます。
  186. 配列を渡すと、そのオプションを用いて新たなストリームコンテキストを作成し、
  187. それを設定します。
  188. </para>
  189. </listitem>
  190. <listitem>
  191. <para>
  192. <firstterm><methodname>getStreamContext()</methodname></firstterm>
  193. アダプタのストリームコンテキストを取得します。
  194. ストリームコンテキストが設定されていない場合は、
  195. デフォルトのストリームコンテキストを作成してそれを返します。
  196. あとは、さまざまなコンテキストオプションの設定や取得を通常の
  197. <acronym>PHP</acronym> ストリームコンテキスト関数で行うことができます。
  198. </para>
  199. </listitem>
  200. </itemizedlist>
  201. </para>
  202. <example id="zend.http.client.adapters.socket.streamcontext.example-1">
  203. <title>ソケットアダプタのストリームコンテキストオプションの設定</title>
  204. <programlisting language="php"><![CDATA[
  205. // オプションの配列
  206. $options = array(
  207. 'socket' => array(
  208. // ソケットのローカル側を特定のインターフェイスにバインドします
  209. 'bindto' => '10.1.2.3:50505'
  210. ),
  211. 'ssl' => array(
  212. // サーバ側の証明書を検証します
  213. // 無効な証明書や自己署名の SSL 証明書は拒否します
  214. 'verify_peer' => true,
  215. 'allow_self_signed' => false,
  216. // ピア証明書を捕捉します
  217. 'capture_peer_cert' => true
  218. )
  219. );
  220. // アダプタオブジェクトを作成し、HTTP クライアントにバインドします
  221. $adapter = new Zend_Http_Client_Adapter_Socket();
  222. $client = new Zend_Http_Client();
  223. $client->setAdapter($adapter);
  224. // 方法 1: オプションの配列を setStreamContext() に渡します
  225. $adapter->setStreamContext($options);
  226. // 方法 2: ストリームコンテキストを作成して setStreamContext() に渡します
  227. $context = stream_context_create($options);
  228. $adapter->setStreamContext($context);
  229. // 方法 3: デフォルトのストリームコンテキストを取得してオプションを設定します
  230. $context = $adapter->getStreamContext();
  231. stream_context_set_option($context, $options);
  232. // リクエストを処理します
  233. $response = $client->request();
  234. // すべてがうまくいけば、これでまたコンテキストにアクセスできます
  235. $opts = stream_context_get_options($adapter->getStreamContext());
  236. echo $opts['ssl']['peer_certificate'];
  237. ]]></programlisting>
  238. </example>
  239. <note>
  240. <para>
  241. ストリームコンテキストのオプションは、
  242. アダプタが実際にリクエストを処理しだす前に設定しなければならないことに注意しましょう。
  243. コンテキストを設定せずにソケットアダプタで <acronym>HTTP</acronym> リクエストを処理すると、
  244. デフォルトのストリームコンテキストが作成されます。
  245. リクエストを処理した後にこのコンテキストリソースにアクセスするには
  246. <methodname>getStreamContext()</methodname> メソッドを使用します。
  247. </para>
  248. </note>
  249. </sect3>
  250. </sect2>
  251. <sect2 id="zend.http.client.adapters.proxy">
  252. <title>プロキシアダプタ</title>
  253. <para>
  254. <classname>Zend_Http_Client_Adapter_Proxy</classname> アダプタはデフォルトのソケットアダプタとほぼ同じです。
  255. ただし、対象となるサーバに直接接続するのではなく
  256. <acronym>HTTP</acronym> プロキシサーバを経由して接続するという点が異なります。
  257. これにより、<classname>Zend_Http_Client</classname> をプロキシサーバの中から使用できるようになります。
  258. セキュリティやパフォーマンス上の理由により、これが必要となる場合もあるでしょう。
  259. </para>
  260. <para>
  261. プロキシアダプタを使用するには、
  262. デフォルトの 'adapter' オプション以外に
  263. いくつか追加のパラメータを設定する必要があります。
  264. <table id="zend.http.client.adapters.proxy.table">
  265. <title>Zend_Http_Client の設定パラメータ</title>
  266. <tgroup cols="4">
  267. <thead>
  268. <row>
  269. <entry>パラメータ</entry>
  270. <entry>説明</entry>
  271. <entry>想定している型</entry>
  272. <entry>値の例</entry>
  273. </row>
  274. </thead>
  275. <tbody>
  276. <row>
  277. <entry>proxy_host</entry>
  278. <entry>プロキシサーバのアドレス</entry>
  279. <entry>string</entry>
  280. <entry>'proxy.myhost.com' あるいは '10.1.2.3'</entry>
  281. </row>
  282. <row>
  283. <entry>proxy_port</entry>
  284. <entry>プロキシサーバの <acronym>TCP</acronym> ポート</entry>
  285. <entry>integer</entry>
  286. <entry>8080 (デフォルト) あるいは 81</entry>
  287. </row>
  288. <row>
  289. <entry>proxy_user</entry>
  290. <entry>必要に応じて、プロキシのユーザ名</entry>
  291. <entry>string</entry>
  292. <entry>'shahar' あるいは指定しない場合は '' (デフォルト)</entry>
  293. </row>
  294. <row>
  295. <entry>proxy_pass</entry>
  296. <entry>必要に応じて、プロキシのパスワード</entry>
  297. <entry>string</entry>
  298. <entry>'secret' あるいは指定しない場合は '' (デフォルト)</entry>
  299. </row>
  300. <row>
  301. <entry>proxy_auth</entry>
  302. <entry>プロキシの <acronym>HTTP</acronym> 認証形式</entry>
  303. <entry>string</entry>
  304. <entry>Zend_Http_Client::AUTH_BASIC (デフォルト)</entry>
  305. </row>
  306. </tbody>
  307. </tgroup>
  308. </table>
  309. </para>
  310. <para>
  311. proxy_host は常に設定しなければなりません。指定しなかった場合は、
  312. 自動的に <classname>Zend_Http_Client_Adapter_Socket</classname> による直接接続に切り替わります。
  313. proxy_port のデフォルトは '8080' です。もし別のポートをプロキシで使用している場合は、
  314. 適切に設定する必要があります。
  315. </para>
  316. <para>
  317. proxy_user および proxy_pass は、
  318. プロキシサーバが認証を必要とする場合にのみ設定します。
  319. これらを指定すると、'Proxy-Authentication'
  320. ヘッダがリクエストに追加されます。プロキシで認証を必要としない場合は、
  321. このふたつのオプションはそのままにしておきます。
  322. </para>
  323. <para>
  324. proxy_auth は、プロキシが認証を必要としている場合に、
  325. その認証形式を指定します。設定できる値は
  326. Zend_Http_Client::setAuth() メソッドと同じです。現在はベーシック認証
  327. (Zend_Http_Client::AUTH_BASIC) のみをサポートしています。
  328. </para>
  329. <example id="zend.http.client.adapters.proxy.example-1">
  330. <title>プロキシサーバを使用した Zend_Http_Client の使用法</title>
  331. <programlisting language="php"><![CDATA[
  332. // 接続パラメータを設定します
  333. $config = array(
  334. 'adapter' => 'Zend_Http_Client_Adapter_Proxy',
  335. 'proxy_host' => 'proxy.int.zend.com',
  336. 'proxy_port' => 8000,
  337. 'proxy_user' => 'shahar.e',
  338. 'proxy_pass' => 'bananashaped'
  339. );
  340. // クライアントオブジェクトのインスタンスを作成します
  341. $client = new Zend_Http_Client('http://www.example.com', $config);
  342. // 作業を続けます...
  343. ]]></programlisting>
  344. </example>
  345. <para>
  346. 説明したとおり、もし proxy_host を省略したり空文字列を設定したりすると、
  347. 自動的に直接接続に切り替わります。これにより、設定パラメータによって
  348. オプションでプロキシを使用できるようなアプリケーションを書くことが可能となります。
  349. </para>
  350. <note>
  351. <para>
  352. プロキシアダプタは <classname>Zend_Http_Client_Adapter_Socket</classname>
  353. を継承しているので、ストリームコンテキストへのアクセスメソッド
  354. (<xref linkend="zend.http.client.adapters.socket.streamcontext" /> を参照ください)
  355. を使用してプロキシ接続におけるストリームコンテキストオプションを設定できます。
  356. その方法については上で説明しました。
  357. </para>
  358. </note>
  359. </sect2>
  360. <sect2 id="zend.http.client.adapters.curl">
  361. <title>cURL アダプタ</title>
  362. <para>
  363. cURL は標準的な <acronym>HTTP</acronym> クライアントライブラリで、
  364. 多くの OS に含まれています。また <acronym>PHP</acronym> からは cURL
  365. 拡張モジュールで使用できます。
  366. <acronym>HTTP</acronym> クライアントで起こりうる多くの特別な例にも対応することができるので、
  367. <acronym>HTTP</acronym> アダプタとしては完璧な選択肢といえるでしょう。
  368. セキュアな接続やプロキシ、そしてあらゆる種類の認証にも対応しており、
  369. 大きなファイルをサーバ間で移動させるときなどにも使用できます。
  370. </para>
  371. <example id="zend.http.client.adapters.curl.example-1">
  372. <title>cURL オプションの設定</title>
  373. <programlisting language="php"><![CDATA[
  374. $config = array(
  375. 'adapter' => 'Zend_Http_Client_Adapter_Curl',
  376. 'curloptions' => array(CURLOPT_FOLLOWLOCATION => true),
  377. );
  378. $client = new Zend_Http_Client($uri, $config);
  379. ]]></programlisting>
  380. </example>
  381. <para>
  382. デフォルトでは、cURL アダプタは
  383. Socket アダプタとまったく同じ挙動となるように設定されています。
  384. また、Socket アダプタおよび Proxy アダプタと同じ設定パラメータを使えます。
  385. cURL のオプションを変更するには、アダプタのコンストラクタでキー
  386. 'curloptions' を指定するか、あるいは
  387. <methodname>setCurlOption($name, $value)</methodname> をコールします。
  388. <code>$name</code> は、cURL 拡張モジュールの
  389. CURL_* 定数に対応します。Curl ハンドルにアクセスするには
  390. <code>$adapter->getHandle();</code> をコールします。
  391. </para>
  392. <example id="zend.http.client.adapters.curl.example-2">
  393. <title>ハンドルによるファイル転送</title>
  394. <para>
  395. cURL を使用して、巨大なファイルを <acronym>HTTP</acronym> 越しに転送できます。
  396. </para>
  397. <programlisting language="php"><![CDATA[
  398. $putFileSize = filesize("filepath");
  399. $putFileHandle = fopen("filepath", "r");
  400. $adapter = new Zend_Http_Client_Adapter_Curl();
  401. $client = new Zend_Http_Client();
  402. $client->setAdapter($adapter);
  403. $adapter->setConfig(array(
  404. 'curloptions' => array(
  405. CURLOPT_INFILE => $putFileHandle,
  406. CURLOPT_INFILESIZE => $putFileSize
  407. )
  408. ));
  409. $client->request("PUT");
  410. ]]></programlisting>
  411. </example>
  412. </sect2>
  413. <sect2 id="zend.http.client.adapters.test">
  414. <title>テストアダプタ</title>
  415. <para>
  416. <acronym>HTTP</acronym> 接続に依存するテストコードを書くのは非常に難しいものです。
  417. たとえば、リモートサーバから <acronym>RSS</acronym> を取得するアプリケーションをテストするには、
  418. ネットワークにつながっている必要があります。常にネットワークが使用できるとは限りません。
  419. </para>
  420. <para>
  421. このようなときのためにあるのが <classname>Zend_Http_Client_Adapter_Test</classname> アダプタです。
  422. <classname>Zend_Http_Client</classname> を使用するアプリケーションを作成し、それをテストしたい場合には、
  423. デフォルトのアダプタを Test アダプタ (モックオブジェクト) に変更します。
  424. これで、サーバに接続せずにテストを行えるようになります。
  425. </para>
  426. <para>
  427. <classname>Zend_Http_Client_Adapter_Test</classname> には setResponse() というメソッドがあります。
  428. このメソッドのパラメータには、<acronym>HTTP</acronym> レスポンスをテキストか
  429. <classname>Zend_Http_Response</classname> オブジェクトで指定できます。
  430. レスポンスを設定すると、Test アダプタは常にこのレスポンスを返すようになります。
  431. 実際の <acronym>HTTP</acronym> リクエストは行いません。
  432. </para>
  433. <example id="zend.http.client.adapters.test.example-1">
  434. <title>HTTP レスポンススタブを使用したテスト</title>
  435. <programlisting language="php"><![CDATA[
  436. // 新しいアダプタとクライアントのインスタンスを作成します
  437. $adapter = new Zend_Http_Client_Adapter_Test();
  438. $client = new Zend_Http_Client('http://www.example.com', array(
  439. 'adapter' => $adapter
  440. ));
  441. // 想定するレスポンスを設定します
  442. $adapter->setResponse(
  443. "HTTP/1.1 200 OK" . "\r\n" .
  444. "Content-type: text/xml" . "\r\n" .
  445. "\r\n" .
  446. '<?xml version="1.0" encoding="UTF-8"?>' .
  447. '<rss version="2.0" ' .
  448. ' xmlns:content="http://purl.org/rss/1.0/modules/content/"' .
  449. ' xmlns:wfw="http://wellformedweb.org/CommentAPI/"' .
  450. ' xmlns:dc="http://purl.org/dc/elements/1.1/">' .
  451. ' <channel>' .
  452. ' <title>Premature Optimization</title>' .
  453. // などなど...
  454. '</rss>');
  455. $response = $client->request('GET');
  456. // .. $response の処理を続けます...
  457. ]]></programlisting>
  458. </example>
  459. <para>
  460. 上の例のようにすると、<acronym>HTTP</acronym> クライアントにお望みのレスポンスを返させることができます。
  461. その際にネットワーク接続は使用しません。また、実際のサーバからのレスポンスも使用しません。
  462. この場合、このテストでテストするのは、
  463. レスポンス本文の <acronym>XML</acronym> をアプリケーションが正しくパースできるかどうかということです。
  464. </para>
  465. <para>
  466. 時には、オブジェクトに対するひとつのメソッド呼び出しの中で複数の
  467. <acronym>HTTP</acronym> トランザクションを行うこともあるでしょう。そのような場合は
  468. setResponse() を単独で使うことはできません。なぜなら、
  469. 結果が呼び出し元に返ってくるまで次のレスポンスを設定できないからです。
  470. </para>
  471. <example id="zend.http.client.adapters.test.example-2">
  472. <title>複数の HTTP レスポンススタブを使用したテスト</title>
  473. <programlisting language="php"><![CDATA[
  474. // 新しいアダプタおよびクライアントのインスタンスを作成します
  475. $adapter = new Zend_Http_Client_Adapter_Test();
  476. $client = new Zend_Http_Client('http://www.example.com', array(
  477. 'adapter' => $adapter
  478. ));
  479. // 最初の応答として期待する値を設定します
  480. $adapter->setResponse(
  481. "HTTP/1.1 302 Found" . "\r\n" .
  482. "Location: /" . "\r\n" .
  483. "Content-Type: text/html" . "\r\n" .
  484. "\r\n" .
  485. '<html>' .
  486. ' <head><title>Moved</title></head>' .
  487. ' <body><p>This page has moved.</p></body>' .
  488. '</html>');
  489. // それに続くレスポンスを設定します
  490. $adapter->addResponse(
  491. "HTTP/1.1 200 OK" . "\r\n" .
  492. "Content-Type: text/html" . "\r\n" .
  493. "\r\n" .
  494. '<html>' .
  495. ' <head><title>My Pet Store Home Page</title></head>' .
  496. ' <body><p>...</p></body>' .
  497. '</html>');
  498. // HTTP クライアントオブジェクト ($client) をテスト対象の
  499. // オブジェクトに注入し、オブジェクトの動きを以下でテストします
  500. ]]></programlisting>
  501. </example>
  502. <para>
  503. setResponse() メソッドは、
  504. <classname>Zend_Http_Client_Adapter_Test</classname> のバッファにあるレスポンスをすべて削除し、
  505. 最初に返されるレスポンスを設定します。addResponse()
  506. メソッドは、それに続くレスポンスを追加します。
  507. </para>
  508. <para>
  509. レスポンスは、それを追加した順に再生されます。
  510. 登録したよりも多くのリクエストが発生した場合は、
  511. 返されるレスポンスは最初のものに戻り、そこからまた順に返されるようになります。
  512. </para>
  513. <para>
  514. 上の例で、このアダプタがテストするように設定されているのは、
  515. 302 リダイレクトが発生した場合のオブジェクトの挙動です。
  516. アプリケーションの内容によって、リダイレクトさせるべきなのかそうでないのかは異なるでしょう。
  517. この例ではリダイレクトさせることを想定しているので、
  518. テストアダプタもそれにあわせて設定しています。
  519. 最初の 302 レスポンスを setResponse() メソッドで設定し、
  520. 次に返される 200 レスポンスを addResponse() メソッドで設定します。
  521. テストアダプタを設定し終えたら、そのアダプタを含む <acronym>HTTP</acronym>
  522. クライアントをテスト対象オブジェクトに注入し、その挙動をテストします。
  523. </para>
  524. <para>
  525. アダプタをわざと失敗させたい場合は
  526. <methodname>setNextRequestWillFail($flag)</methodname> を使用します。
  527. このメソッドは、次に <methodname>connect()</methodname> をコールしたときに
  528. <classname>Zend_Http_Client_Adapter_Exception</classname>
  529. を発生させます。これは、外部のサイトのコンテンツをキャッシュするアプリケーションで、
  530. (外部サイトがダウンしていたときの) 挙動をテストする際に有用です。
  531. </para>
  532. <example id="zend.http.client.adapters.test.example-3">
  533. <title>アダプタを失敗させる</title>
  534. <programlisting language="php"><![CDATA[
  535. // 新たなアダプタとクライアントを作成します
  536. $adapter = new Zend_Http_Client_Adapter_Test();
  537. $client = new Zend_Http_Client('http://www.example.com', array(
  538. 'adapter' => $adapter
  539. ));
  540. // 次のリクエストでわざと例外を発生させます
  541. $adapter->nextRequestWillFail(true);
  542. try {
  543. // これは Zend_Http_Client_Adapter_Exception となります
  544. $client->request();
  545. } catch (Zend_Http_Client_Adapter_Exception $e) {
  546. // ...
  547. }
  548. // これ以降の処理は、 setNextRequestWillFail(true) を次に呼び出すまで
  549. //通常通りに行います
  550. ]]></programlisting>
  551. </example>
  552. </sect2>
  553. <sect2 id="zend.http.client.adapters.extending">
  554. <title>独自の接続アダプタの作成</title>
  555. <para>
  556. 独自の接続アダプタを作成し、それを使用することもできます。
  557. たとえば持続的なソケットを使用するアダプタを作成したり、
  558. キャッシュ機能を追加したアダプタを作成したりなど、
  559. 作成するアプリケーションの要件にあわせたものを作成することが可能です。
  560. </para>
  561. <para>
  562. そのためには、<classname>Zend_Http_Client_Adapter_Interface</classname>
  563. を実装したクラスを作成する必要があります。
  564. 以下の例は、ユーザ定義のアダプタクラスの雛形となります。
  565. この例で定義されているすべてのパブリック関数を、
  566. アダプタで定義する必要があります。
  567. </para>
  568. <example id="zend.http.client.adapters.extending.example-1">
  569. <title>独自の接続アダプタの作成</title>
  570. <programlisting language="php"><![CDATA[
  571. class MyApp_Http_Client_Adapter_BananaProtocol
  572. implements Zend_Http_Client_Adapter_Interface
  573. {
  574. /**
  575. * アダプタの設定配列を設定する
  576. *
  577. * @param array $config
  578. */
  579. public function setConfig($config = array())
  580. {
  581. // ここはほとんど変更することはありません -
  582. // 通常は Zend_Http_Client_Adapter_Socket の実装をコピーします
  583. }
  584. /**
  585. * リモートサーバに接続する
  586. *
  587. * @param string $host
  588. * @param int $port
  589. * @param boolean $secure
  590. */
  591. public function connect($host, $port = 80, $secure = false)
  592. {
  593. // リモートサーバとの接続を確立します
  594. }
  595. /**
  596. * リクエストをリモートサーバに送信する
  597. *
  598. * @param string $method
  599. * @param Zend_Uri_Http $url
  600. * @param string $http_ver
  601. * @param array $headers
  602. * @param string $body
  603. * @return string Request as text
  604. */
  605. public function write($method,
  606. $url,
  607. $http_ver = '1.1',
  608. $headers = array(),
  609. $body = '')
  610. {
  611. // リクエストをリモートサーバに送信します。
  612. // この関数は、リクエスト全体 (ヘッダおよび本文)
  613. // を文字列で返します。
  614. }
  615. /**
  616. * サーバからのレスポンスを読み込む
  617. *
  618. * @return string
  619. */
  620. public function read()
  621. {
  622. // リモートサーバからのレスポンスを読み込み、それを文字列で返します。
  623. }
  624. /**
  625. * サーバとの接続を閉じる
  626. *
  627. */
  628. public function close()
  629. {
  630. // リモートサーバとの接続を閉じます。最後にコールされます。
  631. }
  632. }
  633. // そして、このアダプタを使用します
  634. $client = new Zend_Http_Client(array(
  635. 'adapter' => 'MyApp_Http_Client_Adapter_BananaProtocol'
  636. ));
  637. ]]></programlisting>
  638. </example>
  639. </sect2>
  640. </sect1>