Zend_Auth_Adapter_DbTable.xml 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <!-- EN-Revision: 15617 -->
  3. <!-- Reviewed: no -->
  4. <sect1 id="zend.auth.adapter.dbtable">
  5. <title>Authentification avec une table de base de données</title>
  6. <sect2 id="zend.auth.adapter.dbtable.introduction">
  7. <title>Introduction</title>
  8. <para>
  9. <classname>Zend_Auth_Adapter_DbTable</classname> fournit la possibilité d'authentifier
  10. sur la base de crédits stockés dans une table de base de données. Comme
  11. <classname>Zend_Auth_Adapter_DbTable</classname> requiert qu'une instance de
  12. <classname>Zend_Db_Adapter_Abstract</classname> soit fournie à son constructeur, chaque
  13. instance est liée à une connexion de base de données particulière. Les autres options
  14. de configuration peuvent être réglées grâce au constructeur ou au travers de
  15. différentes méthodes, une pour chaque option.
  16. </para>
  17. <para>
  18. Les options de configuration disponibles incluent&#160;:
  19. <itemizedlist>
  20. <listitem>
  21. <para><code>tableName</code>&#160;: il s'agit du nom de la table dans la base de
  22. données qui contient les crédits d'authentification, et envers laquelle la
  23. requête d'authentification sera réalisée.
  24. </para>
  25. </listitem>
  26. <listitem>
  27. <para>
  28. <code>identityColumn</code>&#160;: il s'agit du nom de la colonne dans la
  29. table utilisée pour représenter l'identité. La colonne d'identité doit
  30. contenir une valeur unique, comme un "<code>username</code>" ou une adresse
  31. émail.
  32. </para>
  33. </listitem>
  34. <listitem>
  35. <para>
  36. <code>credentialColumn</code>&#160;: il s'agit du nom de la colonne dans la
  37. table utilisée pour représenter le crédit. Dans le cas d'une simple
  38. authentification par identité / mot de passe, la valeur de crédit
  39. correspond au mot de passe. Voir aussi l'option
  40. <code>credentialTreatment</code>.
  41. </para>
  42. </listitem>
  43. <listitem>
  44. <para>
  45. <code>credentialTreatment</code>&#160;: dans la plupart des cas, les mots de
  46. passe et autres données sensibles sont cryptés, hachés, encodés, masqués, ou
  47. sinon traités à travers une fonction ou un algorithme. En spécifiant un
  48. traitement paramétrable de chaîne avec cette méthode, comme
  49. <code>'MD5(?)'</code> ou <code>'PASSWORD(?)'</code>, un développeur peut
  50. appliquer un code SQL arbitraire sur les données de crédit fournies. Comme
  51. ces fonctions sont spécifiques à chaque gestionnaire de base de données,
  52. vérifiez le manuel de la base de données pour vérifier la disponibilité de
  53. ces fonctions dans votre système.
  54. </para>
  55. </listitem>
  56. </itemizedlist>
  57. </para>
  58. <example id="zend.auth.adapter.dbtable.introduction.example.basic_usage">
  59. <title>Utilisation basique</title>
  60. <para>
  61. Comme expliqué dans l'introduction, le constructeur de
  62. <classname>Zend_Auth_Adapter_DbTable</classname> requiert une instance de
  63. <classname>Zend_Db_Adapter_Abstract</classname> qui est utilisée comme connexion à
  64. la base de données à laquelle l'instance d'adaptateur d'authentification est liée.
  65. Avant tout, la connexion à la base de donnée devrait être crée.
  66. </para>
  67. <para>
  68. Le code suivant crée un adaptateur pour une base de données en mémoire, crée
  69. un schéma avec une table unique, et insère une ligne sur laquelle nous pouvons
  70. réaliser une requête d'authentification plus tard. Cet exemple requiert que
  71. l'extension PDO SQLite soit disponible&#160;:
  72. <programlisting language="php"><![CDATA[
  73. // Crée une connexion de base de données SQLite en mémoire
  74. $dbAdapter = new Zend_Db_Adapter_Pdo_Sqlite(array('dbname' =>
  75. ':memory:'));
  76. // Construit une requête de création de table
  77. $sqlCreate = 'CREATE TABLE [users] ( '
  78. . '[id] INTEGER NOT NULL PRIMARY KEY, '
  79. . '[username] VARCHAR(50) UNIQUE NOT NULL, '
  80. . '[password] VARCHAR(32) NULL, '
  81. . '[real_name] VARCHAR(150) NULL)';
  82. // Crée la table de crédits d'authentification
  83. $dbAdapter->query($sqlCreate);
  84. // Construit la requête pour insérer une ligne pour laquelle
  85. // l'authentification pourra réussir
  86. $sqlInsert = "INSERT INTO users (username, password, real_name) "
  87. . "VALUES ('my_username', 'my_password', 'My Real Name')";
  88. // Insertion des données
  89. $dbAdapter->query($sqlInsert);
  90. ]]></programlisting>
  91. </para>
  92. <para>
  93. Avec une connexion de base de données et des données disponibles dans la
  94. table, une instance de <classname>Zend_Auth_Adapter_DbTable</classname> peut être
  95. créée. Les valeurs d'options de configuration peuvent être fournies au
  96. constructeur ou en tant que paramètres aux méthodes de réglage après
  97. l'instanciation&#160;:
  98. <programlisting language="php"><![CDATA[
  99. // Configure une instance avec des paramètres de constructeur ...
  100. $authAdapter = new Zend_Auth_Adapter_DbTable($dbAdapter,
  101. 'users',
  102. 'username',
  103. 'password');
  104. // ... ou configure l'instance avec des méthodes de réglage
  105. $authAdapter = new Zend_Auth_Adapter_DbTable($dbAdapter);
  106. $authAdapter->setTableName('users')
  107. ->setIdentityColumn('username')
  108. ->setCredentialColumn('password');
  109. ]]></programlisting>
  110. </para>
  111. <para>
  112. A cet instant, l'instance de l'adaptateur d'authentification est prête à
  113. recevoir des requêtes d'authentification. Dans le but de réaliser une requête
  114. d'authentification, les valeurs des crédits requêtés sont fournies à l'adaptateur
  115. avant d'appeler la méthode <code>authenticate()</code>&#160;:
  116. <programlisting language="php"><![CDATA[
  117. // Règle les valeurs d'entrées des crédits
  118. // (en général, à partir d'un formulaire d'enregistrement)
  119. $authAdapter->setIdentity('my_username')
  120. ->setCredential('my_password');
  121. // Réalise la requête d'authentification, et sauvegarde le résultat
  122. $result = $authAdapter->authenticate();
  123. ]]></programlisting>
  124. </para>
  125. <para>
  126. En plus de la disponibilité de la méthode <code>getIdentity()</code> pour
  127. récupérer l'objet du résultat d'authentification,
  128. <classname>Zend_Auth_Adapter_DbTable</classname> supporte aussi la récupération de
  129. la ligne de la table qui a réussi l'authentification&#160;:
  130. <programlisting language="php"><![CDATA[
  131. // Affiche l'identité
  132. echo $result->getIdentity() . "\n\n";
  133. // Affiche la ligne de résultat
  134. print_r($authAdapter->getResultRowObject());
  135. /* Affiche:
  136. my_username
  137. Array
  138. (
  139. [id] => 1
  140. [username] => my_username
  141. [password] => my_password
  142. [real_name] => My Real Name
  143. )
  144. */
  145. ]]></programlisting>
  146. Puisque la ligne de la table contient la valeur de crédit, il est important de
  147. garantir ces valeurs contre l'accès fortuit.
  148. </para>
  149. </example>
  150. </sect2>
  151. <sect2 id="zend.auth.adapter.dbtable.advanced.storing_result_row">
  152. <title>Utilisation avancée : maintenir persistant l'objet de résultat DbTable</title>
  153. <para>
  154. Par défaut, <classname>Zend_Auth_Adapter_DbTable</classname> retourne l'identité
  155. fournie à l'objet Auth en cas d'authentification couronnée de succès. Un autre scénario
  156. d'utilisation, où les développeurs veulent stocker dans le mécanisme de stockage
  157. persistant du <classname>Zend_Auth</classname> un objet d'identité contenant d'autres
  158. informations utiles, est résolu en utilisant la méthode
  159. <code>getResultRowObject()</code> retournant un objet <code>stdClass</code>. Le petit
  160. bout de code suivant illustre cette utilisation&#160;:
  161. <programlisting language="php"><![CDATA[
  162. // authentifie avec Zend_Auth_Adapter_DbTable
  163. $result = $this->_auth->authenticate($adapter);
  164. if ($result->isValid()) {
  165. // stocke l'identité comme objet dans lequel seulement username et
  166. // real_name sont retournés
  167. $storage = $this->_auth->getStorage();
  168. $storage->write($adapter->getResultRowObject(array('username',
  169. 'real_name')));
  170. // stocke l'identité comme objet dans lequel la colonne password
  171. // a été omis
  172. $storage->write($adapter->getResultRowObject(null, 'password'));
  173. /* ... */
  174. } else {
  175. /* ... */
  176. }
  177. ]]></programlisting>
  178. </para>
  179. </sect2>
  180. <sect2 id="zend.auth.adapter.dbtable.advanced.advanced_usage">
  181. <title>Utilisation avancée par l'exemple</title>
  182. <para>
  183. Bien que le but initial de <classname>Zend_Auth</classname> (et par extension celui de
  184. <classname>Zend_Auth_Adapter_DbTable</classname>) est principalement
  185. l'<emphasis>authentification</emphasis> et non
  186. l'<emphasis>autorisation</emphasis> (ou contrôle d'accès), il existe quelques
  187. exemples et problèmes qui franchissent la limite des domaines auxquels ils
  188. appartiennent. Selon la façon dont vous avez décidé d'expliquer votre problème, il
  189. semble parfois raisonnable de résoudre ce qui pourrait ressembler à un problème
  190. d'autorisation dans l'adaptateur d'authentification.
  191. </para>
  192. <para>
  193. Ceci étant dit, <classname>Zend_Auth_Adapter_DbTable</classname> possède des mécanismes
  194. qui sont construits de telle sorte qu'ils peuvent être démultipliés pour ajouter des
  195. contrôles supplémentaires au moment de l'authentification pour résoudre quelques
  196. problèmes communs d'utilisateur.
  197. </para>
  198. <programlisting language="php"><![CDATA[
  199. // La valeur du champs "etat" d'un compte
  200. // ne doit pas être égal à "compromis"
  201. $adapter = new Zend_Auth_Adapter_DbTable($db,
  202. 'utilisateurs',
  203. 'login',
  204. 'password',
  205. 'MD5(?) AND etat != "compromis"');
  206. // La valeur du champs "actif" d'un compte
  207. // doit être égal à "TRUE"
  208. $adapter = new Zend_Auth_Adapter_DbTable($db,
  209. 'utilisateurs',
  210. 'login',
  211. 'password',
  212. 'MD5(?) AND actif = "TRUE"');
  213. ]]></programlisting>
  214. <para>
  215. Un autre scénario possible est l'implantation d'un mécanisme de "salting".
  216. "Salting" est un terme se référant une technique qui peut fortement améliorer la
  217. sécurité de votre application. C'est basé sur l'idée que concaténer une chaîne
  218. aléatoire à tout mot de passe rend impossible la réussite d'une attaque de type "brute
  219. force" sur la base de données en utilisant des valeurs préalablement hashées issues
  220. d'un dictionnaire.
  221. </para>
  222. <para>
  223. Par conséquent nous devons modifier notre table pour stocker notre chaîne de
  224. "salt" aléatoire&#160;:
  225. </para>
  226. <programlisting language="php"><![CDATA[
  227. $sqlAlter = "ALTER TABLE [users] "
  228. . "ADD COLUMN [password_salt] "
  229. . "AFTER [password]";
  230. $dbAdapter->query($sqlAlter);
  231. ]]></programlisting>
  232. <para>
  233. Voici une méthode simple pour générer une chaîne aléatoire pour chaque
  234. utilisateur à leur enregistrement&#160;:
  235. </para>
  236. <programlisting language="php"><![CDATA[
  237. for ($i = 0; $i < 50; $i++)
  238. {
  239. $dynamicSalt .= chr(rand(33, 126));
  240. }
  241. ]]></programlisting>
  242. <para>Et maintenant, construisons l'adaptateur&#160;:</para>
  243. <programlisting language="php"><![CDATA[
  244. $adapter = new Zend_Auth_Adapter_DbTable(
  245. $db,
  246. 'users',
  247. 'username',
  248. 'password',
  249. "MD5(CONCAT('"
  250. . Zend_Registry::get('staticSalt')
  251. . "', ?, password_salt))"
  252. );
  253. ]]></programlisting>
  254. <note>
  255. <para>
  256. Vous pouvez encore améliorer la sécurité en utilisant une chaîne de "salt"
  257. statique codée en dur dans votre application. Dans le cas ou la base de données est
  258. compromise (par exemple par une attaque de type injection SQL) mais que votre
  259. serveur Web est intact, les données sont inutilisables par l'attaquant.
  260. </para>
  261. </note>
  262. </sect2>
  263. </sect1>