worldesports/laravel-auto-tenancy

Laravel package for post-authentication multi-tenancy with per-tenant database isolation and automatic connection switching

Maintainers

Package info

github.com/World-eSports-LLC/laravel-auto-tenancy

pkg:composer/worldesports/laravel-auto-tenancy

Fund package maintenance!

World-eSports-LLC

Statistics

Installs: 0

Dependents: 0

Suggesters: 0

Stars: 31

Open Issues: 3

v1.0.0 2026-03-30 22:03 UTC

README

Post-authentication multi-tenancy for Laravel applications with runtime tenant connection resolution and automatic database switching after login.

Latest Version on Packagist GitHub Tests Action Status GitHub Code Style Action Status PHPStan Total Downloads

What makes this package different?

Most Laravel multi-tenancy packages are commonly used in domain-first or subdomain-first workflows, where the tenant is identified from the request before or independently of user authentication.

worldesports/laravel-auto-tenancy is built for a different workflow:

  1. The user logs in or registers first.
  2. The application determines the tenant from the authenticated user.
  3. The package chooses or builds the correct database connection at runtime for that tenant.
  4. The application is automatically switched into the correct tenant database/context.

This makes it a strong fit for Laravel applications where tenancy is resolved after authentication rather than solely from the incoming request host.

Example use case

Your app has a shared login screen for all users.

After authentication:

  • alice@acme.com should use the ACME tenant database
  • bob@globex.com should use the Globex tenant database
  • tenant context is determined from the authenticated user
  • the package automatically switches the application into the correct tenant context

Good Fit for This Package

Use this package if your app needs to:

  • resolve tenancy after login or registration
  • determine the tenant from the authenticated user
  • choose or build the correct tenant database connection at runtime
  • switch tenant databases automatically
  • add tenancy to an existing Laravel auth flow without redesigning the app around domain-first tenancy
  • integrate with existing multi-tenant systems

When This May Not Be the Right Fit

If your application is primarily domain-first or subdomain-first and you need broader tenant bootstrapping beyond authenticated-user-driven database switching, another tenancy approach may be a better fit.

Features

  • Post-authentication multi-tenancy: Tenants are automatically detected and switched after user login
  • Runtime connection resolution: The package chooses or builds the correct tenant connection for the authenticated user
  • Multiple database support: Each tenant can use a separate database connection
  • Minimal configuration: Install quickly with sensible defaults
  • Automatic tenant switching: Middleware automatically resolves and switches tenant context
  • Model scoping: Traits for tenant-aware model queries
  • Database isolation: Complete separation between tenant data
  • Multi-driver support: MySQL, PostgreSQL, SQLite, and SQL Server
  • Connection caching: Optimized database connection management
  • Encryption support: Optional encryption for sensitive connection details
  • Comprehensive commands: Full set of Artisan commands for tenant management
  • Event listeners: Automatic tenant creation and cleanup
  • Security features: Access control and validation
  • Testing suite: Comprehensive test coverage

Installation

You can install the package via composer:

composer require worldesports/laravel-auto-tenancy

Quick installation with setup:

php artisan tenant:install --force --migrate

If authentication is missing, the installer will prompt you to install Jetstream:

composer require laravel/jetstream
php artisan jetstream:install livewire
npm install && npm run build
php artisan migrate

Or manual installation:

# Publish and run the migrations
php artisan vendor:publish --tag="multi-tenancy-migrations"
php artisan migrate

# Optionally, publish the config file
php artisan vendor:publish --tag="multi-tenancy-config"

The published config file (config/multi-tenancy.php) contains:

<?php

return [
    // Your User model - automatically detects the model used by your auth system
    'user_model' => env('MULTI_TENANT_USER_MODEL', 'App\\Models\\User'),
    
    // Main database connection
    'main_connection' => env('DB_CONNECTION', 'mysql'),
    
    // Auto-create tenant on user registration (optional)
    'auto_create_tenant' => env('MULTI_TENANT_AUTO_CREATE', false),
    
    // Performance optimizations
    'cache_connections' => env('MULTI_TENANT_CACHE_CONNECTIONS', true),
    
    // Security features
    'encrypt_connection_details' => env('MULTI_TENANT_ENCRYPT', false),
    
    // ... more options
];

Basic Usage

1. User Model Configuration

The package automatically works with your existing User model. If you're using a custom User model or different namespace:

// In config/multi-tenancy.php
'user_model' => App\Models\CustomUser::class,

// Or via environment variable
MULTI_TENANT_USER_MODEL="App\\Models\\CustomUser"

2. Authentication System Compatibility

Laravel Breeze

# Standard Laravel Breeze setup
composer require laravel/breeze
php artisan breeze:install
npm install && npm run dev
php artisan migrate

