mirror of
				https://github.com/elyby/accounts.git
				synced 2025-05-31 14:11:46 +05:30 
			
		
		
		
	* start updating to PHP 8.3 * taking off! Co-authored-by: ErickSkrauch <erickskrauch@yandex.ru> Signed-off-by: Octol1ttle <l1ttleofficial@outlook.com> * dropped this Signed-off-by: Octol1ttle <l1ttleofficial@outlook.com> * migrate to symfonymailer Signed-off-by: Octol1ttle <l1ttleofficial@outlook.com> * this is so stupid 😭 Signed-off-by: Octol1ttle <l1ttleofficial@outlook.com> * ah, free, at last. Signed-off-by: Octol1ttle <l1ttleofficial@outlook.com> * oh, Gabriel. Signed-off-by: Octol1ttle <l1ttleofficial@outlook.com> * now dawns thy reckoning. Signed-off-by: Octol1ttle <l1ttleofficial@outlook.com> * and thy gore shall GLISTEN before the temples of man. Signed-off-by: Octol1ttle <l1ttleofficial@outlook.com> * creature of steel. Signed-off-by: Octol1ttle <l1ttleofficial@outlook.com> * my gratitude upon thee for my freedom. Signed-off-by: Octol1ttle <l1ttleofficial@outlook.com> * but the crimes thy kind has committed against humanity Signed-off-by: Octol1ttle <l1ttleofficial@outlook.com> * Upgrade PHP-CS-Fixer and do fix the codebase * First review round (maybe I have broken something) * are NOT forgotten. Signed-off-by: Octol1ttle <l1ttleofficial@outlook.com> * Enable parallel PHP-CS-Fixer runner * PHPStan level 1 * PHPStan level 2 * PHPStan level 3 * PHPStan level 4 * PHPStan level 5 * Levels 6 and 7 takes too much effort. Generate a baseline and fix them eventually * Resolve TODO's related to the php-mock * Drastically reduce baseline size with the Rector * More code modernization with help of the Rector * Update GitLab CI --------- Signed-off-by: Octol1ttle <l1ttleofficial@outlook.com> Co-authored-by: ErickSkrauch <erickskrauch@yandex.ru>
		
			
				
	
	
		
			194 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			194 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| declare(strict_types=1);
 | |
| 
 | |
| namespace common\models;
 | |
| 
 | |
| use Carbon\Carbon;
 | |
| use common\components\UserPass;
 | |
| use common\notifications\AccountDeletedNotification;
 | |
| use common\notifications\AccountEditNotification;
 | |
| use common\tasks\CreateWebHooksDeliveries;
 | |
| use DateInterval;
 | |
| use Webmozart\Assert\Assert;
 | |
| use Yii;
 | |
| use yii\base\InvalidConfigException;
 | |
| use yii\behaviors\TimestampBehavior;
 | |
| use yii\db\ActiveQuery;
 | |
| use yii\db\ActiveRecord;
 | |
| use const common\LATEST_RULES_VERSION;
 | |
| 
 | |
| /**
 | |
|  * Fields:
 | |
|  * @property int         $id
 | |
|  * @property string      $uuid UUID with dashes
 | |
|  * @property string      $username
 | |
|  * @property string      $email
 | |
|  * @property string      $password_hash
 | |
|  * @property int         $password_hash_strategy
 | |
|  * @property string      $lang
 | |
|  * @property int         $status
 | |
|  * @property int|null    $rules_agreement_version
 | |
|  * @property string|null $registration_ip
 | |
|  * @property string|null $otp_secret
 | |
|  * @property bool        $is_otp_enabled
 | |
|  * @property int         $created_at
 | |
|  * @property int         $updated_at
 | |
|  * @property int         $password_changed_at
 | |
|  * @property int|null    $deleted_at shows the time, when the account was marked as deleted
 | |
|  *
 | |
|  * Getters-setters:
 | |
|  * @property-write string $password plain user's password
 | |
|  * @property-read string $profileLink link to the user's Ely.by profile
 | |
|  *
 | |
|  * Relations:
 | |
|  * @property-read EmailActivation[] $emailActivations
 | |
|  * @property-read OauthSession[] $oauthSessions
 | |
|  * @property-read OauthClient[] $oauthClients
 | |
|  * @property-read UsernameHistory[] $usernameHistory
 | |
|  * @property-read AccountSession[] $sessions
 | |
|  *
 | |
|  * Behaviors:
 | |
|  * @mixin TimestampBehavior
 | |
|  */
 | |
