Zend_Auth.xml 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!-- Reviewed: no -->
  3. <!-- EN-Revision: 15156 -->
  4. <sect1 id="zend.auth.introduction">
  5. <title>導入</title>
  6. <para>
  7. <classname>Zend_Auth</classname> は、認証のための API を提供します。
  8. また、一般的な使用例に対応する具象認証アダプタも用意しています。
  9. </para>
  10. <para>
  11. <classname>Zend_Auth</classname> が扱うのはあくまでも <emphasis role="strong">認証 (authentication)</emphasis>
  12. であり、<emphasis role="strong">承認 (authorization)</emphasis> ではありません。
  13. 認証 (authentication) とはつまり、あるエンティティが何者であるのかを示す
  14. (識別する) ことです。これを、なんらかの条件にもとづいて行います。
  15. 承認 (authorization) とは、あるエンティティが他のエンティティに対して
  16. アクセスしたり何らかの操作をしたりする権限があるかどうかを判定する処理です。
  17. これは <classname>Zend_Auth</classname> の対象外となります。
  18. Zend Framework における認証やアクセス制御の詳細については、
  19. <link linkend="zend.acl"><classname>Zend_Acl</classname></link> を参照ください。
  20. </para>
  21. <note>
  22. <para>
  23. <classname>Zend_Auth</classname> クラスはシングルトンパターン
  24. (クラスのインスタンスは常にひとつだけ)
  25. を実装しており、静的メソッド <code>getInstance()</code> でそれを使用します。
  26. つまり、<code>new</code> 演算子や
  27. <code>clone</code> キーワードは <classname>Zend_Auth</classname>
  28. クラスでは動作しないということです。そのかわりに
  29. <classname>Zend_Auth::getInstance()</classname> を使用します。
  30. </para>
  31. </note>
  32. <sect2 id="zend.auth.introduction.adapters">
  33. <title>アダプタ</title>
  34. <para>
  35. <classname>Zend_Auth</classname> アダプタの使用目的は、
  36. LDAP や RDBMS あるいはファイル
  37. のような特定の型の認証サービスに対する認証を行うことです。
  38. アダプタによってそのオプションや挙動は大きくことなるでしょうが、
  39. いくつかの基本処理は、あらゆる認証アダプタで共通となります。
  40. たとえば認証条件 (いわゆる ID) を受け取り、
  41. 認証サービスに対する問い合わせを行い、
  42. 結果を返すという処理は、すべての <classname>Zend_Auth</classname> アダプタで共通です。
  43. </para>
  44. <para>
  45. 各 <classname>Zend_Auth</classname> アダプタクラスは、<classname>Zend_Auth_Adapter_Interface</classname>
  46. を実装しています。このインターフェイスで定義されているメソッドが
  47. <code>authenticate()</code> で、アダプタクラスは
  48. 認証クエリを実行するためにこれを実装する必要があります。
  49. 各アダプタクラスは、<code>authenticate()</code>
  50. をコールする前に準備を済ませておく必要があります。
  51. つまり、アダプタ側で用意しなければならない機能としては
  52. 認証条件 (ユーザ名およびパスワードなど) の取得や
  53. アダプタ固有のオプションの設定
  54. (データベースのテーブルを使用するアダプタならデータベースへの接続設定など)
  55. があるということです。
  56. </para>
  57. <para>
  58. 以下にあげるのは認証アダプタのサンプルで、
  59. これはユーザ名とパスワードを受け取って認証を行います。
  60. その他の詳細、例えば認証サービスへの実際の問い合わせなどは、
  61. 例を簡潔にするため省略しています。
  62. <programlisting role="php"><![CDATA[
  63. class MyAuthAdapter implements Zend_Auth_Adapter_Interface
  64. {
  65. /**
  66. * 認証用のユーザ名とパスワードを設定します
  67. *
  68. * @return void
  69. */
  70. public function __construct($username, $password)
  71. {
  72. // ...
  73. }
  74. /**
  75. * 認証を試みます
  76. *
  77. * @throws Zend_Auth_Adapter_Exception が、認証処理をできなかった場合に発生します
  78. * @return Zend_Auth_Result
  79. */
  80. public function authenticate()
  81. {
  82. // ...
  83. }
  84. }
  85. ]]></programlisting>
  86. docblock コメントで説明しているとおり、
  87. <code>authenticate()</code> は
  88. <classname>Zend_Auth_Result</classname> (あるいは <classname>Zend_Auth_Result</classname> の派生クラス)
  89. のインスタンスを返す必要があります。何らかの理由で認証の問い合わせができなかった場合は、
  90. <code>authenticate()</code> は
  91. <classname>Zend_Auth_Adapter_Exception</classname>
  92. から派生した例外をスローしなければなりません。
  93. </para>
  94. </sect2>
  95. <sect2 id="zend.auth.introduction.results">
  96. <title>結果</title>
  97. <para>
  98. <classname>Zend_Auth</classname> アダプタは、<classname>authenticate()</classname> の結果として
  99. <classname>Zend_Auth_Result</classname> のインスタンスを返します。
  100. これにより、認証を試みた結果を表します。アダプタのインスタンスを作成した際に
  101. <classname>Zend_Auth_Result</classname> オブジェクトが作成され、
  102. 以下の 4 つのメソッドで <classname>Zend_Auth</classname> アダプタの結果に対する共通の操作ができます。
  103. <itemizedlist>
  104. <listitem>
  105. <para>
  106. <code>isValid()</code> - その結果が
  107. 認証の成功を表している場合にのみ true を返します。
  108. </para>
  109. </listitem>
  110. <listitem>
  111. <para>
  112. <code>getCode()</code> - <classname>Zend_Auth_Result</classname> の定数を返します。
  113. これは、認証に失敗したのか成功したのかを表すものです。
  114. これを使用する場面としては、認証の結果をいくつかの結果から区別したい場合などがあります。
  115. これにより、たとえば認証結果について、より詳細な情報を管理することができるようになります。
  116. 別の使用法としては、ユーザに示すメッセージを変更し、より詳細な情報を伝えられるようにすることなどがあります。
  117. しかし、一般的な「認証失敗」メッセージではなく
  118. ユーザに対して詳細な情報を示す際には、そのリスクを知っておくべきです。
  119. 詳細な情報は、以下の注意を参照ください。
  120. </para>
  121. </listitem>
  122. <listitem>
  123. <para>
  124. <code>getIdentity()</code> - 認証を試みた ID 情報を返します。
  125. </para>
  126. </listitem>
  127. <listitem>
  128. <para>
  129. <code>getMessages()</code> - 認証に失敗した場合に、
  130. 関連するメッセージの配列を返します。
  131. </para>
  132. </listitem>
  133. </itemizedlist>
  134. </para>
  135. <para>
  136. 認証の結果によって処理を分岐させ、より決め細やかな処理を行いたいこともあるでしょう。
  137. 有用な処理としては、たとえば間違ったパスワードを繰り返し入力したアカウントをロックしたり、
  138. 存在しない ID を何度も入力した IP アドレスに印をつけたり、
  139. ユーザに対してよりわかりやすいメッセージを返したりといったことがあります。
  140. 次の結果コードが使用可能です。
  141. <programlisting role="php"><![CDATA[
  142. Zend_Auth_Result::SUCCESS
  143. Zend_Auth_Result::FAILURE
  144. Zend_Auth_Result::FAILURE_IDENTITY_NOT_FOUND
  145. Zend_Auth_Result::FAILURE_IDENTITY_AMBIGUOUS
  146. Zend_Auth_Result::FAILURE_CREDENTIAL_INVALID
  147. Zend_Auth_Result::FAILURE_UNCATEGORIZED
  148. ]]></programlisting>
  149. </para>
  150. <para>
  151. 次の例は、結果コードを処理する方法を示すものです。
  152. <programlisting role="php"><![CDATA[
  153. // AuthController / loginAction の中の処理
  154. $result = $this->_auth->authenticate($adapter);
  155. switch ($result->getCode()) {
  156. case Zend_Auth_Result::FAILURE_IDENTITY_NOT_FOUND:
  157. /** ID が存在しない場合の処理 **/
  158. break;
  159. case Zend_Auth_Result::FAILURE_CREDENTIAL_INVALID:
  160. /** 認証に失敗した場合の処理 **/
  161. break;
  162. case Zend_Auth_Result::SUCCESS:
  163. /** 認証に成功した場合の処理 **/
  164. break;
  165. default:
  166. /** その他の原因で失敗した場合の処理 **/
  167. break;
  168. }
  169. ]]></programlisting>
  170. </para>
  171. </sect2>
  172. <sect2 id="zend.auth.introduction.persistence">
  173. <title>ID の永続性</title>
  174. <para>
  175. 認証情報 (パスワードなど) を含む認証を要求するのは便利なものですが、
  176. リクエストごとにいちいち認証情報を引き回すのではなく、
  177. 認証済みの ID を保持し続けることも重要です。
  178. </para>
  179. <para>
  180. HTTP はステートレスなプロトコルです。しかし、
  181. クッキーやセッションといった技術によって、
  182. サーバサイドのウェブアプリケーションでも
  183. 複数リクエスト間でステート (状態) を保持し続けられるようになりました。
  184. </para>
  185. <sect3 id="zend.auth.introduction.persistence.default">
  186. <title>PHP セッションにおけるデフォルトの持続性</title>
  187. <para>
  188. デフォルトでは、<classname>Zend_Auth</classname> は、
  189. 認証に成功した際の ID 情報を PHP のセッションを使用して保存します。
  190. 認証に成功すると、<classname>Zend_Auth::authenticate()</classname>
  191. は認証結果を持続ストレージに保存します。何も指定しなければ、
  192. <classname>Zend_Auth</classname> が使用するストレージクラスは
  193. <classname>Zend_Auth_Storage_Session</classname> となります。これは
  194. <link linkend="zend.session"><classname>Zend_Session</classname></link> を使用しています。
  195. 独自のクラスを使用するには、<classname>Zend_Auth_Storage_Interface</classname>
  196. を実装したクラスのオブジェクトを <classname>Zend_Auth::setStorage()</classname>
  197. で設定します。
  198. </para>
  199. <note>
  200. <para>
  201. もし ID が自動的に持続ストレージに保存されるのが気に入らない場合は、
  202. <classname>Zend_Auth</classname> クラスをまるごと使用するのを控え、
  203. アダプタクラスを直接使用します。
  204. </para>
  205. </note>
  206. <example id="zend.auth.introduction.persistence.default.example">
  207. <title>セッション名前空間の変更</title>
  208. <para>
  209. <classname>Zend_Auth_Storage_Session</classname> は、セッション名前空間として
  210. 'Zend_Auth' を使用します。これを変更するには、別の値を
  211. <classname>Zend_Auth_Storage_Session</classname> のコンストラクタで指定します。
  212. この値が、内部で <classname>Zend_Session_Namespace</classname>
  213. のコンストラクタに渡されます。これは認証を試みる前に行う必要があります。
  214. なぜなら、<classname>Zend_Auth::authenticate()</classname>
  215. は ID を自動的に保存するからです。
  216. <programlisting role="php"><![CDATA[
  217. // Zend_Auth のシングルトンインスタンスへの参照を保存します
  218. $auth = Zend_Auth::getInstance();
  219. // 'Zend_Auth' のかわりに 'someNamespace' を使用します
  220. $auth->setStorage(new Zend_Auth_Storage_Session('someNamespace'));
  221. /**
  222. * @todo 認証アダプタ $authAdapter を設定します
  223. */
  224. // 認証と結果の保存、そして成功時に ID を持続させます
  225. $result = $auth->authenticate($authAdapter);
  226. ]]></programlisting>
  227. </para>
  228. </example>
  229. </sect3>
  230. <sect3 id="zend.auth.introduction.persistence.custom">
  231. <title>独自のストレージの実装</title>
  232. <para>
  233. <classname>Zend_Auth_Storage_Session</classname> とは異なる形式で、
  234. ID を持続させたくなることもあるでしょう。そのような場合は、
  235. <classname>Zend_Auth_Storage_Interface</classname> を実装したクラスのインスタンスを
  236. <classname>Zend_Auth::setStorage()</classname> で設定します。
  237. </para>
  238. <example id="zend.auth.introduction.persistence.custom.example">
  239. <title>独自のストレージクラスの使用法</title>
  240. <para>
  241. ID を持続させるストレージクラスを
  242. <classname>Zend_Auth_Storage_Session</classname> の代わりに使用するには、
  243. <classname>Zend_Auth_Storage_Interface</classname> を実装します。
  244. <programlisting role="php"><![CDATA[
  245. class MyStorage implements Zend_Auth_Storage_Interface
  246. {
  247. /**
  248. * ストレージが空の場合にのみ true を返す
  249. *
  250. * @throws Zend_Auth_Storage_Exception 空かどうか判断できないとき
  251. * @return boolean
  252. */
  253. public function isEmpty()
  254. {
  255. /**
  256. * @todo 実装が必要
  257. */
  258. }
  259. /**
  260. * ストレージの中身を返す
  261. *
  262. * ストレージが空の場合に挙動は未定義
  263. *
  264. * @throws Zend_Auth_Storage_Exception ストレージの中身を読み込めない場合
  265. * @return mixed
  266. */
  267. public function read()
  268. {
  269. /**
  270. * @todo 実装が必要
  271. */
  272. }
  273. /**
  274. * $contents をストレージに書き込む
  275. *
  276. * @param mixed $contents
  277. * @throws Zend_Auth_Storage_Exception $contents をストレージに書き込めない場合
  278. * @return void
  279. */
  280. public function write($contents)
  281. {
  282. /**
  283. * @todo 実装が必要
  284. */
  285. }
  286. /**
  287. * ストレージの中身を消去する
  288. *
  289. * @throws Zend_Auth_Storage_Exception ストレージの中身を消去できない場合
  290. * @return void
  291. */
  292. public function clear()
  293. {
  294. /**
  295. * @todo 実装が必要
  296. */
  297. }
  298. }
  299. ]]></programlisting>
  300. </para>
  301. <para>
  302. このストレージクラスを使用するには、認証クエリの前に
  303. <classname>Zend_Auth::setStorage()</classname> を実行します。
  304. <programlisting role="php"><![CDATA[
  305. // Zend_Auth に、独自のストレージクラスを使うよう指示します
  306. Zend_Auth::getInstance()->setStorage(new MyStorage());
  307. /**
  308. * @todo 認証アダプタ $authAdapter を設定します
  309. */
  310. // 認証と結果の保存、そして成功時に ID を持続させます
  311. $result = Zend_Auth::getInstance()->authenticate($authAdapter);
  312. ]]></programlisting>
  313. </para>
  314. </example>
  315. </sect3>
  316. </sect2>
  317. <sect2 id="zend.auth.introduction.using">
  318. <title>使用法</title>
  319. <para>
  320. <classname>Zend_Auth</classname> の使用法には、次の二通りがあります。
  321. <orderedlist>
  322. <listitem>
  323. <para>
  324. 間接的に <classname>Zend_Auth::authenticate()</classname> 経由で使用する
  325. </para>
  326. </listitem>
  327. <listitem>
  328. <para>
  329. 直接、アダプタの <code>authenticate()</code> メソッドを使用する
  330. </para>
  331. </listitem>
  332. </orderedlist>
  333. </para>
  334. <para>
  335. 次の例は、<classname>Zend_Auth</classname> アダプタを間接的に
  336. <classname>Zend_Auth</classname> クラスから使用するものです。
  337. <programlisting role="php"><![CDATA[
  338. // Zend_Auth のシングルトンインスタンスへの参照を取得します
  339. $auth = Zend_Auth::getInstance();
  340. // 認証アダプタを設定します
  341. $authAdapter = new MyAuthAdapter($username, $password);
  342. // 認証を試み、その結果を取得します
  343. $result = $auth->authenticate($authAdapter);
  344. if (!$result->isValid()) {
  345. // 認証に失敗したので、原因を表示します
  346. foreach ($result->getMessages() as $message) {
  347. echo "$message\n";
  348. }
  349. } else {
  350. // 認証に成功しました。ID ($username) がセッションに保存されます
  351. // $result->getIdentity() === $auth->getIdentity()
  352. // $result->getIdentity() === $username
  353. }
  354. ]]></programlisting>
  355. </para>
  356. <para>
  357. 上の例においてリクエスト内で認証が行われると、
  358. 認証に成功した際にその ID を取得するのは簡単なことです。
  359. <programlisting role="php"><![CDATA[
  360. $auth = Zend_Auth::getInstance();
  361. if ($auth->hasIdentity()) {
  362. // ID があるのでそれを取得します
  363. $identity = $auth->getIdentity();
  364. }
  365. ]]></programlisting>
  366. </para>
  367. <para>
  368. 永続ストレージから認証 ID を削除するには、単純に
  369. <code>clearIdentity()</code> メソッドを使用します。
  370. これは、アプリケーションの "ログアウト" 処理を実装するためのものです。
  371. <programlisting role="php"><![CDATA[
  372. Zend_Auth::getInstance()->clearIdentity();
  373. ]]></programlisting>
  374. </para>
  375. <para>
  376. 自動的に永続ストレージが用いられるのがまずい場合もあるでしょう。
  377. そんな場合は、<classname>Zend_Auth</classname> クラスをバイパスして
  378. アダプタクラスを直接使用します。
  379. アダプタクラスを直接使用するとは、アダプタオブジェクトの設定と準備を行い、
  380. その <code>authenticate()</code> メソッドをコールするということです。
  381. アダプタ固有の詳細情報については、各アダプタのドキュメントで説明します。
  382. 以下の例は、<code>MyAuthAdapter</code> を直接使用するものです。
  383. <programlisting role="php"><![CDATA[
  384. // 認証アダプタを設定します
  385. $authAdapter = new MyAuthAdapter($username, $password);
  386. // 認証を試み、その結果を取得します
  387. $result = $authAdapter->authenticate();
  388. if (!$result->isValid()) {
  389. // 認証に失敗したので、原因を表示します
  390. foreach ($result->getMessages() as $message) {
  391. echo "$message\n";
  392. }
  393. } else {
  394. // 認証に成功しました
  395. // $result->getIdentity() === $username
  396. }
  397. ]]></programlisting>
  398. </para>
  399. </sect2>
  400. </sect1>
  401. <!--
  402. vim:se ts=4 sw=4 et:
  403. -->