# Then install multi-tenancy
composer require worldesports/laravel-auto-tenancy
php artisan tenant:install --migrate

Laravel Jetstream

# Standard Jetstream setup
composer require laravel/jetstream
php artisan jetstream:install livewire
npm install && npm run build
php artisan migrate

# Then install multi-tenancy
composer require worldesports/laravel-auto-tenancy
php artisan tenant:install --migrate

Laravel Sanctum API

# For API-based applications
composer require laravel/sanctum
php artisan vendor:publish --provider="Laravel\Sanctum\SanctumServiceProvider"
php artisan migrate

# Then install multi-tenancy
composer require worldesports/laravel-auto-tenancy
php artisan tenant:install --migrate

Social Authentication

# With Laravel Socialite
composer require laravel/socialite
# Configure OAuth providers in config/services.php

# Then install multi-tenancy
composer require worldesports/laravel-auto-tenancy
php artisan tenant:install --migrate

Setup and Usage

1. Register the middleware (optional)

If you want to manually control when tenant switching happens, add the middleware to your routes:

// In routes/web.php or routes/api.php
Route::middleware(['auth', 'tenant'])->group(function () {
    // Your tenant-aware routes here
});

2. Add traits to your models

For models that should be automatically scoped to the current tenant database:

<?php

namespace App\Models;

use Illuminate\Database\Eloquent\Model;
use Worldesports\MultiTenancy\Traits\BelongsToTenant;

class Post extends Model
{
    use BelongsToTenant; // Automatically uses tenant database
    
    // Your model code...
}

For models that have a tenant_id column and need tenant-based scoping:

<?php

namespace App\Models;

use Illuminate\Database\Eloquent\Model;
use Worldesports\MultiTenancy\Traits\TenantScoped;

class Document extends Model
{
    use TenantScoped; // Automatically scopes by tenant_id
    
    protected $fillable = ['title', 'content', 'tenant_id'];
}

Usage

Package Status and Management

# Show overall status
php artisan tenant:status

# Show detailed tenant list
php artisan tenant:status --list

# Show specific tenant details
php artisan tenant:status --tenant=1

# Test all database connections
php artisan tenant:status --connections

# Test connections for one driver only
php artisan tenant:status --connections --driver=mysql
php artisan tenant:status --connections --driver=pgsql
php artisan tenant:status --connections --driver=sqlite
php artisan tenant:status --connections --driver=sqlsrv

Creating Tenants

Use the artisan command to create a new tenant. The package supports all major database drivers:

MySQL Tenant (Default)

php artisan tenant:create 1 "MySQL Company" \
  --db-name=tenant_mysql_db \
  --db-driver=mysql \
  --db-host=127.0.0.1 \
  --db-port=3306 \
  --db-username=mysql_user \
  --db-password=secret \
  --create-db

PostgreSQL Tenant

php artisan tenant:create 2 "PostgreSQL Company" \
  --db-name=tenant_postgres_db \
  --db-driver=pgsql \
  --db-host=127.0.0.1 \
  --db-port=5432 \
  --db-username=postgres_user \
  --db-password=secret \
  --create-db

SQLite Tenant

php artisan tenant:create 3 "SQLite Company" \
  --db-name=/path/to/tenant.sqlite \
  --db-driver=sqlite
  # Note: SQLite doesn't need username/password/host/port

SQL Server Tenant

php artisan tenant:create 4 "SQL Server Company" \
  --db-name=tenant_sqlserver_db \
  --db-driver=sqlsrv \
  --db-host=127.0.0.1 \
  --db-port=1433 \
  --db-username=sa \
  --db-password=secret \
  --create-db

Multi-Driver Support

Your Laravel application can have tenants using different database drivers simultaneously:

  • Tenant 1: Uses MySQL on server A
  • Tenant 2: Uses PostgreSQL on server B
  • Tenant 3: Uses SQLite local file
  • Tenant 4: Uses SQL Server on server C

The package automatically handles driver-specific configurations and optimizations.

Database Management

# Run migrations on all tenant databases
php artisan tenant:migrate

# Run migrations on specific tenant
php artisan tenant:migrate --tenant=1

# Run migrations on specific database
php artisan tenant:migrate --database=1

# Fresh migrations with seeding
php artisan tenant:migrate --fresh --seed

# Seed tenant databases
php artisan tenant:seed

# Seed with specific seeder
php artisan tenant:seed --class=UserSeeder

# Test all tenant database connections
php artisan tenant:status --connections

# Test connections for one driver only
php artisan tenant:status --connections --driver=mysql
php artisan tenant:status --connections --driver=pgsql
php artisan tenant:status --connections --driver=sqlite
php artisan tenant:status --connections --driver=sqlsrv

