- Added functionality for users to request an email change, including generating a verification code and setting an expiration time. - Implemented methods in the User model to handle email change requests, code validation, and confirmation. - Created a new job to update the user's email in Stripe after confirmation. - Introduced rate limiting for email change requests and verification attempts to prevent abuse. - Added a new notification for email change verification. - Updated the profile component to manage email change requests and verification UI.
		
			
				
	
	
		
			398 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			398 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						|
 | 
						|
namespace App\Models;
 | 
						|
 | 
						|
use App\Notifications\Channels\SendsEmail;
 | 
						|
use App\Notifications\TransactionalEmails\ResetPassword as TransactionalEmailsResetPassword;
 | 
						|
use App\Traits\DeletesUserSessions;
 | 
						|
use DateTimeInterface;
 | 
						|
use Illuminate\Database\Eloquent\Factories\HasFactory;
 | 
						|
use Illuminate\Foundation\Auth\User as Authenticatable;
 | 
						|
use Illuminate\Notifications\Messages\MailMessage;
 | 
						|
use Illuminate\Notifications\Notifiable;
 | 
						|
use Illuminate\Support\Carbon;
 | 
						|
use Illuminate\Support\Facades\Auth;
 | 
						|
use Illuminate\Support\Facades\Cache;
 | 
						|
use Illuminate\Support\Facades\Config;
 | 
						|
use Illuminate\Support\Facades\URL;
 | 
						|
use Illuminate\Support\Str;
 | 
						|
use Laravel\Fortify\TwoFactorAuthenticatable;
 | 
						|
use Laravel\Sanctum\HasApiTokens;
 | 
						|
use Laravel\Sanctum\NewAccessToken;
 | 
						|
use OpenApi\Attributes as OA;
 | 
						|
 | 
						|
#[OA\Schema(
 | 
						|
    description: 'User model',
 | 
						|
    type: 'object',
 | 
						|
    properties: [
 | 
						|
        'id' => ['type' => 'integer', 'description' => 'The user identifier in the database.'],
 | 
						|
        'name' => ['type' => 'string', 'description' => 'The user name.'],
 | 
						|
        'email' => ['type' => 'string', 'description' => 'The user email.'],
 | 
						|
        'email_verified_at' => ['type' => 'string', 'description' => 'The date when the user email was verified.'],
 | 
						|
        'created_at' => ['type' => 'string', 'description' => 'The date when the user was created.'],
 | 
						|
        'updated_at' => ['type' => 'string', 'description' => 'The date when the user was updated.'],
 | 
						|
        'two_factor_confirmed_at' => ['type' => 'string', 'description' => 'The date when the user two factor was confirmed.'],
 | 
						|
        'force_password_reset' => ['type' => 'boolean', 'description' => 'The flag to force the user to reset the password.'],
 | 
						|
        'marketing_emails' => ['type' => 'boolean', 'description' => 'The flag to receive marketing emails.'],
 | 
						|
    ],
 | 
						|
)]
 | 
						|
