From 5a88229c2dbb5ddeeb7ea04ea0d9ae16049449fb Mon Sep 17 00:00:00 2001 From: ErickSkrauch Date: Thu, 27 Oct 2016 00:06:59 +0300 Subject: [PATCH 1/3] =?UTF-8?q?=D0=A1=D0=BE=D0=B7=D0=B4=D0=B0=D1=91=D0=BC?= =?UTF-8?q?=20=D1=81=D0=BE=D0=B1=D1=8B=D1=82=D0=B8=D0=B5=20=D0=BE=20=D0=BD?= =?UTF-8?q?=D0=BE=D0=B2=D0=BE=D0=BC=20=D0=BD=D0=B8=D0=BA=D0=B5=20=D1=82?= =?UTF-8?q?=D0=BE=D0=BB=D1=8C=D0=BA=D0=BE=20=D0=BF=D1=80=D0=B8=20=D1=83?= =?UTF-8?q?=D1=81=D0=BF=D0=B5=D1=88=D0=BD=D0=BE=D0=B9=20=D1=80=D0=B5=D0=B3?= =?UTF-8?q?=D0=B8=D1=81=D1=82=D1=80=D0=B0=D1=86=D0=B8=D0=B8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- api/models/authentication/ConfirmEmailForm.php | 4 ++++ api/models/authentication/RegistrationForm.php | 4 ---- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/api/models/authentication/ConfirmEmailForm.php b/api/models/authentication/ConfirmEmailForm.php index 0c46680..587ac7d 100644 --- a/api/models/authentication/ConfirmEmailForm.php +++ b/api/models/authentication/ConfirmEmailForm.php @@ -3,6 +3,7 @@ namespace api\models\authentication; use api\models\AccountIdentity; use api\models\base\KeyConfirmationForm; +use api\models\profile\ChangeUsernameForm; use common\models\Account; use common\models\EmailActivation; use Yii; @@ -34,6 +35,9 @@ class ConfirmEmailForm extends KeyConfirmationForm { throw new ErrorException('Unable activate user account.'); } + $changeUsernameForm = new ChangeUsernameForm(); + $changeUsernameForm->createEventTask($account->id, $account->username, null); + $transaction->commit(); } catch (ErrorException $e) { $transaction->rollBack(); diff --git a/api/models/authentication/RegistrationForm.php b/api/models/authentication/RegistrationForm.php index 41e71ee..776da8f 100644 --- a/api/models/authentication/RegistrationForm.php +++ b/api/models/authentication/RegistrationForm.php @@ -4,7 +4,6 @@ namespace api\models\authentication; use api\components\ReCaptcha\Validator as ReCaptchaValidator; use api\models\base\ApiForm; use common\helpers\Error as E; -use api\models\profile\ChangeUsernameForm; use common\components\UserFriendlyRandomKey; use common\models\Account; use common\models\confirmations\RegistrationConfirmation; @@ -118,9 +117,6 @@ class RegistrationForm extends ApiForm { $this->sendMail($emailActivation, $account); - $changeUsernameForm = new ChangeUsernameForm(); - $changeUsernameForm->createEventTask($account->id, $account->username, null); - $transaction->commit(); } catch (Exception $e) { $transaction->rollBack(); From 8049d3c774864d8b7eac5f10cac07770b91bd5dc Mon Sep 17 00:00:00 2001 From: ErickSkrauch Date: Thu, 27 Oct 2016 01:22:39 +0300 Subject: [PATCH 2/3] =?UTF-8?q?=20=D0=A4=D0=B8=D0=BA=D1=81=20=D0=B4=D0=BB?= =?UTF-8?q?=D1=8F=20=D1=84=D0=BE=D1=80=D0=BC=D1=8B=20=D1=81=D0=BC=D0=B5?= =?UTF-8?q?=D0=BD=D1=8B=20=D0=BD=D0=B8=D0=BA=D0=B0=20(=D0=BF=D0=BE=D0=BB?= =?UTF-8?q?=D0=BE=D0=BC=D0=B0=D0=BB=D0=B0=D1=81=D1=8C=20=D0=BB=D0=BE=D0=B3?= =?UTF-8?q?=D0=B8=D0=BA=D0=B0,=20=D0=BA=D0=BE=D0=B3=D0=B4=D0=B0=20=D0=BD?= =?UTF-8?q?=D0=B8=D0=BA=20=D0=BD=D0=B5=20=D0=BC=D0=B5=D0=BD=D1=8F=D0=BB?= =?UTF-8?q?=D1=81=D1=8F)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- api/models/profile/ChangeUsernameForm.php | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/api/models/profile/ChangeUsernameForm.php b/api/models/profile/ChangeUsernameForm.php index ef7f069..81f7957 100644 --- a/api/models/profile/ChangeUsernameForm.php +++ b/api/models/profile/ChangeUsernameForm.php @@ -30,7 +30,10 @@ class ChangeUsernameForm extends ApiForm { public function validateUsername($attribute) { $account = new Account(); + $account->id = $this->getAccount()->id; $account->username = $this->$attribute; + // Это чтобы unique validator учёл, что ник может быть забит на текущий аккаунт + $account->setIsNewRecord(false); if (!$account->validate(['username'])) { $this->addErrors($account->getErrors()); } @@ -46,7 +49,7 @@ class ChangeUsernameForm extends ApiForm { $oldNickname = $account->username; try { $account->username = $this->username; - if (!$account->save()) { + if (!$account->save(false)) { throw new ErrorException('Cannot save account model with new username'); } From 9c8600b2ad4658032077b2a58dcc70797a1a4d3f Mon Sep 17 00:00:00 2001 From: ErickSkrauch Date: Thu, 27 Oct 2016 01:34:51 +0300 Subject: [PATCH 3/3] =?UTF-8?q?=D0=92=20=D0=BF=D1=80=D0=BE=D1=86=D0=B5?= =?UTF-8?q?=D1=81=D1=81=20=D1=80=D0=B5=D0=B3=D0=B8=D1=81=D1=82=D1=80=D0=B0?= =?UTF-8?q?=D1=86=D0=B8=D0=B8=20=D0=B4=D0=BE=D0=B1=D0=B0=D0=B2=D0=BB=D0=B5?= =?UTF-8?q?=D0=BD=D0=B0=20=D0=BF=D0=BE=D0=B4=D0=B4=D0=B5=D1=80=D0=B6=D0=BA?= =?UTF-8?q?=D0=B0=20=D0=BB=D0=BE=D0=B3=D0=B8=D0=BA=D0=B8,=20=D0=BA=D0=BE?= =?UTF-8?q?=D0=B3=D0=B4=D0=B0=20=D0=BC=D0=BE=D0=B6=D0=BD=D0=BE=20=D0=BF?= =?UTF-8?q?=D0=B5=D1=80=D0=B5=D0=B7=D0=B0=D0=BD=D1=8F=D1=82=D1=8C=20=D0=BD?= =?UTF-8?q?=D0=B8=D0=BA=20=D0=B8=D0=BB=D0=B8=20email,=20=D0=BA=D0=BE=D1=82?= =?UTF-8?q?=D0=BE=D1=80=D1=8B=D0=B9=20=D0=B5=D1=89=D1=91=20=D0=BD=D0=B5=20?= =?UTF-8?q?=D0=BE=D0=BA=D0=BE=D0=BD=D1=87=D0=B8=D0=BB=20=D1=80=D0=B5=D0=B3?= =?UTF-8?q?=D0=B8=D1=81=D1=82=D1=80=D0=B0=D1=86=D0=B8=D1=8E.=20=D0=92?= =?UTF-8?q?=D0=B0=D0=BB=D0=B8=D0=B4=D0=B0=D1=82=D0=BE=D1=80=20LanguageVali?= =?UTF-8?q?dator=20=D1=83=D0=B1=D1=80=D0=B0=D0=BD=20=D0=B8=D0=B7=20=D0=BC?= =?UTF-8?q?=D0=BE=D0=B4=D0=B5=D0=BB=D0=B8=20Account?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../authentication/RegistrationForm.php | 45 ++++++++++++++++++- api/models/profile/ChangeLanguageForm.php | 2 +- common/models/Account.php | 4 -- .../api/functional/RegisterCest.php | 35 ++++++++++++++- 4 files changed, 78 insertions(+), 8 deletions(-) diff --git a/api/models/authentication/RegistrationForm.php b/api/models/authentication/RegistrationForm.php index 776da8f..fea1245 100644 --- a/api/models/authentication/RegistrationForm.php +++ b/api/models/authentication/RegistrationForm.php @@ -16,6 +16,7 @@ use Ramsey\Uuid\Uuid; use Yii; use yii\base\ErrorException; use yii\base\InvalidConfigException; +use yii\helpers\ArrayHelper; use const common\LATEST_RULES_VERSION; class RegistrationForm extends ApiForm { @@ -48,6 +49,7 @@ class RegistrationForm extends ApiForm { ['rePassword', 'validatePasswordAndRePasswordMatch'], ['lang', LanguageValidator::class], + ['lang', 'default', 'value' => 'en'], ]; } @@ -80,7 +82,7 @@ class RegistrationForm extends ApiForm { * @throws Exception */ public function signup() { - if (!$this->validate()) { + if (!$this->validate() && !$this->canContinue($this->getFirstErrors())) { return null; } @@ -95,7 +97,7 @@ class RegistrationForm extends ApiForm { $account->status = Account::STATUS_REGISTERED; $account->rules_agreement_version = LATEST_RULES_VERSION; $account->setRegistrationIp(Yii::$app->request->getUserIP()); - if (!$account->save()) { + if (!$account->save(false)) { throw new ErrorException('Account not created.'); } @@ -157,4 +159,43 @@ class RegistrationForm extends ApiForm { } } + /** + * Метод проверяет, можно ли занять указанный при регистрации ник или e-mail. Так случается, + * что пользователи вводят неправильный e-mail или ник, после замечают это и пытаются вновь + * выпонить регистрацию. Мы не будем им мешать и просто удаляем существующие недозарегистрированные + * аккаунты, позволяя им зарегистрироваться. + * + * @param array $errors массив, где ключ - это поле, а значение - первая ошибка из нашего + * стандартного словаря ошибок + * + * @return bool + */ + protected function canContinue(array $errors) : bool { + if (ArrayHelper::getValue($errors, 'username') === E::USERNAME_NOT_AVAILABLE) { + $duplicatedUsername = Account::findOne([ + 'username' => $this->username, + 'status' => Account::STATUS_REGISTERED, + ]); + + if ($duplicatedUsername !== null) { + $duplicatedUsername->delete(); + unset($errors['username']); + } + } + + if (ArrayHelper::getValue($errors, 'email') === E::EMAIL_NOT_AVAILABLE) { + $duplicatedEmail = Account::findOne([ + 'email' => $this->email, + 'status' => Account::STATUS_REGISTERED, + ]); + + if ($duplicatedEmail !== null) { + $duplicatedEmail->delete(); + unset($errors['email']); + } + } + + return empty($errors); + } + } diff --git a/api/models/profile/ChangeLanguageForm.php b/api/models/profile/ChangeLanguageForm.php index fa44d26..391687a 100644 --- a/api/models/profile/ChangeLanguageForm.php +++ b/api/models/profile/ChangeLanguageForm.php @@ -31,7 +31,7 @@ class ChangeLanguageForm extends ApiForm { $account = $this->getAccount(); $account->lang = $this->lang; - if (!$account->save()) { + if (!$account->save(false)) { throw new ErrorException('Cannot change user language'); } diff --git a/common/models/Account.php b/common/models/Account.php index 4227e9d..4c94f08 100644 --- a/common/models/Account.php +++ b/common/models/Account.php @@ -3,7 +3,6 @@ namespace common\models; use common\helpers\Error as E; use common\components\UserPass; -use common\validators\LanguageValidator; use Ely\Yii2\TempmailValidator; use Yii; use yii\base\InvalidConfigException; @@ -79,9 +78,6 @@ class Account extends ActiveRecord { [['email'], 'email', 'checkDNS' => true, 'enableIDN' => true, 'message' => E::EMAIL_INVALID], [['email'], TempmailValidator::class, 'message' => E::EMAIL_IS_TEMPMAIL], [['email'], 'unique', 'message' => E::EMAIL_NOT_AVAILABLE], - - [['lang'], LanguageValidator::class], - [['lang'], 'default', 'value' => 'en'], ]; } diff --git a/tests/codeception/api/functional/RegisterCest.php b/tests/codeception/api/functional/RegisterCest.php index 210afb3..4730d8f 100644 --- a/tests/codeception/api/functional/RegisterCest.php +++ b/tests/codeception/api/functional/RegisterCest.php @@ -1,7 +1,6 @@ true, 'lang' => 'ru', ]); + $this->assertSuccessRegistration($I); + } + + public function testUserCorrectRegistrationWithReassignUsername(FunctionalTester $I) { + $route = new SignupRoute($I); + + $I->wantTo('ensure that signup allow reassign not finished registration username'); + $route->register([ + 'username' => 'howe.garnett', + 'email' => 'custom-email@gmail.com', + 'password' => 'some_password', + 'rePassword' => 'some_password', + 'rulesAgreement' => true, + 'lang' => 'ru', + ]); + $this->assertSuccessRegistration($I); + } + + public function testUserCorrectRegistrationWithReassignEmail(FunctionalTester $I) { + $route = new SignupRoute($I); + + $I->wantTo('ensure that signup allow reassign not finished registration email'); + $route->register([ + 'username' => 'CustomUsername', + 'email' => 'achristiansen@gmail.com', + 'password' => 'some_password', + 'rePassword' => 'some_password', + 'rulesAgreement' => true, + 'lang' => 'ru', + ]); + $this->assertSuccessRegistration($I); + } + + private function assertSuccessRegistration(FunctionalTester $I) { $I->canSeeResponseCodeIs(200); $I->canSeeResponseIsJson(); $I->canSeeResponseContainsJson(['success' => true]);