mirror of
				https://github.com/elyby/accounts.git
				synced 2025-05-31 14:11:46 +05:30 
			
		
		
		
	#274: упразднена базовая модель KeyConfirmationForm, внедрён обновлённый валидатор EmailActivationKeyValidator
This commit is contained in:
		@@ -2,56 +2,53 @@
 | 
			
		||||
namespace api\models\authentication;
 | 
			
		||||
 | 
			
		||||
use api\models\AccountIdentity;
 | 
			
		||||
use api\models\base\KeyConfirmationForm;
 | 
			
		||||
use api\models\base\ApiForm;
 | 
			
		||||
use api\models\profile\ChangeUsernameForm;
 | 
			
		||||
use api\validators\EmailActivationKeyValidator;
 | 
			
		||||
use common\models\Account;
 | 
			
		||||
use common\models\EmailActivation;
 | 
			
		||||
use Yii;
 | 
			
		||||
use yii\base\ErrorException;
 | 
			
		||||
 | 
			
		||||
class ConfirmEmailForm extends KeyConfirmationForm {
 | 
			
		||||
class ConfirmEmailForm extends ApiForm {
 | 
			
		||||
 | 
			
		||||
    public $key;
 | 
			
		||||
 | 
			
		||||
    public function rules() {
 | 
			
		||||
        return [
 | 
			
		||||
            ['key', EmailActivationKeyValidator::class, 'type' => EmailActivation::TYPE_REGISTRATION_EMAIL_CONFIRMATION],
 | 
			
		||||
        ];
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * @return \api\components\User\LoginResult|bool
 | 
			
		||||
     * @throws ErrorException
 | 
			
		||||
     */
 | 
			
		||||
    public function confirm() {
 | 
			
		||||
        if (!$this->validate()) {
 | 
			
		||||
            return false;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $confirmModel = $this->getActivationCodeModel();
 | 
			
		||||
        if ($confirmModel->type !== EmailActivation::TYPE_REGISTRATION_EMAIL_CONFIRMATION) {
 | 
			
		||||
            $confirmModel->delete();
 | 
			
		||||
            // TODO: вот где-то здесь нужно ещё попутно сгенерировать соответствующую ошибку
 | 
			
		||||
            return false;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $transaction = Yii::$app->db->beginTransaction();
 | 
			
		||||
        try {
 | 
			
		||||
            $account = $confirmModel->account;
 | 
			
		||||
            $account->status = Account::STATUS_ACTIVE;
 | 
			
		||||
            if (!$confirmModel->delete()) {
 | 
			
		||||
                throw new ErrorException('Unable remove activation key.');
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            if (!$account->save()) {
 | 
			
		||||
                throw new ErrorException('Unable activate user account.');
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            $changeUsernameForm = new ChangeUsernameForm();
 | 
			
		||||
            $changeUsernameForm->createEventTask($account->id, $account->username, null);
 | 
			
		||||
 | 
			
		||||
            $transaction->commit();
 | 
			
		||||
        } catch (ErrorException $e) {
 | 
			
		||||
            $transaction->rollBack();
 | 
			
		||||
            if (YII_DEBUG) {
 | 
			
		||||
                throw $e;
 | 
			
		||||
            } else {
 | 
			
		||||
                return false;
 | 
			
		||||
            }
 | 
			
		||||
        /** @var \common\models\confirmations\RegistrationConfirmation $confirmModel */
 | 
			
		||||
        $confirmModel = $this->key;
 | 
			
		||||
        $account = $confirmModel->account;
 | 
			
		||||
        $account->status = Account::STATUS_ACTIVE;
 | 
			
		||||
        if (!$confirmModel->delete()) {
 | 
			
		||||
            throw new ErrorException('Unable remove activation key.');
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /** @var \api\components\User\Component $component */
 | 
			
		||||
        $component = Yii::$app->user;
 | 
			
		||||
        if (!$account->save()) {
 | 
			
		||||
            throw new ErrorException('Unable activate user account.');
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return $component->login(new AccountIdentity($account->attributes), true);
 | 
			
		||||
        $changeUsernameForm = new ChangeUsernameForm();
 | 
			
		||||
        $changeUsernameForm->createEventTask($account->id, $account->username, null);
 | 
			
		||||
 | 
			
		||||
        $transaction->commit();
 | 
			
		||||
 | 
			
		||||
        return Yii::$app->user->login(new AccountIdentity($account->attributes), true);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -2,26 +2,30 @@
 | 
			
		||||
namespace api\models\authentication;
 | 
			
		||||
 | 
			
		||||
use api\models\AccountIdentity;
 | 
			
		||||
use api\models\base\KeyConfirmationForm;
 | 
			
		||||
use api\models\base\ApiForm;
 | 
			
		||||
use api\validators\EmailActivationKeyValidator;
 | 
			
		||||
use common\helpers\Error as E;
 | 
			
		||||
use common\models\EmailActivation;
 | 
			
		||||
use common\validators\PasswordValidator;
 | 
			
		||||
use Yii;
 | 
			
		||||
use yii\base\ErrorException;
 | 
			
		||||
 | 
			
		||||
class RecoverPasswordForm extends KeyConfirmationForm {
 | 
			
		||||
class RecoverPasswordForm extends ApiForm {
 | 
			
		||||
 | 
			
		||||
    public $key;
 | 
			
		||||
 | 
			
		||||
    public $newPassword;
 | 
			
		||||
 | 
			
		||||
    public $newRePassword;
 | 
			
		||||
 | 
			
		||||
    public function rules() {
 | 
			
		||||
        return array_merge(parent::rules(), [
 | 
			
		||||
        return [
 | 
			
		||||
            ['key', EmailActivationKeyValidator::class, 'type' => EmailActivation::TYPE_FORGOT_PASSWORD_KEY],
 | 
			
		||||
            ['newPassword', 'required', 'message' => E::NEW_PASSWORD_REQUIRED],
 | 
			
		||||
            ['newRePassword', 'required', 'message' => E::NEW_RE_PASSWORD_REQUIRED],
 | 
			
		||||
            ['newPassword', PasswordValidator::class],
 | 
			
		||||
            ['newRePassword', 'validatePasswordAndRePasswordMatch'],
 | 
			
		||||
        ]);
 | 
			
		||||
        ];
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function validatePasswordAndRePasswordMatch($attribute) {
 | 
			
		||||
@@ -32,46 +36,32 @@ class RecoverPasswordForm extends KeyConfirmationForm {
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * @return \api\components\User\LoginResult|bool
 | 
			
		||||
     * @throws ErrorException
 | 
			
		||||
     */
 | 
			
		||||
    public function recoverPassword() {
 | 
			
		||||
        if (!$this->validate()) {
 | 
			
		||||
            return false;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $confirmModel = $this->getActivationCodeModel();
 | 
			
		||||
        if ($confirmModel->type !== EmailActivation::TYPE_FORGOT_PASSWORD_KEY) {
 | 
			
		||||
            $confirmModel->delete();
 | 
			
		||||
            // TODO: вот где-то здесь нужно ещё попутно сгенерировать соответствующую ошибку
 | 
			
		||||
            return false;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $transaction = Yii::$app->db->beginTransaction();
 | 
			
		||||
        try {
 | 
			
		||||
            $account = $confirmModel->account;
 | 
			
		||||
            $account->password = $this->newPassword;
 | 
			
		||||
            if (!$confirmModel->delete()) {
 | 
			
		||||
                throw new ErrorException('Unable remove activation key.');
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            if (!$account->save(false)) {
 | 
			
		||||
                throw new ErrorException('Unable activate user account.');
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            $transaction->commit();
 | 
			
		||||
        } catch (ErrorException $e) {
 | 
			
		||||
            $transaction->rollBack();
 | 
			
		||||
            if (YII_DEBUG) {
 | 
			
		||||
                throw $e;
 | 
			
		||||
            } else {
 | 
			
		||||
                return false;
 | 
			
		||||
            }
 | 
			
		||||
        /** @var \common\models\confirmations\ForgotPassword $confirmModel */
 | 
			
		||||
        $confirmModel = $this->key;
 | 
			
		||||
        $account = $confirmModel->account;
 | 
			
		||||
        $account->password = $this->newPassword;
 | 
			
		||||
        if (!$confirmModel->delete()) {
 | 
			
		||||
            throw new ErrorException('Unable remove activation key.');
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // TODO: ещё было бы неплохо уведомить пользователя о том, что его пароль изменился
 | 
			
		||||
        if (!$account->save(false)) {
 | 
			
		||||
            throw new ErrorException('Unable activate user account.');
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /** @var \api\components\User\Component $component */
 | 
			
		||||
        $component = Yii::$app->user;
 | 
			
		||||
        $transaction->commit();
 | 
			
		||||
 | 
			
		||||
        return $component->login(new AccountIdentity($account->attributes), false);
 | 
			
		||||
        return Yii::$app->user->login(new AccountIdentity($account->attributes), false);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,33 +0,0 @@
 | 
			
		||||
<?php
 | 
			
		||||
namespace api\models\base;
 | 
			
		||||
 | 
			
		||||
use common\helpers\Error as E;
 | 
			
		||||
use api\validators\EmailActivationKeyValidator;
 | 
			
		||||
use common\models\EmailActivation;
 | 
			
		||||
 | 
			
		||||
class KeyConfirmationForm extends ApiForm {
 | 
			
		||||
 | 
			
		||||
    public $key;
 | 
			
		||||
 | 
			
		||||
    private $model;
 | 
			
		||||
 | 
			
		||||
    public function rules() {
 | 
			
		||||
        return [
 | 
			
		||||
            // TODO: нужно провалидировать количество попыток ввода кода для определённого IP адреса и в случае чего запросить капчу
 | 
			
		||||
            ['key', 'required', 'message' => E::KEY_REQUIRED],
 | 
			
		||||
            ['key', EmailActivationKeyValidator::class],
 | 
			
		||||
        ];
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * @return EmailActivation|null
 | 
			
		||||
     */
 | 
			
		||||
    public function getActivationCodeModel() {
 | 
			
		||||
        if ($this->model === null) {
 | 
			
		||||
            $this->model = EmailActivation::findOne($this->key);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return $this->model;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,60 +1,60 @@
 | 
			
		||||
<?php
 | 
			
		||||
namespace api\models\profile\ChangeEmail;
 | 
			
		||||
 | 
			
		||||
use api\models\base\KeyConfirmationForm;
 | 
			
		||||
use api\models\base\ApiForm;
 | 
			
		||||
use api\validators\EmailActivationKeyValidator;
 | 
			
		||||
use common\helpers\Amqp;
 | 
			
		||||
use common\models\Account;
 | 
			
		||||
use common\models\amqp\EmailChanged;
 | 
			
		||||
use Exception;
 | 
			
		||||
use common\models\EmailActivation;
 | 
			
		||||
use PhpAmqpLib\Message\AMQPMessage;
 | 
			
		||||
use Yii;
 | 
			
		||||
use yii\base\ErrorException;
 | 
			
		||||
 | 
			
		||||
class ConfirmNewEmailForm extends KeyConfirmationForm {
 | 
			
		||||
class ConfirmNewEmailForm extends ApiForm {
 | 
			
		||||
 | 
			
		||||
    public $key;
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * @var Account
 | 
			
		||||
     */
 | 
			
		||||
    private $account;
 | 
			
		||||
 | 
			
		||||
    public function __construct(Account $account, array $config = []) {
 | 
			
		||||
        $this->account = $account;
 | 
			
		||||
        parent::__construct($config);
 | 
			
		||||
    public function rules() {
 | 
			
		||||
        return [
 | 
			
		||||
            ['key', EmailActivationKeyValidator::class, 'type' => EmailActivation::TYPE_NEW_EMAIL_CONFIRMATION],
 | 
			
		||||
        ];
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * @return Account
 | 
			
		||||
     */
 | 
			
		||||
    public function getAccount() : Account {
 | 
			
		||||
    public function getAccount(): Account {
 | 
			
		||||
        return $this->account;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function changeEmail() : bool {
 | 
			
		||||
    public function changeEmail(): bool {
 | 
			
		||||
        if (!$this->validate()) {
 | 
			
		||||
            return false;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $transaction = Yii::$app->db->beginTransaction();
 | 
			
		||||
        try {
 | 
			
		||||
            /** @var \common\models\confirmations\NewEmailConfirmation $activation */
 | 
			
		||||
            $activation = $this->getActivationCodeModel();
 | 
			
		||||
            $activation->delete();
 | 
			
		||||
 | 
			
		||||
            $account = $this->getAccount();
 | 
			
		||||
            $oldEmail = $account->email;
 | 
			
		||||
            $account->email = $activation->newEmail;
 | 
			
		||||
            if (!$account->save()) {
 | 
			
		||||
                throw new ErrorException('Cannot save new account email value');
 | 
			
		||||
            }
 | 
			
		||||
        /** @var \common\models\confirmations\NewEmailConfirmation $activation */
 | 
			
		||||
        $activation = $this->key;
 | 
			
		||||
        $activation->delete();
 | 
			
		||||
 | 
			
		||||
            $this->createTask($account->id, $account->email, $oldEmail);
 | 
			
		||||
 | 
			
		||||
            $transaction->commit();
 | 
			
		||||
        } catch (Exception $e) {
 | 
			
		||||
            $transaction->rollBack();
 | 
			
		||||
            throw $e;
 | 
			
		||||
        $account = $this->getAccount();
 | 
			
		||||
        $oldEmail = $account->email;
 | 
			
		||||
        $account->email = $activation->newEmail;
 | 
			
		||||
        if (!$account->save()) {
 | 
			
		||||
            throw new ErrorException('Cannot save new account email value');
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $this->createTask($account->id, $account->email, $oldEmail);
 | 
			
		||||
 | 
			
		||||
        $transaction->commit();
 | 
			
		||||
 | 
			
		||||
        return true;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
@@ -77,4 +77,9 @@ class ConfirmNewEmailForm extends KeyConfirmationForm {
 | 
			
		||||
        Amqp::sendToEventsExchange('accounts.email-changed', $message);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function __construct(Account $account, array $config = []) {
 | 
			
		||||
        $this->account = $account;
 | 
			
		||||
        parent::__construct($config);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,17 +1,19 @@
 | 
			
		||||
<?php
 | 
			
		||||
namespace api\models\profile\ChangeEmail;
 | 
			
		||||
 | 
			
		||||
use api\models\base\KeyConfirmationForm;
 | 
			
		||||
use api\models\base\ApiForm;
 | 
			
		||||
use api\validators\EmailActivationKeyValidator;
 | 
			
		||||
use common\models\Account;
 | 
			
		||||
use common\models\confirmations\NewEmailConfirmation;
 | 
			
		||||
use common\models\EmailActivation;
 | 
			
		||||
use common\validators\EmailValidator;
 | 
			
		||||
use Yii;
 | 
			
		||||
use yii\base\ErrorException;
 | 
			
		||||
use yii\base\Exception;
 | 
			
		||||
use yii\base\InvalidConfigException;
 | 
			
		||||
 | 
			
		||||
class NewEmailForm extends KeyConfirmationForm {
 | 
			
		||||
class NewEmailForm extends ApiForm {
 | 
			
		||||
 | 
			
		||||
    public $key;
 | 
			
		||||
 | 
			
		||||
    public $email;
 | 
			
		||||
 | 
			
		||||
@@ -20,39 +22,32 @@ class NewEmailForm extends KeyConfirmationForm {
 | 
			
		||||
     */
 | 
			
		||||
    private $account;
 | 
			
		||||
 | 
			
		||||
    public function __construct(Account $account, array $config = []) {
 | 
			
		||||
        $this->account = $account;
 | 
			
		||||
        parent::__construct($config);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function rules() {
 | 
			
		||||
        return array_merge(parent::rules(), [
 | 
			
		||||
        return [
 | 
			
		||||
            ['key', EmailActivationKeyValidator::class, 'type' => EmailActivation::TYPE_CURRENT_EMAIL_CONFIRMATION],
 | 
			
		||||
            ['email', EmailValidator::class],
 | 
			
		||||
        ]);
 | 
			
		||||
        ];
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function getAccount() : Account {
 | 
			
		||||
    public function getAccount(): Account {
 | 
			
		||||
        return $this->account;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function sendNewEmailConfirmation() {
 | 
			
		||||
    public function sendNewEmailConfirmation(): bool {
 | 
			
		||||
        if (!$this->validate()) {
 | 
			
		||||
            return false;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $transaction = Yii::$app->db->beginTransaction();
 | 
			
		||||
        try {
 | 
			
		||||
            $previousActivation = $this->getActivationCodeModel();
 | 
			
		||||
            $previousActivation->delete();
 | 
			
		||||
 | 
			
		||||
            $activation = $this->createCode();
 | 
			
		||||
            $this->sendCode($activation);
 | 
			
		||||
        /** @var \common\models\confirmations\CurrentEmailConfirmation $previousActivation */
 | 
			
		||||
        $previousActivation = $this->key;
 | 
			
		||||
        $previousActivation->delete();
 | 
			
		||||
 | 
			
		||||
            $transaction->commit();
 | 
			
		||||
        } catch (Exception $e) {
 | 
			
		||||
            $transaction->rollBack();
 | 
			
		||||
            throw $e;
 | 
			
		||||
        }
 | 
			
		||||
        $activation = $this->createCode();
 | 
			
		||||
        $this->sendCode($activation);
 | 
			
		||||
 | 
			
		||||
        $transaction->commit();
 | 
			
		||||
 | 
			
		||||
        return true;
 | 
			
		||||
    }
 | 
			
		||||
@@ -98,4 +93,9 @@ class NewEmailForm extends KeyConfirmationForm {
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function __construct(Account $account, array $config = []) {
 | 
			
		||||
        $this->account = $account;
 | 
			
		||||
        parent::__construct($config);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -22,6 +22,8 @@ class EmailActivationKeyValidator extends Validator {
 | 
			
		||||
 | 
			
		||||
    public $expired = E::KEY_EXPIRE;
 | 
			
		||||
 | 
			
		||||
    public $skipOnEmpty = false;
 | 
			
		||||
 | 
			
		||||
    public function validateAttribute($model, $attribute) {
 | 
			
		||||
        $value = $model->$attribute;
 | 
			
		||||
        if (empty($value)) {
 | 
			
		||||
 
 | 
			
		||||
@@ -1,29 +0,0 @@
 | 
			
		||||
<?php
 | 
			
		||||
namespace tests\codeception\api\models\base;
 | 
			
		||||
 | 
			
		||||
use api\models\base\KeyConfirmationForm;
 | 
			
		||||
use Codeception\Specify;
 | 
			
		||||
use common\models\EmailActivation;
 | 
			
		||||
use tests\codeception\api\unit\TestCase;
 | 
			
		||||
use tests\codeception\common\fixtures\EmailActivationFixture;
 | 
			
		||||
 | 
			
		||||
class KeyConfirmationFormTest extends TestCase {
 | 
			
		||||
    use Specify;
 | 
			
		||||
 | 
			
		||||
    public function _fixtures() {
 | 
			
		||||
        return [
 | 
			
		||||
            'emailActivations' => EmailActivationFixture::class,
 | 
			
		||||
        ];
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function testGetActivationCodeModel() {
 | 
			
		||||
        $model = new KeyConfirmationForm();
 | 
			
		||||
        $model->key = $this->tester->grabFixture('emailActivations', 'freshRegistrationConfirmation')['key'];
 | 
			
		||||
        $this->assertInstanceOf(EmailActivation::class, $model->getActivationCodeModel());
 | 
			
		||||
 | 
			
		||||
        $model = new KeyConfirmationForm();
 | 
			
		||||
        $model->key = 'this-is-invalid-key';
 | 
			
		||||
        $this->assertNull($model->getActivationCodeModel());
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
		Reference in New Issue
	
	Block a user