verseles / possession
Laravel user impersonation package with Sanctum compatibility
Installs: 1 033
Dependents: 0
Suggesters: 0
Security: 0
Stars: 4
Watchers: 1
Forks: 0
Open Issues: 0
pkg:composer/verseles/possession
Requires
- php: ^8.2
- laravel/framework: ^9.0|^10.0|^11.0|^12.0
Requires (Dev)
- orchestra/testbench: ^8.0|^9.0|^10.0
- dev-main
- 0.0.11
- 0.0.10
- 0.0.9
- 0.0.8
- 0.0.7
- 0.0.6
- 0.0.5
- 0.0.4
- 0.0.3
- 0.0.2
- 0.0.1
- dev-feature/multi-guard-impersonation-support-3546414680537160352
- dev-fix-unauthenticated-possession-4524390713327178929
- dev-fix-unauthenticated-possession-crash-15692083501152304210
- dev-improve-possession-guard-support-14120053163643008262
- dev-guard-possession-support-14034254722377177042
This package is auto-updated.
Last update: 2026-02-16 23:12:44 UTC
README
A simple user impersonation package for Laravel with Sanctum compatibility.
Features
- Secure user impersonation system
- Global
possess()andunpossess()methods - Comprehensive exception handling
- Session-based impersonation
- Sanctum compatibility
- Simple administration controls
- Visual impersonation indicator
- Easy to integrate
It wasn't tested with all Laravel versions, only with Laravel 11 and 12.
Installation
- Install via Composer:
composer require verseles/possession
- Publish the configuration file (optional):
php artisan vendor:publish --tag=possession-config
- Add the trait to your User model:
use Verseles\Possession\Traits\ImpersonatesUsers; class User extends Authenticatable { use ImpersonatesUsers; // Add your possession logic public function canPossess() { return $this->is_admin; // Your admin check logic } public function canBePossessed() { return !$this->is_admin; // Example restriction } }
Usage
Global Methods
Use the facade for direct impersonation control:
use Verseles\Possession\Facades\Possession; // Possess a user (accepts ID, email, or User model instance) try { Possession::possess($targetUser); } catch (\Verseles\Possession\Exceptions\ImpersonationException $e) { // Handle exception return redirect()->back()->withErrors(['impersonation' => $e->getMessage()]); } // Stop possessing try { Possession::unpossess(); } catch (\Verseles\Possession\Exceptions\ImpersonationException $e) { // Handle exception return redirect()->back()->withErrors(['impersonation' => $e->getMessage()]); }
Web Routes Example
// routes/web.php use Verseles\Possession\Facades\Possession; Route::middleware(['web', 'auth:'.config('possession.admin_guard')])->group(function () { Route::post('/possession/impersonate/{user}', function ($user) { try { Possession::possess($user); return redirect()->route('dashboard'); } catch (\Verseles\Possession\Exceptions\ImpersonationException $e) { return back()->withErrors(['impersonation' => $e->getMessage()]); } }); Route::post('/possession/leave', function () { try { Possession::unpossess(); return redirect()->route('admin.dashboard'); } catch (\Verseles\Possession\Exceptions\ImpersonationException $e) { return back()->withErrors(['impersonation' => $e->getMessage()]); } }); });
Blade Templates
Start Impersonation:
@if(auth()->check() && auth()->user()->canImpersonate()) <form action="{{ route('possession.impersonate', $user->id) }}" method="POST"> @csrf <button type="submit">Impersonate User</button> </form> @endif
Stop Impersonation:
@include('possession::impersonating')
Error Handling
The package throws specific exceptions you can catch:
use Verseles\Possession\Exceptions\ImpersonationException; try { Possession::possess($user); } catch (ImpersonationException $e) { // Handle specific error types: if ($e->getCode() === 403) { // Authorization error } // General error handling return back()->withErrors(['impersonation' => $e->getMessage()]); }
Available exceptions:
UnauthorizedPossessExceptionUnauthorizedUnpossessExceptionTargetCannotBePossessedExceptionSelfPossessionExceptionNoImpersonationActiveExceptionAlreadyImpersonatingExceptionAdminNotFoundException— original admin not found during unpossess (e.g., deleted)
Configuration
Edit config/possession.php after publishing:
return [ 'user_model' => App\Models\User::class, 'admin_guard' => 'web', 'session_keys' => [ 'original_user' => 'possession.original_user_id', ], ];
Security Considerations
- Always protect impersonation routes with middleware
- Use separate guards for admin and regular users
- Regularly review your
canPossess()andcanBePossessed()logic - Never expose impersonation controls to non-administrative users
License
The MIT License (MIT)