225 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			225 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| namespace App\Models;
 | |
| 
 | |
| use App\Notifications\Channels\SendsEmail;
 | |
| use App\Notifications\TransactionalEmails\ResetPassword as TransactionalEmailsResetPassword;
 | |
| 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 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',
 | |
|     ];
 | |
| 
 | |
|     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']);
 | |
|         });
 | |
|     }
 | |
| 
 | |
|     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 getRecepients($notification)
 | |
|     {
 | |
|         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');
 | |
|     }
 | |
| }
 | 
