EachValidator.php 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. <?php
  2. /**
  3. * @link https://www.yiiframework.com/
  4. * @copyright Copyright (c) 2008 Yii Software LLC
  5. * @license https://www.yiiframework.com/license/
  6. */
  7. namespace yii\validators;
  8. use Yii;
  9. use yii\base\DynamicModel;
  10. use yii\base\InvalidConfigException;
  11. use yii\base\Model;
  12. /**
  13. * EachValidator validates an array by checking each of its elements against an embedded validation rule.
  14. *
  15. * ```php
  16. * class MyModel extends Model
  17. * {
  18. * public $categoryIDs = [];
  19. *
  20. * public function rules()
  21. * {
  22. * return [
  23. * // checks if every category ID is an integer
  24. * ['categoryIDs', 'each', 'rule' => ['integer']],
  25. * ]
  26. * }
  27. * }
  28. * ```
  29. *
  30. * > Note: This validator will not work with inline validation rules in case of usage outside the model scope,
  31. * e.g. via [[validate()]] method.
  32. *
  33. * > Note: EachValidator is meant to be used only in basic cases, you should consider usage of tabular input,
  34. * using several models for the more complex case.
  35. *
  36. * @author Paul Klimov <klimov.paul@gmail.com>
  37. * @since 2.0.4
  38. */
  39. class EachValidator extends Validator
  40. {
  41. /**
  42. * @var array|Validator definition of the validation rule, which should be used on array values.
  43. * It should be specified in the same format as at [[\yii\base\Model::rules()]], except it should not
  44. * contain attribute list as the first element.
  45. * For example:
  46. *
  47. * ```php
  48. * ['integer']
  49. * ['match', 'pattern' => '/[a-z]/is']
  50. * ```
  51. *
  52. * Please refer to [[\yii\base\Model::rules()]] for more details.
  53. */
  54. public $rule;
  55. /**
  56. * @var bool whether to use error message composed by validator declared via [[rule]] if its validation fails.
  57. * If enabled, error message specified for this validator itself will appear only if attribute value is not an array.
  58. * If disabled, own error message value will be used always.
  59. */
  60. public $allowMessageFromRule = true;
  61. /**
  62. * @var bool whether to stop validation once first error among attribute value elements is detected.
  63. * When enabled validation will produce single error message on attribute, when disabled - multiple
  64. * error messages mya appear: one per each invalid value.
  65. * Note that this option will affect only [[validateAttribute()]] value, while [[validateValue()]] will
  66. * not be affected.
  67. * @since 2.0.11
  68. */
  69. public $stopOnFirstError = true;
  70. /**
  71. * {@inheritdoc}
  72. */
  73. public function init()
  74. {
  75. parent::init();
  76. if ($this->message === null) {
  77. $this->message = Yii::t('yii', '{attribute} is invalid.');
  78. }
  79. }
  80. /**
  81. * Creates validator object based on the validation rule specified in [[rule]].
  82. * @param Model|null $model model in which context validator should be created.
  83. * @param mixed|null $current value being currently validated.
  84. * @throws \yii\base\InvalidConfigException
  85. * @return Validator validator instance
  86. */
  87. private function createEmbeddedValidator($model = null, $current = null)
  88. {
  89. $rule = $this->rule;
  90. if ($rule instanceof Validator) {
  91. return $rule;
  92. }
  93. if (is_array($rule) && isset($rule[0])) { // validator type
  94. if (!is_object($model)) {
  95. $model = new Model(); // mock up context model
  96. }
  97. $params = array_slice($rule, 1);
  98. $params['current'] = $current;
  99. return Validator::createValidator($rule[0], $model, $this->attributes, $params);
  100. }
  101. throw new InvalidConfigException('Invalid validation rule: a rule must be an array specifying validator type.');
  102. }
  103. /**
  104. * {@inheritdoc}
  105. */
  106. public function validateAttribute($model, $attribute)
  107. {
  108. $arrayOfValues = $model->$attribute;
  109. if (!is_array($arrayOfValues) && !$arrayOfValues instanceof \ArrayAccess) {
  110. $this->addError($model, $attribute, $this->message, []);
  111. return;
  112. }
  113. foreach ($arrayOfValues as $k => $v) {
  114. $dynamicModel = new DynamicModel($model->getAttributes());
  115. $dynamicModel->setAttributeLabels($model->attributeLabels());
  116. $dynamicModel->addRule($attribute, $this->createEmbeddedValidator($model, $v));
  117. $dynamicModel->defineAttribute($attribute, $v);
  118. $dynamicModel->validate();
  119. $arrayOfValues[$k] = $dynamicModel->$attribute; // filtered values like 'trim'
  120. if (!$dynamicModel->hasErrors($attribute)) {
  121. continue;
  122. }
  123. if ($this->allowMessageFromRule) {
  124. $validationErrors = $dynamicModel->getErrors($attribute);
  125. $model->addErrors([$attribute => $validationErrors]);
  126. } else {
  127. $this->addError($model, $attribute, $this->message, ['value' => $v]);
  128. }
  129. if ($this->stopOnFirstError) {
  130. break;
  131. }
  132. }
  133. $model->$attribute = $arrayOfValues;
  134. }
  135. /**
  136. * {@inheritdoc}
  137. */
  138. protected function validateValue($value)
  139. {
  140. if (!is_array($value) && !$value instanceof \ArrayAccess) {
  141. return [$this->message, []];
  142. }
  143. $validator = $this->createEmbeddedValidator();
  144. foreach ($value as $v) {
  145. if ($validator->skipOnEmpty && $validator->isEmpty($v)) {
  146. continue;
  147. }
  148. $result = $validator->validateValue($v);
  149. if ($result !== null) {
  150. if ($this->allowMessageFromRule) {
  151. $result[1]['value'] = $v;
  152. return $result;
  153. }
  154. return [$this->message, ['value' => $v]];
  155. }
  156. }
  157. return null;
  158. }
  159. }