Tenant Cleanup

# Cleanup tenant (keeps database)
php artisan tenant:cleanup 1

# Cleanup and drop database (DANGEROUS!)
php artisan tenant:cleanup 1 --drop-database

# Skip confirmation
php artisan tenant:cleanup 1 --drop-database --force

Working with Tenants in Code

use Worldesports\MultiTenancy\Facades\MultiTenancy;
use Worldesports\MultiTenancy\Models\Tenant;
use Worldesports\MultiTenancy\Models\TenantDatabase;

// Get current tenant
$tenant = MultiTenancy::getTenant();

// Manually set a tenant
$tenant = Tenant::find(1);
MultiTenancy::setTenant($tenant);

// Manually set a tenant and pick a specific tenant database
$database = TenantDatabase::find(5);
MultiTenancy::useDatabase($database); // switches default connection to this DB

// Or pass a database ID when setting the tenant (falls back to primary/first if null)
MultiTenancy::setTenant($tenant, $databaseId = 5);

// Check if tenant is set
if (MultiTenancy::hasTenant()) {
    // Tenant is active, queries will use tenant database
}

// Switch back to main connection
MultiTenancy::switchToMainConnection();

// Reset tenant context completely
MultiTenancy::resetTenant();

// Purge all cached connections
MultiTenancy::purgeConnections();

Querying for a specific tenant (and database) without changing the global context

// Scope a model to a tenant (uses its primary DB)
Invoice::forTenant($tenantId = 10)->get();

// Scope a model to a specific tenant *database* without switching the app default
Invoice::forTenant($tenantId = 10, $databaseId = 22)->get();

Note: The package keeps one active tenant database at a time per request/context. You can pick which tenant DB to use, but queries execute against a single selected database, not multiple concurrently.

Automatic Tenant Detection

The package provides multiple automatic tenant detection strategies - no manual user mapping required!

When a user logs in, the package tries these detection methods in order:

🎯 Strategy 1: Email Domain Detection

Users are automatically mapped to tenants based on their email domain:

# Create tenant for a company
php artisan tenant:create 1 "ACME Corporation" --domain=acme.com

# When user registers with alice@acme.com, they're automatically assigned to ACME tenant
# When user registers with bob@acme.com, they're also assigned to ACME tenant

Configuration:

MULTI_TENANT_AUTO_DETECT_EMAIL=true  # Default: enabled

🎯 Strategy 2: Subdomain Detection

Tenants detected by subdomain in the URL:

# Create tenant with subdomain
php artisan tenant:create 2 "Client Portal" --subdomain=client1

# When users visit client1.yourapp.com, they automatically use Client Portal tenant
# When users visit client2.yourapp.com, they use a different tenant

Configuration:

MULTI_TENANT_AUTO_DETECT_SUBDOMAIN=true  # Default: disabled

🎯 Strategy 3: Auto-Create Tenants

Automatically create tenants for new users:

# No tenant setup required - tenants created automatically on first login

Configuration:

MULTI_TENANT_AUTO_CREATE=true       # Auto-create tenants
MULTI_TENANT_AUTO_CREATE_DB=true    # Also create databases automatically

🎯 Strategy 4: Manual Mapping (Fallback)

Direct user-to-tenant assignment:

php artisan tenant:create 123 "Manual Tenant"  # User ID 123 gets this tenant

The beauty: You can mix and match these strategies!

Using the Middleware

The SetTenant middleware automatically resolves the tenant for authenticated users:

// The middleware is automatically applied after user authentication
// You can also manually apply it to specific routes
Route::middleware(['auth', \Worldesports\MultiTenancy\Middleware\SetTenant::class])
    ->get('/dashboard', [DashboardController::class, 'index']);

// With error handling options
Route::middleware(['auth', \Worldesports\MultiTenancy\Middleware\SetTenant::class . ':error'])
    ->get('/api/data', [ApiController::class, 'index']);

// Require a tenant to be present (403s if missing)
Route::middleware(['auth', 'tenant', 'tenant.required'])
    ->group(function () {
        Route::get('/account', [AccountController::class, 'show']);
    });

Middleware options:

  • redirect (default): Redirect with error message
  • error: Return JSON error response
  • ignore: Log error but continue

Advanced Configuration

Environment Variables

# Auto-create tenant on user registration
MULTI_TENANT_AUTO_CREATE=true

# Cache database connections for performance
MULTI_TENANT_CACHE_CONNECTIONS=true

# Encrypt connection details in database
MULTI_TENANT_ENCRYPT=true

Event Listeners

The package automatically registers these event listeners:

// Auto-set tenant on user login
Event::listen(Login::class, SetTenantOnLogin::class);

