Zend_Validate-WritingValidators.xml 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!-- EN-Revision: 17175 -->
  3. <!-- Reviewed: no -->
  4. <sect1 id="zend.validate.writing_validators">
  5. <title>Schreiben von Prüfern</title>
  6. <para>
  7. <classname>Zend_Validate</classname> bietet ein Set von standardmäßig benötigten Prüfern,
  8. aber zwangsläufig, werden Entwickler wünschen, eigene Prüfer für die eigenen Bedürfnisse zu
  9. schreiben. Die Aufgabe des Schreibens eigener Prüfer wird in diesem Kapitel beschrieben.
  10. </para>
  11. <para>
  12. <classname>Zend_Validate_Interface</classname> definiert drei Methoden,
  13. <methodname>isValid()</methodname>, <methodname>getMessages()</methodname>, und
  14. <methodname>getErrors()</methodname>, welche von Benutzerklassen implementiert werden können
  15. um eigene Prüfobjekte zu erstellen. Ein Objekt welches das
  16. <classname>Zend_Validate_Interface</classname> Interface implementiert kann einer Prüfkette
  17. mit <methodname>Zend_Validate::addValidator()</methodname> hinzugefügt werden. Solche
  18. Objekte können auch mit <link
  19. linkend="zend.filter.input"><classname>Zend_Filter_Input</classname></link> verwendet
  20. werden.
  21. </para>
  22. <para>
  23. Wie man bereits aus der obigen Beschreibung von
  24. <classname>Zend_Validate_Interface</classname> folgern kann, geben die vom Zend Framework
  25. bereitgestellten Prüfklassen einen boolschen Wert zurück, ob die Prüfung des Wertes
  26. erfolgreich war oder nicht. Sie geben auch darüber Informationen
  27. <emphasis>warum</emphasis> ein Wert die Prüfung nicht bestanden hat. Die
  28. Verfügbarkeit der Gründe für fehlgeschlagene Prüfungen kann für eine Anwendung aus vielen
  29. Gründen nützlich sein, wie zum Beispiel das zur Verfügung stellen von Statistiken für
  30. Useability Analysen.
  31. </para>
  32. <para>
  33. Grundlegende Funktionalitäten für fehlgeschlagene Prüfmeldungen ist in
  34. <classname>Zend_Validate_Abstract</classname> implementiert. Um diese Funktionalität
  35. einzubinden wenn eine Prüfklasse erstellt wird, muß einfach
  36. <classname>Zend_Validate_Abstract</classname> erweitert werden. In der existierenden Klasse
  37. wird die Logik der <methodname>isValid()</methodname> Methode implementiert und die
  38. Variablen für die Nachrichten und Nachrichten-Templates definiert werden die zu den Typen
  39. von Prüffehlern passen die auftreten können. Wenn ein Wert die Prüfung nicht besteht, sollte
  40. <methodname>isValid()</methodname> <constant>FALSE</constant> zurückgeben. Wenn der Wert die
  41. Prüfung besteht, sollte <methodname>isValid()</methodname> <constant>TRUE</constant>
  42. zurückgeben.
  43. </para>
  44. <para>
  45. Normalerweise sollte die <methodname>isValid()</methodname> Methode keine Ausnahmen werfen,
  46. ausser wenn es unmöglich ist festzustellen ob ein Eingabewert gültig ist oder nicht. Einige
  47. Beispiele für gute Fälle in denen keine Ausnahme geworfen werden sollte sind, wenn eine
  48. Datei nicht geöffnet werden konnte, ein <acronym>LDAP</acronym> Server nicht erreicht wurde,
  49. oder eine Datenbank Verbindung unerreichbar ist, und wo solche Dinge für Prüfprozesse
  50. benötigt werden um zu erkennen ob die Prüfung gültig oder ungültig ist.
  51. </para>
  52. <example id="zend.validate.writing_validators.example.simple">
  53. <title>Erstellen einer einfachen Prüfklasse</title>
  54. <para>
  55. Das folgende Beispiel demonstriert wie ein sehr einfacher eigener Prüfer geschrieben
  56. werden könnte. In diesem Fall sind die Prüfregeln sehr einfach und der Eingabewert muß
  57. ein Gleitkommawert sein.
  58. <programlisting language="php"><![CDATA[
  59. class MyValid_Float extends Zend_Validate_Abstract
  60. {
  61. const FLOAT = 'float';
  62. protected $_messageTemplates = array(
  63. self::FLOAT => "'%value%' ist kein Gleitkommawert"
  64. );
  65. public function isValid($value)
  66. {
  67. $this->_setValue($value);
  68. if (!is_float($value)) {
  69. $this->_error();
  70. return false;
  71. }
  72. return true;
  73. }
  74. }
  75. ]]></programlisting>
  76. Die Klasse definiert ein Template für Ihre einzige Nachricht bei Prüfungsfehlern, welche
  77. den eingebauten magischen Parameter <code>%value%</code> inkludiert. Der Aufruf von
  78. <methodname>_setValue()</methodname> präpariert das Objekt den getesteten Wert
  79. automatisch in die Fehlernachricht einzufügen, wenn die Prüfung des Wertes fehlschlägt.
  80. Der Aufruf von <methodname>_error()</methodname> spürt den Grund für die fehlgeschlagene
  81. Prüfung auf. Da diese Klasse nur eine Fehlernachricht definiert, ist es nicht notwendig
  82. <methodname>_error()</methodname> den Namen des Templates der Fehlernachricht zu geben.
  83. </para>
  84. </example>
  85. <example id="zend.validate.writing_validators.example.conditions.dependent">
  86. <title>Schreiben einer Prüfklasse die abhängige Konditionen besitzt</title>
  87. <para>
  88. Das folgende Beispiel demonstriert ein komplexeres Set von Prüfregeln, wobei es
  89. notwendig ist das der Eingabewert nummerisch und innerhalb eines Bereiches von Mindest-
  90. und Maximalgrenzwerten ist. Bei einem Eingabewert würde die Prüfung wegen exakt einer
  91. der folgenden Gründe fehlschlagen:
  92. <itemizedlist>
  93. <listitem>
  94. <para>Der Eingabewert ist nicht nummerisch.</para>
  95. </listitem>
  96. <listitem>
  97. <para>Der Eingabewert ist kleiner als der minimal erlaubte Wert.</para>
  98. </listitem>
  99. <listitem>
  100. <para>Der Eingabewert ist größer als der maximal erlaubte Wert.</para>
  101. </listitem>
  102. </itemizedlist>
  103. </para>
  104. <para>
  105. Diese Gründe für fehlgeschlagene Prüfungen werden in Definitionen der Klasse übersetzt:
  106. <programlisting language="php"><![CDATA[
  107. class MyValid_NumericBetween extends Zend_Validate_Abstract
  108. {
  109. const MSG_NUMERIC = 'msgNumeric';
  110. const MSG_MINIMUM = 'msgMinimum';
  111. const MSG_MAXIMUM = 'msgMaximum';
  112. public $minimum = 0;
  113. public $maximum = 100;
  114. protected $_messageVariables = array(
  115. 'min' => 'minimum',
  116. 'max' => 'maximum'
  117. );
  118. protected $_messageTemplates = array(
  119. self::MSG_NUMERIC => "'%value%' ist nicht nummerisch",
  120. self::MSG_MINIMUM => "'%value%' muß mindestens '%min%' sein",
  121. self::MSG_MAXIMUM => "'%value%' darf nicht mehr als '%max%' sein"
  122. );
  123. public function isValid($value)
  124. {
  125. $this->_setValue($value);
  126. if (!is_numeric($value)) {
  127. $this->_error(self::MSG_NUMERIC);
  128. return false;
  129. }
  130. if ($value < $this->minimum) {
  131. $this->_error(self::MSG_MINIMUM);
  132. return false;
  133. }
  134. if ($value > $this->maximum) {
  135. $this->_error(self::MSG_MAXIMUM);
  136. return false;
  137. }
  138. return true;
  139. }
  140. }
  141. ]]></programlisting>
  142. Die öffentlichen Eigenschaften <varname>$minimum</varname> und
  143. <varname>$maximum</varname> wurden eingeführt um die Mindest- und Maximalgrenzen
  144. anzubieten, beziehungsweise, für einen Wert um die Prüfung erfolgreich zu bestehen. Die
  145. Klasse definiert auch zwei Nachrichtenvariablen die zu den öffentlichen Eigenschaften
  146. korrespondieren und es erlauben <code>min</code> und <code>max</code> in den Nachrichten
  147. Templates als magische Parameter zu verwenden, genauso wie <code>value</code>.
  148. </para>
  149. <para>
  150. Zu beachten ist, das wenn eine der Prüfungen in <methodname>isValid()</methodname>
  151. fehlschlägt, eine entsprechende Fehlernachricht vorbereitet wird, und die Methode sofort
  152. <constant>FALSE</constant> zurückgibt. Diese Prüfregeln sind deswegen sequentiell
  153. abhängig. Das bedeuted, wenn einer der Tests fehlschlägt, gibt es keinen Grund eine
  154. weitere nachfolgende Prüfregel zu testen. Das muß aber trotzdem nicht der Fall sein. Das
  155. folgende Beispiel zeigt wie man eine Klasse schreiben kann die unabhängige Prüfregeln
  156. besitzt, wo die Prüfobjekte mehrfache Gründe zurückgeben könnten, warum ein spezieller
  157. Prüfversuch fehlgeschlagen ist.
  158. </para>
  159. </example>
  160. <example id="zend.validate.writing_validators.example.conditions.independent">
  161. <title>Prüfen mit unabhängigen Konditionen, mehrfache Gründe für Fehler</title>
  162. <para>
  163. Angenommen es wird eine Prüfklasse geschrieben für das Erzwingen von Passwortstärke -
  164. wenn ein Benutzer ein Passwort auswählen muß das diversen Kriterien entspricht um zu
  165. Helfen das die Benutzerzugänge sicher sind. Angenommen die Passwort Sicherheitskriterien
  166. erzwingen das folgende Passwort:
  167. <itemizedlist>
  168. <listitem>
  169. <para>mindestens 8 Zeichen Länge,</para>
  170. </listitem>
  171. <listitem>
  172. <para>enthält mindestens ein großgeschriebenes Zeichen,</para>
  173. </listitem>
  174. <listitem>
  175. <para>enthält mindestens ein kleingeschriebenes Zeichen,</para>
  176. </listitem>
  177. <listitem>
  178. <para>und enthält mindestens eine Ziffer.</para>
  179. </listitem>
  180. </itemizedlist>
  181. </para>
  182. <para>
  183. Die folgende Klasse impementiert diese Prüfkriterien:
  184. <programlisting language="php"><![CDATA[
  185. class MyValid_PasswordStrength extends Zend_Validate_Abstract
  186. {
  187. const LENGTH = 'length';
  188. const UPPER = 'upper';
  189. const LOWER = 'lower';
  190. const DIGIT = 'digit';
  191. protected $_messageTemplates = array(
  192. self::LENGTH => "'%value%' muß mindestens 8 Zeichen lang sein",
  193. self::UPPER => "'%value%' muß mindestens ein großgeschriebenes "
  194. . "Zeichen enthalten",
  195. self::LOWER => "'%value%' muß mindestens ein kleingeschriebenes "
  196. . "Zeichen enthalten",
  197. self::DIGIT => "'%value%' muß mindestens eine Ziffer enthalten"
  198. );
  199. public function isValid($value)
  200. {
  201. $this->_setValue($value);
  202. $isValid = true;
  203. if (strlen($value) < 8) {
  204. $this->_error(self::LENGTH);
  205. $isValid = false;
  206. }
  207. if (!preg_match('/[A-Z]/', $value)) {
  208. $this->_error(self::UPPER);
  209. $isValid = false;
  210. }
  211. if (!preg_match('/[a-z]/', $value)) {
  212. $this->_error(self::LOWER);
  213. $isValid = false;
  214. }
  215. if (!preg_match('/\d/', $value)) {
  216. $this->_error(self::DIGIT);
  217. $isValid = false;
  218. }
  219. return $isValid;
  220. }
  221. }
  222. ]]></programlisting>
  223. Zu beachten ist das diese vier Testkriterien in <methodname>isValid()</methodname> nicht
  224. sofort <constant>FALSE</constant> zurückgeben. Das erlaubt der Prüfklasse
  225. <emphasis>alle</emphasis> Gründe anzubieten bei denen das Eingabepasswort
  226. den Prüfvoraussetzungen nicht entsprochen hat. Wenn, zum Beispiel, ein Benutzer den
  227. String "<code>#$%</code>" als Passwort angegeben hat, würde
  228. <methodname>isValid()</methodname> alle vier Prüfungfehlermeldungen zurückgeben bei
  229. einen nachfolgenden Aufruf von <methodname>getMessages()</methodname>.
  230. </para>
  231. </example>
  232. </sect1>
  233. <!--
  234. vim:se ts=4 sw=4 et:
  235. -->