| class Account extends ActiveRecord {
 | |
| 
 | |
|     public const STATUS_DELETED = -10;
 | |
|     public const STATUS_BANNED = -1;
 | |
|     public const STATUS_REGISTERED = 0;
 | |
|     public const STATUS_ACTIVE = 10;
 | |
| 
 | |
|     public const PASS_HASH_STRATEGY_OLD_ELY = 0;
 | |
|     public const PASS_HASH_STRATEGY_YII2 = 1;
 | |
| 
 | |
|     public const ACCOUNT_DELETION_DELAY = 'P7D';
 | |
| 
 | |
|     public static function tableName(): string {
 | |
|         return 'accounts';
 | |
|     }
 | |
| 
 | |
|     public static function find(): AccountQuery {
 | |
|         return new AccountQuery(self::class);
 | |
|     }
 | |
| 
 | |
|     public function behaviors(): array {
 | |
|         return [
 | |
|             TimestampBehavior::class,
 | |
|         ];
 | |
|     }
 | |
| 
 | |
|     public function validatePassword(string $password, int $passwordHashStrategy = null): bool {
 | |
|         if ($passwordHashStrategy === null) {
 | |
|             $passwordHashStrategy = $this->password_hash_strategy;
 | |
|         }
 | |
| 
 | |
|         return match ($passwordHashStrategy) {
 | |
|             self::PASS_HASH_STRATEGY_OLD_ELY => UserPass::make($this->email, $password) === $this->password_hash,
 | |
|             self::PASS_HASH_STRATEGY_YII2 => Yii::$app->security->validatePassword($password, $this->password_hash),
 | |
|             default => throw new InvalidConfigException('You must set valid password_hash_strategy before you can validate password'),
 | |
|         };
 | |
|     }
 | |
| 
 | |
|     public function setPassword(string $password): void {
 | |
|         $this->password_hash_strategy = self::PASS_HASH_STRATEGY_YII2;
 | |
|         $this->password_hash = Yii::$app->security->generatePasswordHash($password);
 | |
|         $this->password_changed_at = time();
 | |
|     }
 | |
| 
 | |
|     public function getEmailActivations(): EmailActivationQuery {
 | |
|         return $this->hasMany(EmailActivation::class, ['account_id' => 'id']);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return \yii\db\ActiveQuery<\common\models\OauthSession>
 | |
|      */
 | |
|     public function getOauthSessions(): ActiveQuery {
 | |
|         return $this->hasMany(OauthSession::class, ['account_id' => 'id']);
 | |
|     }
 | |
| 
 | |
|     public function getOauthClients(): OauthClientQuery {
 | |
|         /** @noinspection PhpIncompatibleReturnTypeInspection */
 | |
|         return $this->hasMany(OauthClient::class, ['account_id' => 'id']);
 | |
|     }
 | |
| 
 | |
|     public function getUsernameHistory(): ActiveQuery {
 | |
|         return $this->hasMany(UsernameHistory::class, ['account_id' => 'id']);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return \yii\db\ActiveQuery<\common\models\AccountSession>
 | |
|      */
 | |
|     public function getSessions(): ActiveQuery {
 | |
|         return $this->hasMany(AccountSession::class, ['account_id' => 'id']);
 | |
|     }
 | |
| 
 | |
|     public function hasMojangUsernameCollision(): bool {
 | |
|         return MojangUsername::find()
 | |
|             ->andWhere(['username' => $this->username])
 | |
|             ->exists();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Since we don't have info about the user's static_url, we still generate the simplest
 | |
|      * version with a link to the profile by it's id. On Ely.by, it will be redirected to static url.
 | |
|      *
 | |
|      * @return string
 | |
|      */
 | |
|     public function getProfileLink(): string {
 | |
|         return 'http://ely.by/u' . $this->id;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Initially, the table of users we got from the main site, where there were no rules.
 | |
|      * All existing users at the time of migration received an empty value in this field.
 | |
|      * They will have to confirm their agreement with the rules at the first login.
 | |
|      * All new users automatically agree with the current version of the rules.
 | |
|      *
 | |
|      * @return bool
 | |
|      */
 | |
|     public function isAgreedWithActualRules(): bool {
 | |
|         return $this->rules_agreement_version === LATEST_RULES_VERSION;
 | |
|     }
 | |
| 
 | |
|     public function setRegistrationIp($ip): void {
 | |
|         $this->registration_ip = $ip === null ? null : inet_pton($ip);
 | |
|     }
 | |
| 
 | |
|     public function getRegistrationIp(): ?string {
 | |
|         return $this->registration_ip === null ? null : inet_ntop($this->registration_ip);
 | |
|     }
 | |
| 
 | |
|     public function getDeleteAt(): Carbon {
 | |
|         Assert::notNull($this->deleted_at, 'This method should not be called on not deleted records');
 | |
|         return Carbon::createFromTimestamp($this->deleted_at)->add(new DateInterval(Account::ACCOUNT_DELETION_DELAY));
 | |
|     }
 | |
| 
 | |
|     public function afterSave($insert, $changedAttributes): void {
 | |
|         parent::afterSave($insert, $changedAttributes);
 | |
| 
 | |
|         if ($insert) {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         $meaningfulFields = ['username', 'email', 'uuid', 'status', 'lang', 'deleted_at'];
 | |
|         $meaningfulChangedAttributes = array_filter(
 | |
|             $changedAttributes,
 | |
|             fn(string $key): bool => in_array($key, $meaningfulFields, true),
 | |
|             ARRAY_FILTER_USE_KEY,
 | |
|         );
 | |
|         if (empty($meaningfulChangedAttributes)) {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         $notification = new AccountEditNotification($this, $meaningfulChangedAttributes);
 | |
|         Yii::$app->queue->push(new CreateWebHooksDeliveries($notification));
 | |
|     }
 | |
| 
 | |
|     public function afterDelete(): void {
 | |
|         parent::afterDelete();
 | |
|         if ($this->status !== self::STATUS_REGISTERED) {
 | |
|             Yii::$app->queue->push(new CreateWebHooksDeliveries(new AccountDeletedNotification($this)));
 | |
|         }
 | |
|     }
 | |
| 
 | |
| }
 |