CaptchaValidator.php 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118
  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\captcha;
  8. use Yii;
  9. use yii\base\InvalidConfigException;
  10. use yii\helpers\Json;
  11. use yii\validators\ValidationAsset;
  12. use yii\validators\Validator;
  13. /**
  14. * CaptchaValidator validates that the attribute value is the same as the verification code displayed in the CAPTCHA.
  15. *
  16. * CaptchaValidator should be used together with [[CaptchaAction]].
  17. *
  18. * Note that once CAPTCHA validation succeeds, a new CAPTCHA will be generated automatically. As a result,
  19. * CAPTCHA validation should not be used in AJAX validation mode because it may fail the validation
  20. * even if a user enters the same code as shown in the CAPTCHA image which is actually different from the latest CAPTCHA code.
  21. *
  22. * @author Qiang Xue <qiang.xue@gmail.com>
  23. * @since 2.0
  24. */
  25. class CaptchaValidator extends Validator
  26. {
  27. /**
  28. * @var bool whether to skip this validator if the input is empty.
  29. */
  30. public $skipOnEmpty = false;
  31. /**
  32. * @var bool whether the comparison is case sensitive. Defaults to false.
  33. */
  34. public $caseSensitive = false;
  35. /**
  36. * @var string the route of the controller action that renders the CAPTCHA image.
  37. */
  38. public $captchaAction = 'site/captcha';
  39. /**
  40. * {@inheritdoc}
  41. */
  42. public function init()
  43. {
  44. parent::init();
  45. if ($this->message === null) {
  46. $this->message = Yii::t('yii', 'The verification code is incorrect.');
  47. }
  48. }
  49. /**
  50. * {@inheritdoc}
  51. */
  52. protected function validateValue($value)
  53. {
  54. $captcha = $this->createCaptchaAction();
  55. $valid = !is_array($value) && $captcha->validate($value, $this->caseSensitive);
  56. return $valid ? null : [$this->message, []];
  57. }
  58. /**
  59. * Creates the CAPTCHA action object from the route specified by [[captchaAction]].
  60. * @return \yii\captcha\CaptchaAction the action object
  61. * @throws InvalidConfigException
  62. */
  63. public function createCaptchaAction()
  64. {
  65. $ca = Yii::$app->createController($this->captchaAction);
  66. if ($ca !== false) {
  67. /* @var $controller \yii\base\Controller */
  68. list($controller, $actionID) = $ca;
  69. $action = $controller->createAction($actionID);
  70. if ($action !== null) {
  71. return $action;
  72. }
  73. }
  74. throw new InvalidConfigException('Invalid CAPTCHA action ID: ' . $this->captchaAction);
  75. }
  76. /**
  77. * {@inheritdoc}
  78. */
  79. public function clientValidateAttribute($model, $attribute, $view)
  80. {
  81. ValidationAsset::register($view);
  82. $options = $this->getClientOptions($model, $attribute);
  83. return 'yii.validation.captcha(value, messages, ' . Json::htmlEncode($options) . ');';
  84. }
  85. /**
  86. * {@inheritdoc}
  87. */
  88. public function getClientOptions($model, $attribute)
  89. {
  90. $captcha = $this->createCaptchaAction();
  91. $code = $captcha->getVerifyCode(false);
  92. $hash = $captcha->generateValidationHash($this->caseSensitive ? $code : strtolower($code));
  93. $options = [
  94. 'hash' => $hash,
  95. 'hashKey' => 'yiiCaptcha/' . $captcha->getUniqueId(),
  96. 'caseSensitive' => $this->caseSensitive,
  97. 'message' => Yii::$app->getI18n()->format($this->message, [
  98. 'attribute' => $model->getAttributeLabel($attribute),
  99. ], Yii::$app->language),
  100. ];
  101. if ($this->skipOnEmpty) {
  102. $options['skipOnEmpty'] = 1;
  103. }
  104. return $options;
  105. }
  106. }