// Auto-create tenant on user registration (optional)
Event::listen(Registered::class, CreateTenantOnRegistration::class);

Security Features

// Check if user has access to tenant
if (!MultiTenancy::userHasAccessToTenant($user, $tenant)) {
    throw new UnauthorizedException('Access denied');
}

// Sanitized connection details (excludes sensitive info)
$safeDetails = $tenantDatabase->safe_connection_details;

Model Scoping Examples

// Using BelongsToTenant trait
class Invoice extends Model
{
    use BelongsToTenant;
    
    // Automatically queries tenant database
    public static function recent()
    {
        return static::where('created_at', '>', now()->subDays(30))->get();
    }
}

// Using TenantScoped trait (for models with tenant_id)
class Order extends Model
{
    use TenantScoped;
    
    // Automatically scoped to current tenant
    public function scopeUnpaid($query)
    {
        return $query->where('paid', false);
    }
    
    // Bypass tenant scoping when needed
    public static function allTenantsOrders()
    {
        return static::withoutTenantScoping()->get();
    }
}

Multi-User & Concurrent Session Support

✅ Multiple Users, Different Tenants

Each user session maintains its own tenant context:

// User A logs in (alice@acme.com) → automatically uses ACME tenant database
// User B logs in (bob@widget.com) → automatically uses Widget tenant database  
// User C logs in (carol@acme.com) → automatically uses ACME tenant database

// All three users can be using the app simultaneously with different tenant databases!

✅ Session Isolation

  • Each user's session maintains independent tenant context
  • No interference between concurrent users
  • Thread-safe tenant switching
  • Automatic cleanup on logout

✅ API & Web Support

// Web users (sessions)
Route::middleware(['auth:web', SetTenant::class])->group(function () {
    // Each web session gets its tenant context
});

// API users (tokens) 
Route::middleware(['auth:sanctum', SetTenant::class])->group(function () {
    // Each API request gets tenant context based on the authenticated user
});

Example 1: Social Media Authentication with Multi-Tenancy

// In your SocialAuthController (using Laravel Socialite)
class SocialAuthController extends Controller
{
    public function redirectToProvider($provider)
    {
        return Socialite::driver($provider)->redirect();
    }

    public function handleProviderCallback($provider)
    {
        $socialUser = Socialite::driver($provider)->user();
        
        // Find or create user
        $user = User::firstOrCreate(
            ['email' => $socialUser->getEmail()],
            [
                'name' => $socialUser->getName(),
                'password' => bcrypt(Str::random(16)), // Random password for social users
            ]
        );

        // Log the user in
        Auth::login($user);
        
        // Multi-tenancy package automatically detects and switches tenant!
        // No additional code needed - the package listens to Login event
        
        return redirect()->intended('/dashboard');
    }
}

Example 2: API Authentication with Sanctum

// API route with automatic tenant switching
Route::middleware(['auth:sanctum'])->group(function () {
    // The SetTenant middleware automatically applies tenant context
    Route::get('/api/tenant-data', function (Request $request) {
        // All queries automatically use the user's tenant database
        $data = SomeModel::all(); // Automatically scoped to tenant
        return response()->json($data);
    });
});

// The package automatically handles tenant switching for API requests

Example 3: Custom Authentication Guard

// If using custom authentication guard
'guards' => [
    'custom' => [
        'driver' => 'session',
        'provider' => 'custom_users',
    ],
],

'providers' => [
    'custom_users' => [
        'driver' => 'eloquent',
        'model' => App\Models\CustomUser::class,
    ],
],

// In your config/multi-tenancy.php
'user_model' => App\Models\CustomUser::class,

// The package works automatically with any authentication guard!

Example 4: Multi-Guard Authentication

// For applications with multiple user types (admin, customer, etc.)
Route::middleware(['auth:web'])->group(function () {
    // Regular customer routes - automatic tenant switching
});

Route::middleware(['auth:admin'])->group(function () {
    // Admin routes - can bypass tenant scoping when needed
    Route::get('/admin/all-tenants', function () {
        return Tenant::withoutGlobalScopes()->get(); // See all tenants
    });
});

Example 5: Automated Tenant Creation for New Users

// Enable auto-tenant creation in config/multi-tenancy.php
'auto_create_tenant' => true,

// Or via environment variable
MULTI_TENANT_AUTO_CREATE=true

// Now when users register (through ANY method), they automatically get a tenant:
// - Social media registration
// - Email/password registration  
// - API registration
// - SSO registration
// - Any custom registration flow

Testing

composer test

Changelog

Please see CHANGELOG for more information on what has changed recently.

Security Vulnerabilities

Please review our security policy on how to report security vulnerabilities.

Credits

License

The MIT License (MIT). Please see License File for more information.