Zend_Auth_Adapter_DbTable.xml 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  1. <sect1 id="zend.auth.adapter.dbtable">
  2. <title>数据库表认证</title>
  3. <sect2 id="zend.auth.adapter.dbtable.introduction">
  4. <title>简介</title>
  5. <para>
  6. <code>Zend_Auth_Adapter_DbTable</code>提供依靠存储在数据库表中的证书来认证的能力。因为<code>Zend_Auth_Adapter_DbTable</code>需要<code>Zend_Db_Adapter_Abstract</code>的实例来传递给它的构造器,所以每个实例要和特定的数据库连接绑定。其它配置选项可以通过构造器和实例方法设置,每个选项有一个配置。
  7. </para>
  8. <para>
  9. 可用的配置选项包括:
  10. <itemizedlist>
  11. <listitem>
  12. <para>
  13. <code>tableName</code>: 包含认证证书的数据库表名,执行数据库认证查询需要依靠这个证书。
  14. </para>
  15. </listitem>
  16. <listitem>
  17. <para>
  18. <code>identityColumn</code>: 数据库表的列的名称,用来表示身份。身份列必须包含唯一的值,例如用户名或者e-mail地址。
  19. </para>
  20. </listitem>
  21. <listitem>
  22. <para>
  23. <code>credentialColumn</code>: 数据库表的列的名称,用来表示证书。在一个简单的身份和密码认证scheme下,证书的值对应为密码。参见 <code>credentialTreatment</code> 选项。
  24. </para>
  25. </listitem>
  26. <listitem>
  27. <para>
  28. <code>credentialTreatment</code>: 在许多情况下,密码和其他敏感数据是加密的(encrypted, hashed, encoded, obscured,salted 或者通过以下函数或算法来加工)。通过指定参数化的字串来使用这个方法,例如<code>'MD5(?)'</code> 或者 <code>'PASSWORD(?)'</code>,开发者可以在输入证书数据时使用任意的SQL。因为这些函数对其下面的RDBMS是专用的, 请查看数据库手册来确保你所用的数据库的函数的可用性。
  29. </para>
  30. </listitem>
  31. </itemizedlist>
  32. </para>
  33. <example id="zend.auth.adapter.dbtable.introduction.example.basic_usage">
  34. <title>基本用法</title>
  35. <para>
  36. 正如在简介中所解释的,<code>Zend_Auth_Adapter_DbTable</code>构造器需要一个<code>Zend_Db_Adapter_Abstract</code>的实例,这个实例用做数据库连结,并且认证适配器实例绑定到这个数据库连接。首先,应该创建数据库连接。
  37. </para>
  38. <para>
  39. 下面的代码为in-memory数据库创建一个适配器,创建一个简单的表schema,并插入我们将来可以执行认证查询的一行(数据)。这个例子需要PDO SQLite extension可用:
  40. </para>
  41. <programlisting role="php"><![CDATA[
  42. // 创建一个 in-memory SQLite 数据库连接
  43. $dbAdapter = new Zend_Db_Adapter_Pdo_Sqlite(array('dbname' =>
  44. ':memory:'));
  45. // 构造一个简单表的创建语句
  46. $sqlCreate = 'CREATE TABLE [users] ('
  47. . '[id] INTEGER NOT NULL PRIMARY KEY, '
  48. . '[username] VARCHAR(50) UNIQUE NOT NULL, '
  49. . '[password] VARCHAR(32) NULL, '
  50. . '[real_name] VARCHAR(150) NULL)';
  51. // 创建认证证书表
  52. $dbAdapter->query($sqlCreate);
  53. // 构造用来插入一行可以成功认证的数据的语句
  54. $sqlInsert = "INSERT INTO users (username, password, real_name) "
  55. . "VALUES ('my_username', 'my_password', 'My Real Name')";
  56. // 插入数据
  57. $dbAdapter->query($sqlInsert);
  58. ]]>
  59. </programlisting>
  60. <para>
  61. 随着数据库连接和表数据已经可用,<code>Zend_Auth_Adapter_DbTable</code>可以被创建。配置选项的值可以传递给构造器或者延后在实例化以后用做setter方法的参数:
  62. </para>
  63. <programlisting role="php"><![CDATA[
  64. // 用构造器参数来配置实例...
  65. $authAdapter = new Zend_Auth_Adapter_DbTable(
  66. $dbAdapter,
  67. 'users',
  68. 'username',
  69. 'password'
  70. );
  71. // 用构造器参数来配置实例...
  72. $authAdapter = new Zend_Auth_Adapter_DbTable($dbAdapter);
  73. $authAdapter
  74. ->setTableName('users')
  75. ->setIdentityColumn('username')
  76. ->setCredentialColumn('password')
  77. ;
  78. ]]>
  79. </programlisting>
  80. <para>
  81. 在这点上,认证适配器实例已经可以接受认证查询。为了合成一个认证查询,在调用<code>authenticate()</code>方法之前,输入的证书的值要传递给适配器:
  82. </para>
  83. <programlisting role="php"><![CDATA[
  84. // 设置输入的证书的值(例如,从登陆的表单)
  85. $authAdapter
  86. ->setIdentity('my_username')
  87. ->setCredential('my_password')
  88. ;
  89. // 执行认证查询,并保存结果
  90. $result = $authAdapter->authenticate();
  91. ]]>
  92. </programlisting>
  93. <para>
  94. 除了基于认证结果对象的 <code>getIdentity()</code> 方法的可用性之外,<code>Zend_Auth_Adapter_DbTable</code>也支持从认证成功的表中读取一行数据:
  95. </para>
  96. <programlisting role="php"><![CDATA[
  97. // 输出身份
  98. echo $result->getIdentity() . "\n\n";
  99. // 输出结果行
  100. print_r($authAdapter->getResultRowObject());
  101. /* Output:
  102. my_username
  103. Array
  104. (
  105. [id] => 1
  106. [username] => my_username
  107. [password] => my_password
  108. [real_name] => My Real Name
  109. )
  110. */
  111. ]]>
  112. </programlisting>
  113. <para>
  114. 因为表行里包含证书值,通过防止无意识地访问来安全化这个值很重要。
  115. </para>
  116. </example>
  117. </sect2>
  118. <sect2 id="zend.auth.adapter.dbtable.advanced.storing_result_row">
  119. <title>高级使用:持久一个 DbTable 结果对象</title>
  120. <para>
  121. 缺省地,基于成功的认证<code>Zend_Auth_Adapter_DbTable</code> 返回提供给auth对象的身份。对于其他用例情景,如开发者想给<code>Zend_Auth</code> 的持久存储机制存储一个包括其他有用信息的身份对象,已经通过使用<code>getResultRowObject()</code> 方法返回一个<code>stdClass</code>对象解决了。下面的代码片段举例说明它的用法:
  122. </para>
  123. <programlisting role="php"><![CDATA[
  124. // authenticate with Zend_Auth_Adapter_DbTable
  125. $result = $this->_auth->authenticate($adapter);
  126. if ($result->isValid()) {
  127. // store the identity as an object where only the username and
  128. //real_name have been returned
  129. $storage = $this->_auth->getStorage();
  130. $storage->write($adapter->getResultRowObject(array(
  131. 'username',
  132. 'real_name',
  133. )));
  134. // store the identity as an object where the password column has
  135. // been omitted
  136. $storage->write($adapter->getResultRowObject(
  137. null,
  138. 'password'
  139. ));
  140. /* ... */
  141. } else {
  142. /* ... */
  143. }
  144. ]]>
  145. </programlisting>
  146. </sect2>
  147. <sect2 id="zend.auth.adapter.dbtable.advanced.advanced_usage">
  148. <title>高级用法范例</title>
  149. <para>
  150. 虽然 Zend_Auth (和它的继承者 Zend_Auth_Adapter_DbTable )主要用来 <emphasis role="strong">认证</emphasis> 而不是 <emphasis role="strong">授权</emphasis> ,但是基于它们用在哪个域名下,还是有一些实例和问题。根据如何解释你的问题,有时候通过在认证适配器里检查授权问题也许能解决问题。
  151. </para>
  152. <para>
  153. 用一点不恰当的免责声明,Zend_Auth_Adapter_DbTable 有内建的机制可以用来利用添加另外的认证时的检查来解决一些普通的用户问题。
  154. </para>
  155. <programlisting role="php"><![CDATA[
  156. // The status field value of an account is not equal to "compromised"
  157. $adapter = new Zend_Auth_Adapter_DbTable(
  158. $db,
  159. 'users',
  160. 'username',
  161. 'password',
  162. 'MD5(?) AND status != "compromised"'
  163. );
  164. // The active field value of an account is equal to "TRUE"
  165. $adapter = new Zend_Auth_Adapter_DbTable(
  166. $db,
  167. 'users',
  168. 'username',
  169. 'password',
  170. 'MD5(?) AND active = "TRUE"'
  171. );
  172. ]]>
  173. </programlisting>
  174. <para>
  175. 另外一个场景是免疫机制的实现。免疫是指大幅提高程序安全的技术术语。
  176. 它的想法是基于连接随机字符串到每个密码来使从字典里预先计算好的哈希值来完成强力攻击数据库成为可能。
  177. </para>
  178. <para>
  179. 因此我们需要修改表来存储免疫的字符串:
  180. </para>
  181. <programlisting role="php"><![CDATA[
  182. $sqlAlter = "ALTER TABLE [users] "
  183. . "ADD COLUMN [password_salt] "
  184. . "AFTER [password]";
  185. $dbAdapter->query($sqlAlter);
  186. ]]>
  187. </programlisting>
  188. <para>
  189. 这里是在注册时给每个用户生成免疫字符串的简单的方法:
  190. </para>
  191. <programlisting role="php"><![CDATA[
  192. for ($i = 0; $i < 50; $i++)
  193. {
  194. $dynamicSalt .= chr(rand(33, 126));
  195. }
  196. ]]>
  197. </programlisting>
  198. <para>
  199. 构造适配器:
  200. </para>
  201. <programlisting role="php"><![CDATA[
  202. $adapter = new Zend_Auth_Adapter_DbTable(
  203. $db,
  204. 'users',
  205. 'username',
  206. 'password',
  207. "MD5(CONCAT('"
  208. . Zend_Registry::get('staticSalt')
  209. . "', ?, password_salt))"
  210. );
  211. ]]>
  212. </programlisting>
  213. <note>
  214. <para>
  215. 你可以通过使用静态免疫值硬编码到程序里来更好地改善安全问题。
  216. 万一你的数据库有安全隐患(例如 SQL 注入攻击),但你的 web 服务器
  217. 依然完整,攻击者仍得不到你的数据。
  218. </para>
  219. </note>
  220. </sect2>
  221. </sect1>