Zend_Validate-WritingValidators.xml 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!-- Reviewed: no -->
  3. <!-- EN-Revision: 15103 -->
  4. <sect1 id="zend.validate.writing_validators">
  5. <title>バリデータの書き方</title>
  6. <para>
  7. Zend_Validate には、よく使うバリデータ群が付属しています。しかし、
  8. 特定の目的のために使用する独自のバリデータを書くことは避けられないでしょう。
  9. ここでは、独自のバリデータを書く方法について説明します。
  10. </para>
  11. <para>
  12. <classname>Zend_Validate_Interface</classname> では、3つのメソッド <code>isValid()</code>、
  13. <code>getMessages()</code> および <code>getErrors()</code>
  14. を定義しています。これらを自分のクラスで実装して
  15. 独自のバリデーションオブジェクトを作成します。
  16. <classname>Zend_Validate_Interface</classname> を実装したクラスは、
  17. <classname>Zend_Validate::addValidator()</classname>
  18. でバリデータチェインに追加することができます。
  19. このオブジェクトは
  20. <link linkend="zend.filter.input"><classname>Zend_Filter_Input</classname></link>
  21. でも使用します。
  22. </para>
  23. <para>
  24. 上の <classname>Zend_Validate_Interface</classname> についての説明からも推測できるように、
  25. Zend Framework が提供しているバリデーションクラスの返り値は、
  26. 検証に成功したか失敗したかを表す boolean 値となります。また、<emphasis role="bold">なぜ</emphasis>
  27. 検証が失敗したのかについての情報も提供します。この理由がわかると、
  28. アプリケーション側で何かと便利です。たとえば、ユーザビリティ改善のための統計情報として利用することなどができます。 
  29. </para>
  30. <para>
  31. 基本的な検証失敗メッセージ機能を実装しているのが <classname>Zend_Validate_Abstract</classname> です。
  32. バリデーションクラスを作成する際にこの機能を組み込むには、
  33. <classname>Zend_Validate_Abstract</classname> を継承します。
  34. 継承したクラス内で <code>isValid()</code> メソッドのロジックを実装し、
  35. 発生しうる失敗の形式に対応したメッセージ変数とメッセージテンプレートを定義します。
  36. 検証に失敗した場合には <code>isValid()</code> は <code>false</code>
  37. を返さなければなりません。検証を通過した場合は、<code>isValid()</code>
  38. は <code>true</code> を返さなければなりません。
  39. </para>
  40. <para>
  41. 一般に、<code>isValid()</code> メソッドは例外をスローすべきではありません。
  42. 例外をスローするのは、入力値が妥当かそうでないかの判断ができない場合のみとします。
  43. 例外をスローすることになる場面としては、たとえばファイルのオープンに失敗したり
  44. LDAP サーバとの接続に失敗したり、データベースとの接続に失敗したり
  45. といった原因で入力値が正しいのかどうかが判断できない場合が考えられます。
  46. </para>
  47. <example id="zend.validate.writing_validators.example.simple">
  48. <title>単純なバリデーションクラスの作成</title>
  49. <para>
  50. 次の例は、非常に単純なバリデータの書き方を説明するものです。
  51. ここで定義している検証規則は非常に単純で、入力値が浮動小数点値かどうかのみを調べています。
  52. <programlisting role="php"><![CDATA[
  53. class MyValid_Float extends Zend_Validate_Abstract
  54. {
  55. const FLOAT = 'float';
  56. protected $_messageTemplates = array(
  57. self::FLOAT => "'%value%' は浮動小数点値ではありません"
  58. );
  59. public function isValid($value)
  60. {
  61. $this->_setValue($value);
  62. if (!is_float($value)) {
  63. $this->_error();
  64. return false;
  65. }
  66. return true;
  67. }
  68. }
  69. ]]>
  70. </programlisting>
  71. このクラス内には、検証が失敗したときのメッセージ用のテンプレートがひとつ定義されており、
  72. その中では組み込みのマジックパラメータ <code>%value%</code> を使用しています。
  73. <code>_setValue()</code> のコールによって、検証した値をこのメッセージに自動的に格納します。
  74. これにより、検証に失敗したときに、この値をメッセージに含められるようになります。
  75. <code>_error()</code> のコールによって、検証に失敗した原因を取得します。
  76. このクラスでは失敗時のメッセージをひとつしか用意していないので、
  77. <code>_error()</code> でメッセージテンプレートの名前を指定する必要はありません。
  78. </para>
  79. </example>
  80. <example id="zend.validate.writing_validators.example.conditions.dependent">
  81. <title>依存条件を伴うバリデーションクラスの作成</title>
  82. <para>
  83. 次の例は、複数の検証規則を組み合わせた複雑なものとなります。
  84. ここでは、まず入力値が数値であること、そして指定した最小値と最大値の間にあることを調べます。
  85. 以下のいずれかが発生したときに、検証は失敗します。
  86. <itemizedlist>
  87. <listitem>
  88. <para>入力値が数値ではない</para>
  89. </listitem>
  90. <listitem>
  91. <para>入力値が最小値より小さい</para>
  92. </listitem>
  93. <listitem>
  94. <para>入力値が最大値より大きい</para>
  95. </listitem>
  96. </itemizedlist>
  97. </para>
  98. <para>
  99. これらの原因に応じて、クラス内でメッセージへの変換が行われます。
  100. <programlisting role="php"><![CDATA[
  101. class MyValid_NumericBetween extends Zend_Validate_Abstract
  102. {
  103. const MSG_NUMERIC = 'msgNumeric';
  104. const MSG_MINIMUM = 'msgMinimum';
  105. const MSG_MAXIMUM = 'msgMaximum';
  106. public $minimum = 0;
  107. public $maximum = 100;
  108. protected $_messageVariables = array(
  109. 'min' => 'minimum',
  110. 'max' => 'maximum'
  111. );
  112. protected $_messageTemplates = array(
  113. self::MSG_NUMERIC => "'%value%' は数値ではありません",
  114. self::MSG_MINIMUM => "'%value%' は '%min%' 以上でなければなりません",
  115. self::MSG_MAXIMUM => "'%value%' は '%max%' 以下でなければなりません"
  116. );
  117. public function isValid($value)
  118. {
  119. $this->_setValue($value);
  120. if (!is_numeric($value)) {
  121. $this->_error(self::MSG_NUMERIC);
  122. return false;
  123. }
  124. if ($value < $this->minimum) {
  125. $this->_error(self::MSG_MINIMUM);
  126. return false;
  127. }
  128. if ($value > $this->maximum) {
  129. $this->_error(self::MSG_MAXIMUM);
  130. return false;
  131. }
  132. return true;
  133. }
  134. }
  135. ]]>
  136. </programlisting>
  137. パブリックプロパティ <code>$minimum</code> および <code>$maximum</code>
  138. でそれぞれ最小値と最大値を定義し、値がこの間にあった場合に検証が成功したことにしています。
  139. このクラスではまた、それぞれのパブリックプロパティに対応するふたつのメッセージ変数を定義しています。
  140. そしてメッセージテンプレートの中で <code>value</code> と同様に使えるマジックパラメータとして
  141. <code>min</code> および <code>max</code> も用意しています。
  142. </para>
  143. <para>
  144. <code>isValid()</code> での妥当性チェックのいずれかが失敗すると、
  145. 適切なメッセージを準備して即時に <code>false</code> を返すことに注意しましょう。
  146. つまり、これらの検証は、その並び順に依存します。
  147. どれかひとつのチェックが失敗すると、それ以降の検証規則を確認する必要はなくなるからです。
  148. しかし、時にはこのような処理が不要な場合もあります。
  149. 次の例は、個々の検証規則を独立させたクラスを書く方法を示すものです。
  150. このバリデーションオブジェクトは、検証に失敗したときに複数の理由を返すことがあります。
  151. </para>
  152. </example>
  153. <example id="zend.validate.writing_validators.example.conditions.independent">
  154. <title>個々に独立した条件による検証を行い、失敗時に複数の原因を返す</title>
  155. <para>
  156. パスワードの強度を確認するバリデーションクラスを書くことを考えてみましょう。
  157. ユーザに対して、安全なユーザアカウントを発行するために
  158. ある条件を満たしたパスワードを設定してもらう際に使用するものです。
  159. パスワードの条件としては、次のようなものを前提とします。
  160. <itemizedlist>
  161. <listitem>
  162. <para>最低 8 文字以上であること</para>
  163. </listitem>
  164. <listitem>
  165. <para>最低ひとつの大文字を含むこと</para>
  166. </listitem>
  167. <listitem>
  168. <para>最低ひとつの小文字を含むこと</para>
  169. </listitem>
  170. <listitem>
  171. <para>最低ひとつの数字を含むこと</para>
  172. </listitem>
  173. </itemizedlist>
  174. </para>
  175. <para>
  176. この検証規則を実装したクラスは、次のようになります。
  177. <programlisting role="php"><![CDATA[
  178. class MyValid_PasswordStrength extends Zend_Validate_Abstract
  179. {
  180. const LENGTH = 'length';
  181. const UPPER = 'upper';
  182. const LOWER = 'lower';
  183. const DIGIT = 'digit';
  184. protected $_messageTemplates = array(
  185. self::LENGTH => "'%value%' は少なくとも 8 文字以上でなければなりません",
  186. self::UPPER => "'%value%' には最低ひとつの大文字が必要です",
  187. self::LOWER => "'%value%' には最低ひとつの小文字が必要です",
  188. self::DIGIT => "'%value%' には最低ひとつの数字が必要です"
  189. );
  190. public function isValid($value)
  191. {
  192. $this->_setValue($value);
  193. $isValid = true;
  194. if (strlen($value) < 8) {
  195. $this->_error(self::LENGTH);
  196. $isValid = false;
  197. }
  198. if (!preg_match('/[A-Z]/', $value)) {
  199. $this->_error(self::UPPER);
  200. $isValid = false;
  201. }
  202. if (!preg_match('/[a-z]/', $value)) {
  203. $this->_error(self::LOWER);
  204. $isValid = false;
  205. }
  206. if (!preg_match('/\d/', $value)) {
  207. $this->_error(self::DIGIT);
  208. $isValid = false;
  209. }
  210. return $isValid;
  211. }
  212. }
  213. ]]>
  214. </programlisting>
  215. <code>isValid()</code> では四つのチェックを行っていますが、
  216. チェックに失敗してもその場では <code>false</code> を返していないことに注目しましょう。
  217. これにより、入力されたパスワードが満たしていない条件を <emphasis role="bold">すべて</emphasis>
  218. 示すことができるようになります。たとえば、パスワードとして入力された値が
  219. "<code>#$%</code>" だったとすると、<code>isValid()</code>
  220. は四つのメッセージをすべて作成し、後で <code>getMessages()</code>
  221. をコールした際にすべて取得できるようになります。
  222. </para>
  223. </example>
  224. </sect1>
  225. <!--
  226. vim:se ts=4 sw=4 et:
  227. -->