2016-01-15 12:21:27 +03:00
|
|
|
<?php
|
|
|
|
namespace api\components\ReCaptcha;
|
|
|
|
|
2016-08-03 15:56:08 +03:00
|
|
|
use common\helpers\Error as E;
|
2017-05-18 02:09:26 +03:00
|
|
|
use GuzzleHttp\ClientInterface;
|
2017-05-18 17:06:01 +03:00
|
|
|
use GuzzleHttp\Exception\ConnectException;
|
|
|
|
use GuzzleHttp\Exception\ServerException;
|
|
|
|
use Psr\Http\Message\ResponseInterface;
|
2016-01-15 12:21:27 +03:00
|
|
|
use Yii;
|
|
|
|
use yii\base\Exception;
|
2017-05-18 02:09:26 +03:00
|
|
|
use yii\di\Instance;
|
2016-01-15 12:21:27 +03:00
|
|
|
|
|
|
|
class Validator extends \yii\validators\Validator {
|
|
|
|
|
2017-05-18 17:06:01 +03:00
|
|
|
private const SITE_VERIFY_URL = 'https://www.google.com/recaptcha/api/siteverify';
|
|
|
|
|
|
|
|
private const REPEAT_LIMIT = 3;
|
|
|
|
private const REPEAT_TIMEOUT = 1;
|
2016-01-15 12:21:27 +03:00
|
|
|
|
|
|
|
public $skipOnEmpty = false;
|
|
|
|
|
2016-08-03 15:56:08 +03:00
|
|
|
public $message = E::CAPTCHA_INVALID;
|
|
|
|
|
|
|
|
public $requiredMessage = E::CAPTCHA_REQUIRED;
|
2016-01-15 12:21:27 +03:00
|
|
|
|
2017-05-18 02:09:26 +03:00
|
|
|
/**
|
|
|
|
* @var Component|string
|
|
|
|
*/
|
|
|
|
public $component = 'reCaptcha';
|
|
|
|
|
|
|
|
private $client;
|
|
|
|
|
|
|
|
public function __construct(ClientInterface $client, array $config = []) {
|
|
|
|
parent::__construct($config);
|
|
|
|
$this->client = $client;
|
|
|
|
}
|
|
|
|
|
2016-01-15 12:21:27 +03:00
|
|
|
public function init() {
|
|
|
|
parent::init();
|
2017-05-18 02:09:26 +03:00
|
|
|
$this->component = Instance::ensure($this->component, Component::class);
|
2016-01-15 12:21:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @inheritdoc
|
|
|
|
*/
|
|
|
|
protected function validateValue($value) {
|
|
|
|
if (empty($value)) {
|
2016-08-03 15:56:08 +03:00
|
|
|
return [$this->requiredMessage, []];
|
2016-01-15 12:21:27 +03:00
|
|
|
}
|
|
|
|
|
2017-05-18 17:06:01 +03:00
|
|
|
$repeats = 0;
|
|
|
|
do {
|
|
|
|
$isSuccess = true;
|
|
|
|
try {
|
|
|
|
$response = $this->performRequest($value);
|
|
|
|
} catch (ConnectException | ServerException $e) {
|
|
|
|
if (++$repeats >= self::REPEAT_LIMIT) {
|
|
|
|
throw $e;
|
|
|
|
}
|
|
|
|
|
|
|
|
$isSuccess = false;
|
|
|
|
sleep(self::REPEAT_TIMEOUT);
|
|
|
|
}
|
|
|
|
} while (!$isSuccess);
|
|
|
|
|
|
|
|
/** @noinspection PhpUndefinedVariableInspection */
|
|
|
|
$data = json_decode($response->getBody(), true);
|
|
|
|
if (!isset($data['success'])) {
|
|
|
|
throw new Exception('Invalid recaptcha verify response.');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$data['success']) {
|
|
|
|
return [$this->message, []];
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $value
|
|
|
|
* @throws \GuzzleHttp\Exception\GuzzleException
|
|
|
|
* @return ResponseInterface
|
|
|
|
*/
|
|
|
|
protected function performRequest(string $value): ResponseInterface {
|
|
|
|
return $this->client->request('POST', self::SITE_VERIFY_URL, [
|
2016-08-03 15:56:08 +03:00
|
|
|
'form_params' => [
|
2017-05-18 02:09:26 +03:00
|
|
|
'secret' => $this->component->secret,
|
2016-08-03 15:56:08 +03:00
|
|
|
'response' => $value,
|
|
|
|
'remoteip' => Yii::$app->getRequest()->getUserIP(),
|
|
|
|
],
|
|
|
|
]);
|
2016-01-15 12:21:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|