2016-05-16 01:33:19 +03:00
|
|
|
|
<?php
|
|
|
|
|
namespace api\models\profile\ChangeEmail;
|
|
|
|
|
|
2016-07-25 14:07:14 +03:00
|
|
|
|
use api\models\base\ApiForm;
|
|
|
|
|
use api\validators\PasswordRequiredValidator;
|
2016-07-17 20:46:04 +03:00
|
|
|
|
use common\helpers\Error as E;
|
2016-05-16 01:33:19 +03:00
|
|
|
|
use common\models\Account;
|
|
|
|
|
use common\models\confirmations\CurrentEmailConfirmation;
|
|
|
|
|
use common\models\EmailActivation;
|
|
|
|
|
use Yii;
|
|
|
|
|
use yii\base\ErrorException;
|
|
|
|
|
use yii\base\Exception;
|
|
|
|
|
use yii\base\InvalidConfigException;
|
|
|
|
|
|
2016-07-25 14:07:14 +03:00
|
|
|
|
class InitStateForm extends ApiForm {
|
2016-05-16 01:33:19 +03:00
|
|
|
|
|
|
|
|
|
public $email;
|
|
|
|
|
|
2016-07-25 14:07:14 +03:00
|
|
|
|
public $password;
|
|
|
|
|
|
2016-05-16 01:33:19 +03:00
|
|
|
|
private $account;
|
|
|
|
|
|
|
|
|
|
public function __construct(Account $account, array $config = []) {
|
|
|
|
|
$this->account = $account;
|
2016-07-17 20:46:04 +03:00
|
|
|
|
$this->email = $account->email;
|
2016-05-16 01:33:19 +03:00
|
|
|
|
parent::__construct($config);
|
|
|
|
|
}
|
|
|
|
|
|
2016-06-16 23:32:23 +03:00
|
|
|
|
public function getAccount() : Account {
|
2016-05-16 01:33:19 +03:00
|
|
|
|
return $this->account;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function rules() {
|
2016-07-25 14:07:14 +03:00
|
|
|
|
return [
|
2016-07-17 20:46:04 +03:00
|
|
|
|
['email', 'validateFrequency'],
|
2016-07-25 14:07:14 +03:00
|
|
|
|
['password', PasswordRequiredValidator::class, 'account' => $this->account],
|
|
|
|
|
];
|
2016-05-16 01:33:19 +03:00
|
|
|
|
}
|
|
|
|
|
|
2016-07-17 20:46:04 +03:00
|
|
|
|
public function validateFrequency($attribute) {
|
|
|
|
|
if (!$this->hasErrors()) {
|
|
|
|
|
$emailConfirmation = $this->getEmailActivation();
|
|
|
|
|
if ($emailConfirmation !== null && !$emailConfirmation->canRepeat()) {
|
|
|
|
|
$this->addError($attribute, E::RECENTLY_SENT_MESSAGE);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-06-16 23:32:23 +03:00
|
|
|
|
public function sendCurrentEmailConfirmation() : bool {
|
2016-05-16 01:33:19 +03:00
|
|
|
|
if (!$this->validate()) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$transaction = Yii::$app->db->beginTransaction();
|
|
|
|
|
try {
|
2016-07-17 20:46:04 +03:00
|
|
|
|
$this->removeOldCode();
|
2016-05-16 01:33:19 +03:00
|
|
|
|
$activation = $this->createCode();
|
|
|
|
|
$this->sendCode($activation);
|
|
|
|
|
|
|
|
|
|
$transaction->commit();
|
|
|
|
|
} catch (Exception $e) {
|
|
|
|
|
$transaction->rollBack();
|
|
|
|
|
throw $e;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @return CurrentEmailConfirmation
|
|
|
|
|
* @throws ErrorException
|
|
|
|
|
*/
|
2016-06-16 23:32:23 +03:00
|
|
|
|
public function createCode() : CurrentEmailConfirmation {
|
2016-05-16 01:33:19 +03:00
|
|
|
|
$account = $this->getAccount();
|
|
|
|
|
$emailActivation = new CurrentEmailConfirmation();
|
|
|
|
|
$emailActivation->account_id = $account->id;
|
|
|
|
|
if (!$emailActivation->save()) {
|
|
|
|
|
throw new ErrorException('Cannot save email activation model');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return $emailActivation;
|
|
|
|
|
}
|
|
|
|
|
|
2016-07-17 20:46:04 +03:00
|
|
|
|
/**
|
|
|
|
|
* Удаляет старый ключ активации, если он существует
|
|
|
|
|
*/
|
|
|
|
|
public function removeOldCode() {
|
|
|
|
|
$emailActivation = $this->getEmailActivation();
|
|
|
|
|
if ($emailActivation === null) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$emailActivation->delete();
|
|
|
|
|
}
|
|
|
|
|
|
2016-05-16 01:33:19 +03:00
|
|
|
|
public function sendCode(EmailActivation $code) {
|
|
|
|
|
$mailer = Yii::$app->mailer;
|
|
|
|
|
$fromEmail = Yii::$app->params['fromEmail'];
|
|
|
|
|
if (!$fromEmail) {
|
|
|
|
|
throw new InvalidConfigException('Please specify fromEmail app in app params');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$acceptor = $code->account;
|
|
|
|
|
$message = $mailer->compose([
|
|
|
|
|
'html' => '@app/mails/current-email-confirmation-html',
|
|
|
|
|
'text' => '@app/mails/current-email-confirmation-text',
|
|
|
|
|
], [
|
|
|
|
|
'key' => $code->key,
|
|
|
|
|
])
|
|
|
|
|
->setTo([$acceptor->email => $acceptor->username])
|
|
|
|
|
->setFrom([$fromEmail => 'Ely.by Accounts'])
|
|
|
|
|
->setSubject('Ely.by Account change E-mail confirmation');
|
|
|
|
|
|
|
|
|
|
if (!$message->send()) {
|
|
|
|
|
throw new ErrorException('Unable send email with activation code.');
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-07-17 20:46:04 +03:00
|
|
|
|
/**
|
|
|
|
|
* Возвращает E-mail активацию, которая использовалась внутри процесса для перехода на следующий шаг.
|
|
|
|
|
* Метод предназначен для проверки, не слишком ли часто отправляются письма о смене E-mail.
|
|
|
|
|
* Проверяем тип подтверждения нового E-mail, поскольку при переходе на этот этап, активация предыдущего
|
|
|
|
|
* шага удаляется.
|
|
|
|
|
* @return EmailActivation|null
|
|
|
|
|
* @throws ErrorException
|
|
|
|
|
*/
|
|
|
|
|
public function getEmailActivation() {
|
|
|
|
|
return $this->getAccount()
|
|
|
|
|
->getEmailActivations()
|
|
|
|
|
->andWhere([
|
|
|
|
|
'type' => [
|
|
|
|
|
EmailActivation::TYPE_CURRENT_EMAIL_CONFIRMATION,
|
|
|
|
|
EmailActivation::TYPE_NEW_EMAIL_CONFIRMATION,
|
|
|
|
|
],
|
|
|
|
|
])
|
|
|
|
|
->one();
|
|
|
|
|
}
|
|
|
|
|
|
2016-05-16 01:33:19 +03:00
|
|
|
|
}
|