Zend_Validate-WritingValidators.xml 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296
  1. <sect1 id="zend.validate.writing_validators">
  2. <title>Escribiendo Validadores</title>
  3. <para>
  4. Zend_Validate provee un conjunto de validadores que suelen necesitarse,
  5. pero inevitablemente, los desarrolladores quiere escribir sus propios
  6. validadores personalizados para sus necesidades particulares.
  7. La tarea de escribir un validador personalizado se describe en esta sección.
  8. </para>
  9. <para>
  10. <code>Zend_Validate_Interface</code> define tres métodos, isValid(),
  11. getMessages(), y getErrors(), que pueden ser implementadas por clases de usuario
  12. a fin de crear objetos de validación personalizados.
  13. Un objeto que implementa una interfaz <code>Zend_Validate_Interface</code>
  14. puede añadirse a una cadena de validación con <code>Zend_Validate::addValidator()</code>.
  15. Tales objetos también pueden ser usados con
  16. <link linkend="zend.filter.input"><code>Zend_Filter_Input</code></link>.
  17. </para>
  18. <para>
  19. De la descripción anterior de <code>Zend_Validate_Interface</code>, podrá inferir
  20. que las clases de validación que proporciona Zend Framework devuelven un
  21. valor booleano para cuando un valor se valida satisfactoriamente o no.
  22. También proporcionan información sobre <emphasis role="bold">por qué</emphasis>
  23. un valor falló en la validación.
  24. La disponibilidad de las razones para los fracasos de validación puede ser
  25. valiosa para una aplicación por diversos motivos, tales como proporcionar
  26. estadísticas para análisis de usabilidad.
  27. </para>
  28. <para>
  29. La funcionalidad de los mensajes de validación básica de fallos están
  30. implementados en <code>Zend_Validate_Abstract</code>.
  31. A fin de incluir esta funcionalidad al crear una clase de validación,
  32. simplemente extienda <code>Zend_Validate_Abstract</code>.
  33. En la extensión de la clase deberá aplicar la lógica del método
  34. <code>isValid()</code> y definir las variables y plantillas de mensajes
  35. que correspondan a los tipos de fallos de validación que puedan suceder.
  36. Si falla un valor en su test de validación, entonces <code>isValid()</code>
  37. deberá devolver <code>false</code>.
  38. Si el valor pasa su test de validación, entonces <code>isValid()</code>
  39. deberá devolver <code>true</code>.
  40. </para>
  41. <para>
  42. En general, el método <code>isValid()</code> no debería arrojar
  43. excepciones, salvo que sea imposible determinar si el valor de entrada
  44. es válido o no.
  45. Algunos ejemplos de casos razonables para lanzar una excepción podría ser
  46. si un archivo no puede abrirse, que un servidor LDAP no pudiera ser
  47. contactado, o una conexión a una base de datos no estuviera disponible.
  48. Estos son casos en los que puede ser necesario determinar el éxito o
  49. fracaso de la validación.
  50. </para>
  51. <example id="zend.validate.writing_validators.example.simple">
  52. <title>Crear una Clase de Validación sencilla</title>
  53. <para>
  54. El siguiente ejemplo demuestra cómo podría escribirse un sencillo
  55. validador personalizado.
  56. En este caso las reglas de validación son simplemente que el valor
  57. de entrada debe ser de punto flotante.
  58. <programlisting role="php"><![CDATA[
  59. class MyValid_Float extends Zend_Validate_Abstract
  60. {
  61. const FLOAT = 'float';
  62. protected $_messageTemplates = array(
  63. self::FLOAT => "'%value%' no es un valor de punto flotante"
  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. ]]>
  76. </programlisting>
  77. La clase define una plantilla para su único mensaje de fallo de
  78. validación, que incluye el mágico parámetro <code>%value%</code>.
  79. La llamada a <code>_setValue()</code> prepara al objeto para insertar
  80. automáticamente en el mensaje de fallo al valor probado, si éste
  81. falla en la validación.
  82. La llamada a <code>_error()</code> sigue las pistas para establecer
  83. una razón por el fracaso de la validación.
  84. Dado que esta clase sólo define un mensaje de fallo, no es necesario
  85. darle a <code>_error()</code> el nombre de la plantilla del mensaje
  86. de fallo.
  87. </para>
  88. </example>
  89. <example id="zend.validate.writing_validators.example.conditions.dependent">
  90. <title>Escribiendo una Clase de Validación habiendo Condiciones Dependientes </title>
  91. <para>
  92. El siguiente ejemplo muestra un conjunto de reglas de validación
  93. más complejo, donde es necesario que el valor de entrada ser numérico
  94. y dentro del límite de un rango de valores mínimos y máximos.
  95. Un valor de entrada podría fallar en la validación exactamente por una
  96. de las siguientes razones:
  97. <itemizedlist>
  98. <listitem>
  99. <para>El valor de entrada no es numérico.</para>
  100. </listitem>
  101. <listitem>
  102. <para>El valor de entrada es menor que el valor mínimo permitido.</para>
  103. </listitem>
  104. <listitem>
  105. <para>El valor de entrada es mayor que el valor máximo permitido.</para>
  106. </listitem>
  107. </itemizedlist>
  108. </para>
  109. <para>
  110. Estas razones en el fallo de validación, son traducidas a las definiciones en la clase:
  111. <programlisting role="php"><![CDATA[
  112. class MyValid_NumericBetween extends Zend_Validate_Abstract
  113. {
  114. const MSG_NUMERIC = 'msgNumeric';
  115. const MSG_MINIMUM = 'msgMinimum';
  116. const MSG_MAXIMUM = 'msgMaximum';
  117. public $minimum = 0;
  118. public $maximum = 100;
  119. protected $_messageVariables = array(
  120. 'min' => 'minimum',
  121. 'max' => 'maximum'
  122. );
  123. protected $_messageTemplates = array(
  124. self::MSG_NUMERIC => "'%value%' no es numérico",
  125. self::MSG_MINIMUM => "'%value%' debe ser al menos '%min%'",
  126. self::MSG_MAXIMUM => "'%value%' debe ser no mayor a '%max%'"
  127. );
  128. public function isValid($value)
  129. {
  130. $this->_setValue($value);
  131. if (!is_numeric($value)) {
  132. $this->_error(self::MSG_NUMERIC);
  133. return false;
  134. }
  135. if ($value < $this->minimum) {
  136. $this->_error(self::MSG_MINIMUM);
  137. return false;
  138. }
  139. if ($value > $this->maximum) {
  140. $this->_error(self::MSG_MAXIMUM);
  141. return false;
  142. }
  143. return true;
  144. }
  145. }
  146. ]]>
  147. </programlisting>
  148. Las propiedades públicas <code>$minimum</code> y <code>$maximum</code>
  149. se han establecido para proporcionar los límites mínimo y máximo,
  150. respectivamente, de un valor a validar.
  151. La clase también define dos variables de mensajes que corresponden a
  152. las propiedades públicas y permiten usar <code>min</code> y <code>max</code>
  153. en plantillas de mensajes como parámetros mágicos, al igual que con
  154. <code>value</code>.
  155. </para>
  156. <para>
  157. Tenga en cuenta que si cualquiera de las comprobaciones de validación
  158. falla en <code>isValid()</code>, ya está preparado un mensaje apropiado,
  159. y el método inmediatamente devuelve <code>false</code>.
  160. Estas reglas de validación son por lo tanto secuencialmente dependientes.
  161. Es decir, si uno de los tests falla, no hay necesidad de poner a
  162. prueba las posteriores reglas de validación.
  163. Sin embargo, esta necesidad no será el caso.
  164. El siguiente ejemplo ilustra cómo escribir una clase con reglas de
  165. validación independientes, donde el objeto validación puede devolver
  166. múltiples razones por las cuales fracasó un intento de validación en
  167. particular.
  168. </para>
  169. </example>
  170. <example id="zend.validate.writing_validators.example.conditions.independent">
  171. <title>Validación con Condiciones Independientes, Múltiples Razones del Fracaso</title>
  172. <para>
  173. Considere escribir una clase de validación y control de
  174. contraseñas - cuando es necesario que un usuario elija una
  175. contraseña que cumple determinados criterios para ayudar a tener
  176. cuentas de usuario seguras. Supongamos que la seguridad de la contraseña
  177. aplica criterios que fuerzan a lo siguiente:
  178. <itemizedlist>
  179. <listitem>
  180. <para>debe tener al menos una longitud de 8 caracteres,</para>
  181. </listitem>
  182. <listitem>
  183. <para>contener al menos una letra en mayúscula,</para>
  184. </listitem>
  185. <listitem>
  186. <para>contener al menos una letra en minúscula,</para>
  187. </listitem>
  188. <listitem>
  189. <para>contener al menos un dígito.</para>
  190. </listitem>
  191. </itemizedlist>
  192. </para>
  193. <para>
  194. La siguiente clase implementa estos criterios de validación:
  195. <programlisting role="php"><![CDATA[
  196. class MyValid_PasswordStrength extends Zend_Validate_Abstract
  197. {
  198. const LENGTH = 'length';
  199. const UPPER = 'upper';
  200. const LOWER = 'lower';
  201. const DIGIT = 'digit';
  202. protected $_messageTemplates = array(
  203. self::LENGTH => "'%value%' debe tener al menos una longitud de 8 caracteres",
  204. self::UPPER => "'%value%' debe contener al menos una letra en mayúscula",
  205. self::LOWER => "'%value%' debe contener al menos una letra en minúscula",
  206. self::DIGIT => "'%value%' debe contener al menos un dígito"
  207. );
  208. public function isValid($value)
  209. {
  210. $this->_setValue($value);
  211. $isValid = true;
  212. if (strlen($value) < 8) {
  213. $this->_error(self::LENGTH);
  214. $isValid = false;
  215. }
  216. if (!preg_match('/[A-Z]/', $value)) {
  217. $this->_error(self::UPPER);
  218. $isValid = false;
  219. }
  220. if (!preg_match('/[a-z]/', $value)) {
  221. $this->_error(self::LOWER);
  222. $isValid = false;
  223. }
  224. if (!preg_match('/\d/', $value)) {
  225. $this->_error(self::DIGIT);
  226. $isValid = false;
  227. }
  228. return $isValid;
  229. }
  230. }
  231. ]]>
  232. </programlisting>
  233. Las cuatro pruebas de criterio en <code>isValid()</code> no devuelven
  234. inmediatamente <code>false</code>.
  235. Esto permite a la clase de validación a proporcionar <emphasis role="bold">todas</emphasis>
  236. las razones por las que la contraseña de entrada no cumplió los requisitos
  237. de validación.
  238. Si, por ejemplo, un usuario ingresara el string "<code>#$%</code>"
  239. como contraseña, <code>isValid()</code> causaría que los cuatro
  240. mensajes de fracaso de validación sean devueltos por un llamado posterior
  241. a <code>getMessages()</code>.
  242. </para>
  243. </example>
  244. </sect1>
  245. <!--
  246. vim:se ts=4 sw=4 et:
  247. -->