Zend_Session-AdvancedUsage.xml 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!-- Reviewed: no -->
  3. <!-- EN-Revision: 15103 -->
  4. <sect1 id="zend.session.advanced_usage">
  5. <title>高度な使用法</title>
  6. <para>
  7. 基本的な使用法の例で Zend Framework のセッションを完全に使用することができますが、
  8. よりよい方法もあります。ここでは、セッションの処理方法や
  9. Zend_Session コンポーネントのより行動な使用法を説明します。
  10. </para>
  11. <sect2 id="zend.session.advanced_usage.starting_a_session">
  12. <title>セッションの開始</title>
  13. <para>
  14. すべてのリクエストで Zend_Session の機能を使用してセッション管理したい場合は、
  15. 起動ファイルでセッションを開始します。
  16. </para>
  17. <example id="zend.session.advanced_usage.starting_a_session.example">
  18. <title>グローバルセッションの開始</title>
  19. <programlisting role="php"><![CDATA[
  20. Zend_Session::start();
  21. ]]>
  22. </programlisting>
  23. </example>
  24. <para>
  25. 起動ファイルでセッションを開始する際には、
  26. ヘッダがブラウザに送信される前に確実にセッションが始まるようにします。
  27. そうしないと例外が発生してしまい、おそらくユーザが見るページは崩れてしまうでしょう。
  28. さまざまな高度な機能を使用するには、まず <classname>Zend_Session::start()</classname>
  29. が必要です (高度な機能の詳細については後で説明します)。
  30. </para>
  31. <para>
  32. Zend_Session を使用してセッションを開始する方法は四通りありますが、
  33. そのうち二つは間違った方法です。
  34. </para>
  35. <orderedlist>
  36. <listitem>
  37. <para>
  38. 間違い: PHP の
  39. <ulink url="http://www.php.net/manual/ja/ref.session.php#ini.session.auto-start"><code>session.auto_start</code>
  40. </ulink> を有効にしてはいけません。
  41. もし mod_php (やそれと同等のもの) を使用しており、
  42. <code>php.ini</code> でこの設定が有効になっている、かつそれを無効にすることができない
  43. という場合は、<code>.htaccess</code> ファイル (通常は HTML のドキュメントルートにあります)
  44. に以下の内容を追加します。
  45. <programlisting role="httpd.conf"><![CDATA[
  46. php_value session.auto_start 0
  47. ]]>
  48. </programlisting>
  49. </para>
  50. </listitem>
  51. <listitem>
  52. <para>
  53. 間違い: PHP の
  54. <ulink url="http://www.php.net/session_start"><code>session_start()</code></ulink>
  55. 関数を直接使用してはいけません。
  56. <code>session_start()</code> を直接使用した後で <classname>Zend_Session_Namespace</classname> を使用した場合は、
  57. <classname>Zend_Session::start()</classname> が例外 ("session has already been started")
  58. をスローします。<classname>Zend_Session_Namespace</classname> を使用するか
  59. 明示的に <classname>Zend_Session::start()</classname> で開始した後で
  60. <code>session_start()</code> をコールすると、<code>E_NOTICE</code>
  61. が発生し、そのコールは無視されます。
  62. </para>
  63. </listitem>
  64. <listitem>
  65. <para>
  66. 正解: <classname>Zend_Session::start()</classname> を使用します。
  67. すべてのリクエストでセッションを使用したい場合は、
  68. この関数コールを起動コードの最初のほうで無条件に記述します。
  69. セッションにはある程度のオーバーヘッドがあります。
  70. セッションを使用したいリクエストとそうでないリクエストがある場合は、
  71. </para>
  72. <itemizedlist mark="opencircle">
  73. <listitem>
  74. <para>
  75. 起動コード内で、<classname>Zend_Session::setOptions()</classname> を使用して
  76. 無条件にオプション <code>strict</code> を <code>true</code> にします。
  77. </para>
  78. </listitem>
  79. <listitem>
  80. <para>
  81. セッションを必要とするリクエスト内で、
  82. <classname>Zend_Session_Namespace</classname> のインスタンスを作成する前に
  83. <classname>Zend_Session::start()</classname> をコールします。
  84. </para>
  85. </listitem>
  86. <listitem>
  87. <para>
  88. 通常どおり、必要に応じて "<code>new Zend_Session_Namespace()</code>"
  89. を使用します。事前に <classname>Zend_Session::start()</classname>
  90. がコールされていることを確認しておきましょう。
  91. </para>
  92. </listitem>
  93. </itemizedlist>
  94. <para>
  95. <code>strict</code> オプションにより、<code>new Zend_Session_Namespace()</code>
  96. が自動的に <classname>Zend_Session::start()</classname> でセッションを開始することがなくなります。
  97. したがって、このオプションを使用すると、アプリケーションの開発者が
  98. 特定のリクエストにはセッションを使用しないという設計をおこなうことができます。
  99. このオプションを使用すると、明示的に
  100. <classname>Zend_Session::start()</classname> をコールする前に Zend_Session_Namespace
  101. のインスタンスを作成しようとしたときに例外がスローされます。
  102. 開発者は、<classname>Zend_Session::setOptions()</classname>
  103. の使用がユーザにどれだけの影響を与えるかを注意するようにしましょう。
  104. これらのオプションは
  105. (もととなる ext/session のオプションと同様)、
  106. 全体に副作用を及ぼすからです。
  107. </para>
  108. </listitem>
  109. <listitem>
  110. <para>
  111. 正解: 必要に応じて <classname>Zend_Session_Namespace</classname> のインスタンスを作成します。
  112. PHP のセッションは、自動的に開始されます。
  113. これはもっともシンプルな使用法で、たいていの場合にうまく動作します。
  114. しかし、デフォルトであるクッキーベースのセッション (強く推奨します)
  115. を使用している場合には、PHP がクライアントに何らかの出力
  116. (<ulink url="http://www.php.net/headers_sent">HTTP ヘッダ</ulink> など)
  117. をする <emphasis role="strong">前に</emphasis>、確実に
  118. 最初の <code>new Zend_Session_Namespace()</code> をコールしなければなりません。
  119. 詳細は <xref linkend="zend.session.global_session_management.headers_sent" />
  120. を参照ください。
  121. </para>
  122. </listitem>
  123. </orderedlist>
  124. </sect2>
  125. <sect2 id="zend.session.advanced_usage.locking">
  126. <title>セッション名前空間のロック</title>
  127. <para>
  128. セッション名前空間をロックし、
  129. それ以降その名前空間のデータに手を加えられないようにすることができます。
  130. 特定の名前空間を読み取り専用にするには
  131. <code>lock()</code> を、そして
  132. 読み取り専用の名前空間を読み書きできるようにするには <code>unLock()</code>
  133. を使用します。<code>isLocked()</code> を使用すると、
  134. その名前空間がロックされているかどうかを調べることができます。
  135. このロックは一時的なものであり、そのリクエスト内でのみ有効となります。
  136. 名前空間をロックしても、その名前空間に保存されているオブジェクトの
  137. セッターメソッドには何の影響も及ぼしません。
  138. しかし、名前空間自体のセッターメソッドは使用できず、
  139. 名前空間に直接格納されたオブジェクトの削除や置換ができなくなります。同様に、
  140. <classname>Zend_Session_Namespace</classname> のインスタンスをロックしたとしても、
  141. 同じデータをさすシンボルテーブルの使用をとめることはできません
  142. (<ulink url="http://www.php.net/references">PHP
  143. のリファレンスについての説明</ulink>も参照ください)。
  144. </para>
  145. <example id="zend.session.advanced_usage.locking.example.basic">
  146. <title>セッション名前空間のロック</title>
  147. <programlisting role="php"><![CDATA[
  148. $userProfileNamespace = new Zend_Session_Namespace('userProfileNamespace');
  149. // このセッションに読み取り専用ロックをかけます
  150. $userProfileNamespace->lock();
  151. // 読み取り専用ロックを解除します
  152. if ($userProfileNamespace->isLocked()) {
  153. $userProfileNamespace->unLock();
  154. }
  155. ]]>
  156. </programlisting>
  157. </example>
  158. </sect2>
  159. <sect2 id="zend.session.advanced_usage.expiration">
  160. <title>名前空間の有効期限</title>
  161. <para>
  162. 名前空間およびその中の個々のキーについて、その寿命を制限することができます。
  163. これは、たとえばリクエスト間で一時的な情報を渡す際に使用します。
  164. これにより、認証内容などの機密情報へアクセスできないようにし、
  165. セキュリティリスクを下げます。有効期限の設定は経過秒数によって決めることもできますし、
  166. "ホップ" 数によって決めることもできます。ホップ数とは、
  167. 一連のリクエストの回数を表します。
  168. </para>
  169. <example id="zend.session.advanced_usage.expiration.example">
  170. <title>有効期限切れの例</title>
  171. <programlisting role="php"><![CDATA[
  172. $s = new Zend_Session_Namespace('expireAll');
  173. $s->a = 'apple';
  174. $s->p = 'pear';
  175. $s->o = 'orange';
  176. $s->setExpirationSeconds(5, 'a'); // キー "a" だけは 5 秒で有効期限切れとなります
  177. // 名前空間全体は、5 "ホップ" で有効期限切れとなります
  178. $s->setExpirationHops(5);
  179. $s->setExpirationSeconds(60);
  180. // "expireAll" 名前空間は、60 秒が経過するか
  181. // 5 ホップに達するかのどちらかが発生した時点で
  182. // "有効期限切れ" となります
  183. ]]>
  184. </programlisting>
  185. </example>
  186. <para>
  187. 現在のリクエストで期限切れになったデータを扱うにあたり、
  188. データを取得する際には注意が必要です。
  189. データは参照で返されますが、それを変更したとしても
  190. 期限切れのデータを現在のリクエストから持ち越すことはできません。
  191. 有効期限を "リセット" するには、取得したデータをいったん一時変数に格納し、
  192. 名前空間上の内容を削除し、あらためて適切なキーで再設定します。
  193. </para>
  194. </sect2>
  195. <sect2 id="zend.session.advanced_usage.controllers">
  196. <title>コントローラでのセッションのカプセル化</title>
  197. <para>
  198. 名前空間を使用すると、コントローラによるセッションへのアクセスの際に
  199. 変数の汚染を防ぐこともできます。
  200. たとえば、認証コントローラでは、セキュリティの観点から
  201. そのセッション状態データを他のコントローラとは別に管理することになるでしょう。
  202. </para>
  203. <example id="zend.session.advanced_usage.controllers.example">
  204. <title>コントローラでの名前空間つきセッションによる有効期限の管理</title>
  205. <para>
  206. 次のコードは、質問を表示するコントローラの一部です。
  207. ここでは論理型の変数を用意して、質問に対する回答を受け付けるかどうかを表しています。
  208. この場合は、表示されている質問に 300 秒以内に答えることになります。
  209. </para>
  210. <programlisting role="php"><![CDATA[
  211. // ...
  212. // 質問を表示するコントローラ
  213. $testSpace = new Zend_Session_Namespace('testSpace');
  214. // この変数にだけ有効期限を設定します
  215. $testSpace->setExpirationSeconds(300, 'accept_answer');
  216. $testSpace->accept_answer = true;
  217. //...
  218. ]]>
  219. </programlisting>
  220. <para>
  221. 次に、回答を処理するコントローラを示します。
  222. 時間内に回答したかどうかをもとにして、回答を受け付けるかどうかを判断しています。
  223. </para>
  224. <programlisting role="php"><![CDATA[
  225. // ...
  226. // 回答を処理するコントローラ
  227. $testSpace = new Zend_Session_Namespace('testSpace');
  228. if ($testSpace->accept_answer === true) {
  229. // 時間内
  230. }
  231. else {
  232. // 時間切れ
  233. }
  234. // ...
  235. ]]>
  236. </programlisting>
  237. </example>
  238. </sect2>
  239. <sect2 id="zend.session.advanced_usage.single_instance">
  240. <title>名前空間内あたりのインスタンス数をひとつに絞り込む</title>
  241. <para>
  242. <link linkend="zend.session.advanced_usage.locking">セッションのロック</link>
  243. を利用すれば、名前空間つきセッションデータを予期せず使用してしまうことはある程度防げます。
  244. しかし、<classname>Zend_Session_Namespace</classname> には、
  245. 単一の名前空間内で複数のインスタンスを作成することを防ぐ機能もあります。
  246. </para>
  247. <para>
  248. この機能を有効にするには、<classname>Zend_Session_Namespace</classname>
  249. のインスタンスを作成する際に、コンストラクタの第二引数に <code>true</code>
  250. を渡します。それ以降は、同一名前空間でインスタンスを作成しようとすると例外がスローされます。
  251. </para>
  252. <example id="zend.session.advanced_usage.single_instance.example">
  253. <title>セッション名前空間へのアクセスを単一のインスタンスに制限する</title>
  254. <programlisting role="php"><![CDATA[
  255. // 名前空間のインスタンスを作成します
  256. $authSpaceAccessor1 = new Zend_Session_Namespace('Zend_Auth');
  257. // 同じ名前空間で別のインスタンスを作成します。
  258. // しかし今後はインスタンスを作成できないようにします
  259. $authSpaceAccessor2 = new Zend_Session_Namespace('Zend_Auth', true);
  260. // 参照をすることは可能です
  261. $authSpaceAccessor3 = $authSpaceAccessor2;
  262. $authSpaceAccessor1->foo = 'bar';
  263. assert($authSpaceAccessor2->foo, 'bar');
  264. try {
  265. $aNamespaceObject = new Zend_Session_Namespace('Zend_Auth');
  266. } catch (Zend_Session_Exception $e) {
  267. echo 'この名前空間ではインスタンスを作成できません。すでに ' .
  268. '$authSpaceAccessor2 があるからです\n';
  269. }
  270. ]]>
  271. </programlisting>
  272. </example>
  273. <para>
  274. 上の例では、コンストラクタの第二引数を用いて
  275. "<classname>Zend_Auth</classname>" 名前空間では今後インスタンスを作成させないよう
  276. <classname>Zend_Session_Namespace</classname> に指示しています。
  277. インスタンスを作成しようとすると、コンストラクタから例外がスローされます。
  278. したがって、このセッション名前空間へのアクセスが必要となった場合は、
  279. 今後は現在あるインスタンス (上の例の場合なら <code>$authSpaceAccessor1</code>、
  280. <code>$authSpaceAccessor2</code> あるいは <code>$authSpaceAccessor3</code>)
  281. のどれかを使うことになるわけです。
  282. たとえば、名前空間への参照を静的変数に格納したり、
  283. <ulink url="http://www.martinfowler.com/eaaCatalog/registry.html">レジストリ</ulink>
  284. (<xref linkend="zend.registry" /> を参照ください) に格納したり、
  285. あるいは名前空間へのアクセスを必要とするその他のメソッドで使用したりします。
  286. </para>
  287. </sect2>
  288. <sect2 id="zend.session.advanced_usage.arrays">
  289. <title>配列の使用</title>
  290. <para>
  291. PHP のマジックメソッドの実装上の理由で、バージョン 5.2.1 より前の PHP
  292. では名前空間内の配列の修正ができません。
  293. もし PHP 5.2.1 以降を使っている場合は、<link
  294. linkend="zend.session.advanced_usage.objects">このセクションは読み飛ばしてください</link>。
  295. </para>
  296. <example id="zend.session.advanced_usage.arrays.example.modifying">
  297. <title>セッション名前空間内での配列データの修正</title>
  298. <para>
  299. 問題の再現手順は、このようになります。
  300. </para>
  301. <programlisting role="php"><![CDATA[
  302. $sessionNamespace = new Zend_Session_Namespace();
  303. $sessionNamespace->array = array();
  304. // PHP 5.2.1 より前のバージョンでは、期待通りに動作しません
  305. $sessionNamespace->array['testKey'] = 1;
  306. echo $sessionNamespace->array['testKey'];
  307. ]]>
  308. </programlisting>
  309. </example>
  310. <example id="zend.session.advanced_usage.arrays.example.building_prior">
  311. <title>セッションに保存する前に配列を作成する</title>
  312. <para>
  313. 可能なら、先に配列のすべての値を設定してからセッションに格納するようにすればこの問題を回避できます。
  314. </para>
  315. <programlisting role="php"><![CDATA[
  316. $sessionNamespace = new Zend_Session_Namespace('Foo');
  317. $sessionNamespace->array = array('a', 'b', 'c');
  318. ]]>
  319. </programlisting>
  320. </example>
  321. <para>
  322. この問題の影響を受けるバージョンの PHP を使っている場合で、
  323. セッション名前空間に代入した後に配列を修正したい場合は、
  324. 以下の回避策のうちのいずれかを使用します。
  325. </para>
  326. <example id="zend.session.advanced_usage.arrays.example.workaround.reassign">
  327. <title>回避策: 修正した配列を再度代入する</title>
  328. <para>
  329. 以下のコードでは、保存されている配列のコピーを作成してそれを修正し、
  330. 修正したコピーを再度代入してもとの配列を上書きします。
  331. </para>
  332. <programlisting role="php"><![CDATA[
  333. $sessionNamespace = new Zend_Session_Namespace();
  334. // 配列を代入します
  335. $sessionNamespace->array = array('tree' => 'apple');
  336. // そのコピーを作成します
  337. $tmp = $sessionNamespace->array;
  338. // コピーのほうを修正します
  339. $tmp['fruit'] = 'peach';
  340. // 修正したコピーをセッション名前空間に書き戻します
  341. $sessionNamespace->array = $tmp;
  342. echo $sessionNamespace->array['fruit']; // prints "peach"
  343. ]]>
  344. </programlisting>
  345. </example>
  346. <example id="zend.session.advanced_usage.arrays.example.workaround.reference">
  347. <title>回避策: 参照を含む配列を格納する</title>
  348. <para>
  349. あるいは、実際の配列への参照を含む配列を格納しておき、
  350. 間接的にアクセスするようにします。
  351. </para>
  352. <programlisting role="php"><![CDATA[
  353. $myNamespace = new Zend_Session_Namespace('myNamespace');
  354. $a = array(1, 2, 3);
  355. $myNamespace->someArray = array( &$a );
  356. $a['foo'] = 'bar';
  357. echo $myNamespace->someArray['foo']; // "bar" と表示されます
  358. ]]>
  359. </programlisting>
  360. </example>
  361. </sect2>
  362. <sect2 id="zend.session.advanced_usage.objects">
  363. <title>セッションでのオブジェクトの使用</title>
  364. <para>
  365. オブジェクトを PHP セッション内で持続的に使用したい場合は、
  366. <ulink url="http://www.php.net/manual/ja/language.oop.serialization.php">シリアライズ</ulink>
  367. を使用します。したがって、PHP セッションから永続オブジェクトを取得したら、
  368. そのシリアライズを解除しなければなりません。
  369. ということは、永続オブジェクトをセッションから読み出す前に、
  370. そのオブジェクトのクラスが定義されていなければならないということです。
  371. クラスが定義されていない場合は、<code>stdClass</code>
  372. のオブジェクトとして復元されます。
  373. </para>
  374. </sect2>
  375. <sect2 id="zend.session.advanced_usage.testing">
  376. <title>ユニットテストでのセッションの使用</title>
  377. <para>
  378. Zend Framework 自体のテストには PHPUnit を使用しています。
  379. 多くの開発者は、このテストスイートを拡張して自分のアプリケーションのコードをテストしています。
  380. ユニットテスト中で、セッションの終了後に書き込み関連のメソッドを使用すると
  381. "<emphasis role="strong">Zend_Session is currently marked as read-only</emphasis>"
  382. という例外がスローされます。しかし、Zend_Session を使用するユニットテストには要注意です。
  383. セッションを閉じたり (<classname>Zend_Session::writeClose()</classname>)
  384. 破棄したり (<classname>Zend_Session::destroy()</classname>) したら、
  385. それ以降は <classname>Zend_Session_Namespace</classname> のインスタンスへのキーの設定や削除ができなくなります。
  386. これは、ext/session や、PHP の
  387. PHP <code>session_destroy()</code> および <code>session_write_close()</code>
  388. の仕様によるものです, これらには、ユニットテストの setup/teardown
  389. 時に使用できるような、いわゆる "undo" 機能が備わっていないのです。
  390. </para>
  391. <para>
  392. この問題の回避策は、
  393. <code>SessionTest.php</code> および <code>SessionTestHelper.php</code>
  394. (どちらも <code>tests/Zend/Session</code> にあります)
  395. のユニットテストテスト <code>testSetExpirationSeconds()</code> を参照ください。
  396. これは、PHP の <code>exec()</code> によって別プロセスを起動しています。
  397. 新しいプロセスが、ブラウザからの二番目以降のリクエストをシミュレートします。
  398. この別プロセスの開始時にはセッションを "初期化" します。
  399. ちょうど、ふつうの PHP スクリプトがウェブリクエストを実行する場合と同じような動作です。
  400. また、呼び出し元のプロセスで <code>$_SESSION</code> を変更すると、
  401. 子プロセスでそれが反映されます。親側では
  402. <code>exec()</code> を使用する前にセッションを閉じています。
  403. </para>
  404. <example id="zend.session.advanced_usage.testing.example">
  405. <title>PHPUnit で Zend_Session を使用したコードをテストする例</title>
  406. <programlisting role="php"><![CDATA[
  407. // testing setExpirationSeconds()
  408. $script = 'SessionTestHelper.php';
  409. $s = new Zend_Session_Namespace('space');
  410. $s->a = 'apple';
  411. $s->o = 'orange';
  412. $s->setExpirationSeconds(5);
  413. Zend_Session::regenerateId();
  414. $id = Zend_Session::getId();
  415. session_write_close(); // release session so process below can use it
  416. sleep(4); // not long enough for things to expire
  417. exec($script . "expireAll $id expireAll", $result);
  418. $result = $this->sortResult($result);
  419. $expect = ';a === apple;o === orange;p === pear';
  420. $this->assertTrue($result === $expect,
  421. "iteration over default Zend_Session namespace failed; " .
  422. "expecting result === '$expect', but got '$result'");
  423. sleep(2); // long enough for things to expire (total of 6 seconds
  424. // waiting, but expires in 5)
  425. exec($script . "expireAll $id expireAll", $result);
  426. $result = array_pop($result);
  427. $this->assertTrue($result === '',
  428. "iteration over default Zend_Session namespace failed; " .
  429. "expecting result === '', but got '$result')");
  430. session_start(); // resume artificially suspended session
  431. // We could split this into a separate test, but actually, if anything
  432. // leftover from above contaminates the tests below, that is also a
  433. // bug that we want to know about.
  434. $s = new Zend_Session_Namespace('expireGuava');
  435. $s->setExpirationSeconds(5, 'g'); // now try to expire only 1 of the
  436. // keys in the namespace
  437. $s->g = 'guava';
  438. $s->p = 'peach';
  439. $s->p = 'plum';
  440. session_write_close(); // release session so process below can use it
  441. sleep(6); // not long enough for things to expire
  442. exec($script . "expireAll $id expireGuava", $result);
  443. $result = $this->sortResult($result);
  444. session_start(); // resume artificially suspended session
  445. $this->assertTrue($result === ';p === plum',
  446. "iteration over named Zend_Session namespace failed (result=$result)");
  447. ]]>
  448. </programlisting>
  449. </example>
  450. </sect2>
  451. </sect1>