feat: implement multi-tenancy and super admin impersonation with security banner

This commit is contained in:
jeremy bayse
2026-02-21 20:15:47 +01:00
parent a0e904d69d
commit 63e448ef22
31 changed files with 819 additions and 51 deletions

View File

@@ -8,11 +8,14 @@ use Illuminate\Foundation\Auth\User as Authenticatable;
use Illuminate\Notifications\Notifiable;
use Spatie\Permission\Traits\HasRoles;
use App\Traits\BelongsToStructure;
class User extends Authenticatable
{
/** @use HasFactory<\Database\Factories\UserFactory> */
use HasFactory, Notifiable, HasRoles;
use HasFactory, Notifiable, BelongsToStructure;
use HasRoles {
hasRole as traitHasRole;
}
/**
* The attributes that are mass assignable.
@@ -23,6 +26,7 @@ class User extends Authenticatable
'name',
'email',
'password',
'structure_id',
];
/**
@@ -47,4 +51,29 @@ class User extends Authenticatable
'password' => 'hashed',
];
}
/**
* Override de Spatie HasRoles pour qu'un SuperAdmin valide toutes les vérifications de rôle
* Cela permet notamment de parcourir les locataires (Tenant) sans être bloqué par les "hasRole('Admin')"
*/
public function hasRole($roles, string $guard = null): bool
{
// Si on ne demande pas explicitement le rôle SuperAdmin, on vérifie si l'utilisateur l'a globalement.
// On passe par DB::table pour éviter que le GlobalScope 'structure' ne filtre nos propres rôles
// lorsqu'on est en train de simuler une autre structure.
if ($roles !== 'SuperAdmin') {
$isSuperAdmin = \Illuminate\Support\Facades\DB::table('model_has_roles')
->join('roles', 'roles.id', '=', 'model_has_roles.role_id')
->where('model_has_roles.model_id', $this->id)
->where('model_has_roles.model_type', self::class)
->where('roles.name', 'SuperAdmin')
->exists();
if ($isSuperAdmin) {
return true;
}
}
return $this->traitHasRole($roles, $guard);
}
}