mirror of
				https://github.com/elyby/accounts.git
				synced 2025-05-31 14:11:46 +05:30 
			
		
		
		
	
		
			
				
	
	
		
			77 lines
		
	
	
		
			2.1 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			77 lines
		
	
	
		
			2.1 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						|
namespace api\validators;
 | 
						|
 | 
						|
use common\helpers\Error as E;
 | 
						|
use common\models\Account;
 | 
						|
use OTPHP\TOTP;
 | 
						|
use RangeException;
 | 
						|
use Yii;
 | 
						|
use yii\base\InvalidConfigException;
 | 
						|
use yii\validators\Validator;
 | 
						|
 | 
						|
class TotpValidator extends Validator {
 | 
						|
 | 
						|
    /**
 | 
						|
     * @var Account
 | 
						|
     */
 | 
						|
    public $account;
 | 
						|
 | 
						|
    /**
 | 
						|
     * @var int|null Specifies the window in the interval of which the code will be checked.
 | 
						|
     * Allows you to avoid the situation when the user entered the code in the last second of its existence
 | 
						|
     * and while the request was being sent, it has changed. The value is set in +- periods, not seconds.
 | 
						|
     */
 | 
						|
    public $window;
 | 
						|
 | 
						|
    /**
 | 
						|
     * @var int|callable|null Allows you to set the exact time against which the validation will be performed.
 | 
						|
     * It may be the unix time or a function returning a unix time.
 | 
						|
     * If not specified, the current time will be used.
 | 
						|
     */
 | 
						|
    public $timestamp;
 | 
						|
 | 
						|
    public $skipOnEmpty = false;
 | 
						|
 | 
						|
    public function init() {
 | 
						|
        parent::init();
 | 
						|
        if ($this->account === null) {
 | 
						|
            $this->account = Yii::$app->user->identity;
 | 
						|
        }
 | 
						|
 | 
						|
        if (!$this->account instanceof Account) {
 | 
						|
            throw new InvalidConfigException('account should be instance of ' . Account::class);
 | 
						|
        }
 | 
						|
 | 
						|
        if (empty($this->account->otp_secret)) {
 | 
						|
            throw new InvalidConfigException('account should have not empty otp_secret');
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    protected function validateValue($value) {
 | 
						|
        try {
 | 
						|
            $totp = TOTP::create($this->account->otp_secret);
 | 
						|
            if (!$totp->verify((string)$value, $this->getTimestamp(), $this->window)) {
 | 
						|
                return [E::TOTP_INCORRECT, []];
 | 
						|
            }
 | 
						|
        } catch (RangeException $e) {
 | 
						|
            return [E::TOTP_INCORRECT, []];
 | 
						|
        }
 | 
						|
 | 
						|
        return null;
 | 
						|
    }
 | 
						|
 | 
						|
    private function getTimestamp(): ?int {
 | 
						|
        $timestamp = $this->timestamp;
 | 
						|
        if (is_callable($timestamp)) {
 | 
						|
            $timestamp = call_user_func($this->timestamp);
 | 
						|
        }
 | 
						|
 | 
						|
        if ($timestamp === null) {
 | 
						|
            return null;
 | 
						|
        }
 | 
						|
 | 
						|
        return (int)$timestamp;
 | 
						|
    }
 | 
						|
 | 
						|
}
 |