class User extends Authenticatable implements SendsEmail
 | 
						|
{
 | 
						|
    use DeletesUserSessions, HasApiTokens, HasFactory, Notifiable, TwoFactorAuthenticatable;
 | 
						|
 | 
						|
    protected $guarded = [];
 | 
						|
 | 
						|
    protected $hidden = [
 | 
						|
        'password',
 | 
						|
        'remember_token',
 | 
						|
        'two_factor_recovery_codes',
 | 
						|
        'two_factor_secret',
 | 
						|
    ];
 | 
						|
 | 
						|
    protected $casts = [
 | 
						|
        'email_verified_at' => 'datetime',
 | 
						|
        'force_password_reset' => 'boolean',
 | 
						|
        'show_boarding' => 'boolean',
 | 
						|
        'email_change_code_expires_at' => 'datetime',
 | 
						|
    ];
 | 
						|
 | 
						|
    protected static function boot()
 | 
						|
    {
 | 
						|
        parent::boot();
 | 
						|
 | 
						|
        static::created(function (User $user) {
 | 
						|
            $team = [
 | 
						|
                'name' => $user->name."'s Team",
 | 
						|
                'personal_team' => true,
 | 
						|
                'show_boarding' => true,
 | 
						|
            ];
 | 
						|
            if ($user->id === 0) {
 | 
						|
                $team['id'] = 0;
 | 
						|
                $team['name'] = 'Root Team';
 | 
						|
            }
 | 
						|
            $new_team = Team::create($team);
 | 
						|
            $user->teams()->attach($new_team, ['role' => 'owner']);
 | 
						|
        });
 | 
						|
 | 
						|
        static::deleting(function (User $user) {
 | 
						|
            \DB::transaction(function () use ($user) {
 | 
						|
                $teams = $user->teams;
 | 
						|
                foreach ($teams as $team) {
 | 
						|
                    $user_alone_in_team = $team->members->count() === 1;
 | 
						|
 | 
						|
                    // Prevent deletion if user is alone in root team
 | 
						|
                    if ($team->id === 0 && $user_alone_in_team) {
 | 
						|
                        throw new \Exception('User is alone in the root team, cannot delete');
 | 
						|
                    }
 | 
						|
 | 
						|
                    if ($user_alone_in_team) {
 | 
						|
                        static::finalizeTeamDeletion($user, $team);
 | 
						|
                        // Delete any pending team invitations for this user
 | 
						|
                        TeamInvitation::whereEmail($user->email)->delete();
 | 
						|
 | 
						|
                        continue;
 | 
						|
                    }
 | 
						|
 | 
						|
                    // Load the user's role for this team
 | 
						|
                    $userRole = $team->members->where('id', $user->id)->first()?->pivot?->role;
 | 
						|
 | 
						|
                    if ($userRole === 'owner') {
 | 
						|
                        $found_other_owner_or_admin = $team->members->filter(function ($member) use ($user) {
 | 
						|
                            return ($member->pivot->role === 'owner' || $member->pivot->role === 'admin') && $member->id !== $user->id;
 | 
						|
                        })->first();
 | 
						|
 | 
						|
                        if ($found_other_owner_or_admin) {
 | 
						|
                            $team->members()->detach($user->id);
 | 
						|
 | 
						|
                            continue;
 | 
						|
                        } else {
 | 
						|
                            $found_other_member_who_is_not_owner = $team->members->filter(function ($member) {
 | 
						|
                                return $member->pivot->role === 'member';
 | 
						|
                            })->first();
 | 
						|
 | 
						|
                            if ($found_other_member_who_is_not_owner) {
 | 
						|
                                $found_other_member_who_is_not_owner->pivot->role = 'owner';
 | 
						|
                                $found_other_member_who_is_not_owner->pivot->save();
 | 
						|
                                $team->members()->detach($user->id);
 | 
						|
                            } else {
 | 
						|
                                static::finalizeTeamDeletion($user, $team);
 | 
						|
                            }
 | 
						|
 | 
						|
                            continue;
 | 
						|
                        }
 | 
						|
                    } else {
 | 
						|
                        $team->members()->detach($user->id);
 | 
						|
                    }
 | 
						|
                }
 | 
						|
            });
 | 
						|
        });
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Finalize team deletion by cleaning up all associated resources
 | 
						|
     */
 | 
						|
    private static function finalizeTeamDeletion(User $user, Team $team)
 | 
						|
    {
 | 
						|
        $servers = $team->servers;
 | 
						|
        foreach ($servers as $server) {
 | 
						|
            $resources = $server->definedResources();
 | 
						|
            foreach ($resources as $resource) {
 | 
						|
                $resource->forceDelete();
 | 
						|
            }
 | 
						|
            $server->forceDelete();
 | 
						|
        }
 | 
						|
 | 
						|
        $projects = $team->projects;
 | 
						|
        foreach ($projects as $project) {
 | 
						|
            $project->forceDelete();
 | 
						|
        }
 | 
						|
 | 
						|
        $team->members()->detach($user->id);
 | 
						|
        $team->delete();
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Delete the user if they are not verified and have a force password reset.
 | 
						|
     * This is used to clean up users that have been invited, did not accept the invitation (and did not verify their email and have a force password reset).
 | 
						|
     */
 | 
						|
    public function deleteIfNotVerifiedAndForcePasswordReset()
 | 
						|
    {
 | 
						|
        if ($this->hasVerifiedEmail() === false && $this->force_password_reset === true) {
 | 
						|
            $this->delete();
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    public function recreate_personal_team()
 | 
						|
    {
 | 
						|
        $team = [
 | 
						|
            'name' => $this->name."'s Team",
 | 
						|
            'personal_team' => true,
 | 
						|
            'show_boarding' => true,
 | 
						|
        ];
 | 
						|
        if ($this->id === 0) {
 | 
						|
            $team['id'] = 0;
 | 
						|
            $team['name'] = 'Root Team';
 | 
						|
        }
 | 
						|
        $new_team = Team::create($team);
 | 
						|
        $this->teams()->attach($new_team, ['role' => 'owner']);
 | 
						|
 | 
						|
        return $new_team;
 | 
						|
    }
 | 
						|
 | 
						|
    public function createToken(string $name, array $abilities = ['*'], ?DateTimeInterface $expiresAt = null)
 | 
						|
    {
 | 
						|
        $plainTextToken = sprintf(
 | 
						|
            '%s%s%s',
 | 
						|
            config('sanctum.token_prefix', ''),
 | 
						|
            $tokenEntropy = Str::random(40),
 | 
						|
            hash('crc32b', $tokenEntropy)
 | 
						|
        );
 | 
						|
 | 
						|
        $token = $this->tokens()->create([
 | 
						|
            'name' => $name,
 | 
						|
            'token' => hash('sha256', $plainTextToken),
 | 
						|
            'abilities' => $abilities,
 | 
						|
            'expires_at' => $expiresAt,
 | 
						|
            'team_id' => session('currentTeam')->id,
 | 
						|
        ]);
 | 
						|
 | 
						|
        return new NewAccessToken($token, $token->getKey().'|'.$plainTextToken);
 | 
						|
    }
 | 
						|
 | 
						|
    public function teams()
 | 
						|
    {
 | 
						|
        return $this->belongsToMany(Team::class)->withPivot('role');
 | 
						|
    }
 | 
						|
 | 
						|
    public function changelogReads()
 | 
						|
    {
 | 
						|
        return $this->hasMany(UserChangelogRead::class);
 | 
						|
    }
 | 
						|
 | 
						|
    public function getUnreadChangelogCount(): int
 | 
						|
    {
 | 
						|
        return app(\App\Services\ChangelogService::class)->getUnreadCountForUser($this);
 | 
						|
    }
 | 
						|
 | 
						|
    public function getRecipients(): array
 | 
						|
    {
 | 
						|
        return [$this->email];
 | 
						|
    }
 | 
						|
 | 
						|
    public function sendVerificationEmail()
 | 
						|
    {
 | 
						|
        $mail = new MailMessage;
 | 
						|
        $url = Url::temporarySignedRoute(
 | 
						|
            'verify.verify',
 | 
						|
            Carbon::now()->addMinutes(Config::get('auth.verification.expire', 60)),
 | 
						|
            [
 | 
						|
                'id' => $this->getKey(),
 | 
						|
                'hash' => sha1($this->getEmailForVerification()),
 | 
						|
            ]
 | 
						|
        );
 | 
						|
        $mail->view('emails.email-verification', [
 | 
						|
            'url' => $url,
 | 
						|
        ]);
 | 
						|
        $mail->subject('Coolify: Verify your email.');
 | 
						|
        send_user_an_email($mail, $this->email);
 | 
						|
    }
 | 
						|
 | 
						|
    public function sendPasswordResetNotification($token): void
 | 
						|
    {
 | 
						|
        $this?->notify(new TransactionalEmailsResetPassword($token));
 | 
						|
    }
 | 
						|
 | 
						|
    public function isAdmin()
 | 
						|
    {
 | 
						|
        return $this->role() === 'admin' || $this->role() === 'owner';
 | 
						|
    }
 | 
						|
 | 
						|
    public function isOwner()
 | 
						|
    {
 | 
						|
        return $this->role() === 'owner';
 | 
						|
    }
 | 
						|
 | 
						|
    public function isMember()
 | 
						|
    {
 | 
						|
        return $this->role() === 'member';
 | 
						|
    }
 | 
						|
 | 
						|
    public function isAdminFromSession()
 | 
						|
    {
 | 
						|
        if (Auth::id() === 0) {
 | 
						|
            return true;
 | 
						|
        }
 | 
						|
        $teams = $this->teams()->get();
 | 
						|
 | 
						|
        $is_part_of_root_team = $teams->where('id', 0)->first();
 | 
						|
        $is_admin_of_root_team = $is_part_of_root_team &&
 | 
						|
            ($is_part_of_root_team->pivot->role === 'admin' || $is_part_of_root_team->pivot->role === 'owner');
 | 
						|
 | 
						|
        if ($is_part_of_root_team && $is_admin_of_root_team) {
 | 
						|
            return true;
 | 
						|
        }
 | 
						|
        $team = $teams->where('id', session('currentTeam')->id)->first();
 | 
						|
        $role = data_get($team, 'pivot.role');
 | 
						|
 | 
						|
        return $role === 'admin' || $role === 'owner';
 | 
						|
    }
 | 
						|
 | 
						|
    public function isInstanceAdmin()
 | 
						|
    {
 | 
						|
        $found_root_team = Auth::user()->teams->filter(function ($team) {
 | 
						|
            if ($team->id == 0) {
 | 
						|
                if (! Auth::user()->isAdmin()) {
 | 
						|
                    return false;
 | 
						|
                }
 | 
						|
 | 
						|
                return true;
 | 
						|
            }
 | 
						|
 | 
						|
            return false;
 | 
						|
        });
 | 
						|
 | 
						|
        return $found_root_team->count() > 0;
 | 
						|
    }
 | 
						|
 | 
						|
    public function currentTeam()
 | 
						|
    {
 | 
						|
        return Cache::remember('team:'.Auth::id(), 3600, function () {
 | 
						|
            if (is_null(data_get(session('currentTeam'), 'id')) && Auth::user()->teams->count() > 0) {
 | 
						|
                return Auth::user()->teams[0];
 | 
						|
            }
 | 
						|
 | 
						|
            return Team::find(session('currentTeam')->id);
 | 
						|
        });
 | 
						|
    }
 | 
						|
 | 
						|
    public function otherTeams()
 | 
						|
    {
 | 
						|
        return Auth::user()->teams->filter(function ($team) {
 | 
						|
            return $team->id != currentTeam()->id;
 | 
						|
        });
 | 
						|
    }
 | 
						|
 | 
						|
    public function role()
 | 
						|
    {
 | 
						|
        if (data_get($this, 'pivot')) {
 | 
						|
            return $this->pivot->role;
 | 
						|
        }
 | 
						|
        $user = Auth::user()->teams->where('id', currentTeam()->id)->first();
 | 
						|
 | 
						|
        return data_get($user, 'pivot.role');
 | 
						|
    }
 | 
						|
 | 
						|
    public function requestEmailChange(string $newEmail): void
 | 
						|
    {
 | 
						|
        // Generate 6-digit code
 | 
						|
        $code = sprintf('%06d', mt_rand(0, 999999));
 | 
						|
 | 
						|
        // Set expiration using config value
 | 
						|
        $expiryMinutes = config('constants.email_change.verification_code_expiry_minutes', 10);
 | 
						|
        $expiresAt = Carbon::now()->addMinutes($expiryMinutes);
 | 
						|
 | 
						|
        $this->update([
 | 
						|
            'pending_email' => $newEmail,
 | 
						|
            'email_change_code' => $code,
 | 
						|
            'email_change_code_expires_at' => $expiresAt,
 | 
						|
        ]);
 | 
						|
 | 
						|
        // Send verification email to new address
 | 
						|
        $this->notify(new \App\Notifications\TransactionalEmails\EmailChangeVerification($this, $code, $newEmail, $expiresAt));
 | 
						|
    }
 | 
						|
 | 
						|
    public function isEmailChangeCodeValid(string $code): bool
 | 
						|
    {
 | 
						|
        return $this->email_change_code === $code
 | 
						|
            && $this->email_change_code_expires_at
 | 
						|
            && Carbon::now()->lessThan($this->email_change_code_expires_at);
 | 
						|
    }
 | 
						|
 | 
						|
    public function confirmEmailChange(string $code): bool
 | 
						|
    {
 | 
						|
        if (! $this->isEmailChangeCodeValid($code)) {
 | 
						|
            return false;
 | 
						|
        }
 | 
						|
 | 
						|
        $oldEmail = $this->email;
 | 
						|
        $newEmail = $this->pending_email;
 | 
						|
 | 
						|
        // Update email and clear change request fields
 | 
						|
        $this->update([
 | 
						|
            'email' => $newEmail,
 | 
						|
            'pending_email' => null,
 | 
						|
            'email_change_code' => null,
 | 
						|
            'email_change_code_expires_at' => null,
 | 
						|
        ]);
 | 
						|
 | 
						|
        // For cloud users, dispatch job to update Stripe customer email asynchronously
 | 
						|
        if (isCloud() && $this->currentTeam()->subscription) {
 | 
						|
            dispatch(new \App\Jobs\UpdateStripeCustomerEmailJob(
 | 
						|
                $this->currentTeam(),
 | 
						|
                $this->id,
 | 
						|
                $newEmail,
 | 
						|
                $oldEmail
 | 
						|
            ));
 | 
						|
        }
 | 
						|
 | 
						|
        return true;
 | 
						|
    }
 | 
						|
 | 
						|
    public function clearEmailChangeRequest(): void
 | 
						|
    {
 | 
						|
        $this->update([
 | 
						|
            'pending_email' => null,
 | 
						|
            'email_change_code' => null,
 | 
						|
            'email_change_code_expires_at' => null,
 | 
						|
        ]);
 | 
						|
    }
 | 
						|
 | 
						|
    public function hasEmailChangeRequest(): bool
 | 
						|
    {
 | 
						|
        return ! is_null($this->pending_email)
 | 
						|
            && ! is_null($this->email_change_code)
 | 
						|
            && $this->email_change_code_expires_at
 | 
						|
            && Carbon::now()->lessThan($this->email_change_code_expires_at);
 | 
						|
    }
 | 
						|
}
 |