Zend_Auth_Adapter_DbTable.xml 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!-- Reviewed: no -->
  3. <!-- EN-Revision: 24249 -->
  4. <sect1 id="zend.auth.adapter.dbtable">
  5. <title>データベースのテーブルでの認証</title>
  6. <sect2 id="zend.auth.adapter.dbtable.introduction">
  7. <title>導入</title>
  8. <para>
  9. <classname>Zend_Auth_Adapter_DbTable</classname> は、
  10. データベースのテーブルに保存された証明情報に基づいた認証の機能を提供します。
  11. <classname>Zend_Auth_Adapter_DbTable</classname> のコンストラクタには
  12. <classname>Zend_Db_Adapter_Abstract</classname> のインスタンスを渡す必要があるので、
  13. 各インスタンスは特定のデータベース接続に関連付けられます。
  14. コンストラクタではその他の設定オプションも指定できます。
  15. これらは個別にインスタンスメソッドで指定することもできます。
  16. </para>
  17. <para>
  18. 次のような設定オプションが使用可能です。
  19. </para>
  20. <itemizedlist>
  21. <listitem>
  22. <para>
  23. <emphasis><property>tableName</property></emphasis>:
  24. これはデータベースのテーブル名です。証明情報が含まれ、
  25. 認証クエリの問い合わせ先となるテーブル名を指定します。
  26. </para>
  27. </listitem>
  28. <listitem>
  29. <para>
  30. <emphasis><property>identityColumn</property></emphasis>:
  31. これは、ID を表すデータベーステーブルのカラム名です。
  32. このカラムには、ユーザ名やメールアドレスのような一意な値が含まれている必要があります。
  33. </para>
  34. </listitem>
  35. <listitem>
  36. <para>
  37. <emphasis><property>credentialColumn</property></emphasis>:
  38. これは、証明情報を表すデータベーステーブルのカラム名です。
  39. 単純な ID およびパスワードによる認証方式では、この値がパスワードに対応します。
  40. <property>credentialTreatment</property> オプションも参照ください。
  41. </para>
  42. </listitem>
  43. <listitem>
  44. <para>
  45. <emphasis><property>credentialTreatment</property></emphasis>:
  46. 多くの場合、パスワードやその他機密情報は、何らかの関数やアルゴリズムで
  47. 暗号化、ハッシュ化、符号化、ぼかしなどの処理が行われています。
  48. これらの処理を表すパラメータつきの文字列、たとえば
  49. '<methodname>MD5(?)</methodname>' や
  50. '<methodname>PASSWORD(?)</methodname>'
  51. を指定することで、
  52. 証明データに対して任意の <acronym>SQL</acronym> を適用できるようになります。
  53. これらの関数は <acronym>RDBMS</acronym> によって異なります。
  54. データベースシステムでどのような関数が使えるのかについては、
  55. データベースのマニュアルをご確認ください。
  56. </para>
  57. </listitem>
  58. </itemizedlist>
  59. <example id="zend.auth.adapter.dbtable.introduction.example.basic_usage">
  60. <title>基本的な使用法</title>
  61. <para>
  62. 導入部で説明したとおり、<classname>Zend_Auth_Adapter_DbTable</classname>
  63. のコンストラクタには <classname>Zend_Db_Adapter_Abstract</classname>
  64. のインスタンスを渡す必要があります。これは、認証アダプタのインスタンスと
  65. 関連付けるデータベース接続を表します。
  66. まず、データベース接続を作成する必要があります。
  67. </para>
  68. <para>
  69. 次のコードは、メモリ内データベースのアダプタを作成し、
  70. 簡単なテーブルスキーマを作成し、そして後で認証クエリを実行するための行を追加します。
  71. この例を実行するには、<acronym>PDO</acronym> SQLite 拡張モジュールが有効になっている必要があります。
  72. </para>
  73. <programlisting language="php"><![CDATA[
  74. // メモリ内で SQLite データベース接続を作成します
  75. $dbAdapter = new Zend_Db_Adapter_Pdo_Sqlite(array('dbname' =>
  76. ':memory:'));
  77. // 単純なテーブルを作成するクエリ
  78. $sqlCreate = 'CREATE TABLE [users] ('
  79. . '[id] INTEGER NOT NULL PRIMARY KEY, '
  80. . '[username] VARCHAR(50) UNIQUE NOT NULL, '
  81. . '[password] VARCHAR(32) NULL, '
  82. . '[real_name] VARCHAR(150) NULL)';
  83. // 認証情報テーブルを作成します
  84. $dbAdapter->query($sqlCreate);
  85. // 認証を成功させるために行を追加します
  86. $sqlInsert = "INSERT INTO users (username, password, real_name) "
  87. . "VALUES ('my_username', 'my_password', 'My Real Name')";
  88. // データを挿入します
  89. $dbAdapter->query($sqlInsert);]]></programlisting>
  90. <para>
  91. データベース接続およびテーブルが使用可能となったので
  92. <classname>Zend_Auth_Adapter_DbTable</classname> のインスタンスが作成できます。
  93. 設定オプションの値は、コンストラクタで渡すか、
  94. あるいはインスタンスを作成した後に設定用メソッドで指定します。
  95. </para>
  96. <programlisting language="php"><![CDATA[
  97. // コンストラクタにパラメータを渡し、インスタンスを設定します
  98. $authAdapter = new Zend_Auth_Adapter_DbTable(
  99. $dbAdapter,
  100. 'users',
  101. 'username',
  102. 'password'
  103. );
  104. // あるいは、設定用メソッドでインスタンスの設定を行います
  105. $authAdapter = new Zend_Auth_Adapter_DbTable($dbAdapter);
  106. $authAdapter
  107. ->setTableName('users')
  108. ->setIdentityColumn('username')
  109. ->setCredentialColumn('password')
  110. ;]]></programlisting>
  111. <para>
  112. この時点で、認証アダプタのインスタンスは認証クエリを受け付ける準備ができました。
  113. 認証クエリを処理するには、入力された証明情報をアダプタに渡してから
  114. <methodname>authenticate()</methodname> メソッドをコールします。
  115. </para>
  116. <programlisting language="php"><![CDATA[
  117. // 入力情報 (ログインフォームからの値など) を設定します
  118. $authAdapter
  119. ->setIdentity('my_username')
  120. ->setCredential('my_password')
  121. ;
  122. // 認証クエリを実行し、結果を保存します
  123. ]]></programlisting>
  124. <para>
  125. 認証結果オブジェクトでの <methodname>getIdentity()</methodname>
  126. メソッドに加え、<classname>Zend_Auth_Adapter_DbTable</classname>
  127. は認証の成功時にテーブルの行を取得する機能もサポートしています。
  128. </para>
  129. <programlisting language="php"><![CDATA[
  130. // ID を表示します
  131. echo $result->getIdentity() . "\n\n";
  132. // 結果の行を表示します
  133. print_r($authAdapter->getResultRowObject());
  134. /* 出力結果
  135. my_username
  136. Array
  137. (
  138. [id] => 1
  139. [username] => my_username
  140. [password] => my_password
  141. [real_name] => My Real Name
  142. )
  143. */]]></programlisting>
  144. <para>
  145. テーブルの行には証明情報が含まれているので、
  146. 予期せぬアクセスからその内容を守ることが重要となります。
  147. </para>
  148. </example>
  149. </sect2>
  150. <sect2 id="zend.auth.adapter.dbtable.advanced.storing_result_row">
  151. <title>応用例: 持続的な DbTable 結果オブジェクト</title>
  152. <para>
  153. デフォルトでは <classname>Zend_Auth_Adapter_DbTable</classname> は、
  154. 認証に成功した際に認証情報を返します。場合によっては、
  155. <classname>Zend_Auth</classname> の持続ストレージの仕組みを利用して
  156. 別の有用な情報を格納したいこともあるでしょう。その場合は、
  157. <methodname>getResultRowObject()</methodname> メソッドを使用して
  158. <emphasis>stdClass</emphasis> オブジェクトを返します。
  159. 次のコードで、使用法をご確認ください。
  160. </para>
  161. <programlisting language="php"><![CDATA[
  162. // Zend_Auth_Adapter_DbTable による認証を行います
  163. $result = $this->_auth->authenticate($adapter);
  164. if ($result->isValid()) {
  165. // 認証情報をオブジェクトとして保存し、username と real_name のみを返します
  166. $storage = $this->_auth->getStorage();
  167. $storage->write($adapter->getResultRowObject(array(
  168. 'username',
  169. 'real_name',
  170. )));
  171. // 認証情報をオブジェクトとして保存し、password のみを省略します
  172. $storage->write($adapter->getResultRowObject(
  173. null,
  174. 'password'
  175. ));
  176. /* ... */
  177. } else {
  178. /* ... */
  179. }]]></programlisting>
  180. </sect2>
  181. <sect2 id="zend.auth.adapter.dbtable.advanced.advanced_usage">
  182. <title>高度な使用例</title>
  183. <para>
  184. <classname>Zend_Auth</classname> (そして <classname>Zend_Auth_Adapter_DbTable</classname>) の主な目的は
  185. <emphasis>認証 (authentication)</emphasis> であって
  186. <emphasis>認可 (authorization)</emphasis>,
  187. ではありませんが、認可にもかかわる問題も多少あります。
  188. 問題によっては、認証アダプタの中で認可にかかわる問題を解決することもあるでしょう。
  189. </para>
  190. <para>
  191. ちょっとしたおまけとして <classname>Zend_Auth_Adapter_DbTable</classname>
  192. に組み込まれている仕組みを使用すると、
  193. 認証時にありがちな問題を解決するチェックを加えることができます。
  194. </para>
  195. <programlisting language="php"><![CDATA[
  196. // アカウントの status フィールドが "compromised" ではない
  197. $adapter = new Zend_Auth_Adapter_DbTable(
  198. $db,
  199. 'users',
  200. 'username',
  201. 'password',
  202. 'MD5(?) AND status != "compromised"'
  203. );
  204. // アカウントの active フィールドが "TRUE" に等しい
  205. $adapter = new Zend_Auth_Adapter_DbTable(
  206. $db,
  207. 'users',
  208. 'username',
  209. 'password',
  210. 'MD5(?) AND active = "TRUE"'
  211. );]]></programlisting>
  212. <para>
  213. もうひとつの例として、salt メカニズムの実装を見てみましょう。
  214. salt とは、アプリケーションのセキュリティを格段に向上させるテクニックを指す用語です。
  215. パスワードにランダムな文字列を連結することで、
  216. 辞書を用いた総当たり攻撃からパスワードを保護するという仕組みになっています。
  217. </para>
  218. <para>
  219. salt 文字列を格納するために、テーブルの構造を変更する必要があります。
  220. </para>
  221. <programlisting language="php"><![CDATA[
  222. $sqlAlter = "ALTER TABLE [users] "
  223. . "ADD COLUMN [password_salt] "
  224. . "AFTER [password]";
  225. $dbAdapter->query($sqlAlter);]]></programlisting>
  226. <para>
  227. すべてのユーザに対して登録時に salt 文字列を生成するシンプルな方法を示します。
  228. </para>
  229. <programlisting language="php"><![CDATA[
  230. for ($i = 0; $i < 50; $i++) {
  231. $dynamicSalt .= chr(rand(33, 126));
  232. }]]></programlisting>
  233. <para>
  234. それではアダプタを作成してみましょう。
  235. </para>
  236. <programlisting language="php"><![CDATA[
  237. $adapter = new Zend_Auth_Adapter_DbTable(
  238. $db,
  239. 'users',
  240. 'username',
  241. 'password',
  242. "MD5(CONCAT('"
  243. . Zend_Registry::get('staticSalt')
  244. . "', ?, password_salt))"
  245. );]]></programlisting>
  246. <note>
  247. <para>
  248. salt がたとえハードコーディングされた固定文字列であったとしても、
  249. セキュリティを向上させることができます。
  250. 仮に (<acronym>SQL</acronym> インジェクション攻撃などで) データベースに侵入されたとしても、
  251. ウェブサーバは無傷なのでデータを攻撃者に悪用されることはありません。
  252. </para>
  253. </note>
  254. <para>
  255. もうひとつの方法は、アダプタを作成したあとで <classname>Zend_Auth_Adapter_DbTable</classname>
  256. の <methodname>getDbSelect()</methodname> メソッドを使うことです。
  257. このメソッドが返す <classname>Zend_Db_Select</classname> オブジェクトのインスタンスで
  258. <methodname>authenticate()</methodname> を実行します。このメソッドは、<methodname>authenticate()</methodname>
  259. をコールしたかどうかにかかわらず同じオブジェクトを返すことに注意しましょう。
  260. このオブジェクトには識別や認証のための情報は含まれておらず、
  261. <methodname>authenticate()</methodname> によってそれらが組み込まれます。
  262. </para>
  263. <para>
  264. <methodname>getDbSelect()</methodname> メソッドを使いたくなるひとつの例としては、
  265. たとえばユーザの状態のチェック、
  266. つまりそのユーザアカウントが有効になっているかどうかの確認などがあります。
  267. </para>
  268. <programlisting language="php"><![CDATA[
  269. // 上の例の続き
  270. $adapter = new Zend_Auth_Adapter_DbTable(
  271. $db,
  272. 'users',
  273. 'username',
  274. 'password',
  275. 'MD5(?)'
  276. );
  277. // select オブジェクトを (参照として) 取得します
  278. $select = $adapter->getDbSelect();
  279. $select->where('active = "TRUE"');
  280. // 認証。これにより、users.active = TRUE であることを保証します
  281. $adapter->authenticate();
  282. ]]></programlisting>
  283. </sect2>
  284. </sect1>