revoltify / tenantify
Single database multi-tenant solution for Laravel
Requires
- php: ^8.2
- illuminate/support: ^11.0
Requires (Dev)
- larastan/larastan: ^3.0
- laravel/pint: ^1.20
- orchestra/testbench: ^9.9
- pestphp/pest: ^3.7
- spatie/valuestore: ^1.3
This package is auto-updated.
Last update: 2025-03-25 02:30:56 UTC
README
A powerful and flexible single-database multi-tenant solution for Laravel 11, built with SOLID principles in mind.
Features
- Early Identification: Initialize tenants during application boot for maximum performance
- Flexible Resolution: Support for domain and subdomain-based tenant resolution
- Customizable Bootstrapping: Add your own tenant bootstrapping logic
- Tenant-Aware Systems:
- Queue system with automatic tenant context
- Cache system with tenant-specific prefixes
- Session management with tenant isolation
- Built-in Spatie Permissions Support: Automatic tenant-specific permission caching
- Automatic Tenant Scoping: Zero-effort tenant data isolation with the
BelongsToTenant
trait - High Performance: Optimized resolver with optional caching
- Developer Friendly: Clear API with helper functions
Requirements
- PHP 8.2 or higher
- Laravel 11.x
Installation
- Install the package via Composer:
composer require revoltify/tenantify
- Run the installation command:
php artisan tenantify:install
This will publish the configuration file and migrations.
- Run the migrations:
php artisan migrate
Configuration
The package can be configured via the config/tenantify.php
file. Here are the key configuration options:
Early Initialization
'early' => env('TENANTIFY_EARLY', false),
true
: Initializes tenant during application boot (recommended for fully tenant-aware applications)false
: Manual initialization through middleware (recommended for partially tenant-aware applications)
Custom Models
'models' => [ 'tenant' => \App\Models\Tenant::class, 'domain' => \App\Models\Domain::class, ],
Resolver Configuration
'resolver' => [ 'class' => \Revoltify\Tenantify\Resolvers\DomainResolver::class, 'cache' => [ 'enabled' => env('TENANTIFY_CACHE_ENABLED', false), 'ttl' => env('TENANTIFY_CACHE_TTL', 3600), ], ],
Basic Usage
Creating a Tenant
use Revoltify\Tenantify\Models\Tenant; use Revoltify\Tenantify\Models\Domain; $tenant = Tenant::create([ 'name' => 'Example Organization' ]); $tenant->domains()->create([ 'domain' => 'example.com' ]);
Manual Tenant Initialization
// Using facade Tenantify::initialize($tenant); // Using helper function tenantify()->initialize($tenant); // Initialize by ID tenantify()->initialize($tenantId);
Accessing Current Tenant
// Using helper functions $tenant = tenant(); $tenantId = tenant_id(); // Using facade $tenant = Tenantify::tenant();
Advanced Usage
Custom Bootstrapper
Create a custom bootstrapper by extending the AbstractBootstrapper
class:
use Revoltify\Tenantify\Bootstrappers\AbstractBootstrapper; use Revoltify\Tenantify\Models\Contracts\TenantInterface; class CustomBootstrapper extends AbstractBootstrapper { protected int $priority = 10; public function bootstrap(TenantInterface $tenant): void { // Your bootstrapping logic here } public function revert(): void { // Your cleanup logic here } }
Register your bootstrapper in the configuration:
'bootstrappers' => [ \App\Bootstrappers\CustomBootstrapper::class, ],
Model Tenant Scoping
Add automatic tenant scoping to your models using the BelongsToTenant
trait:
use Illuminate\Database\Eloquent\Model; use Revoltify\Tenantify\Models\Concerns\BelongsToTenant; class Project extends Model { use BelongsToTenant; protected $fillable = ['name', 'description']; }
This will automatically:
- Add tenant ID on model creation
- Scope all queries to the current tenant
- Establish the tenant relationship
Usage example:
// Creates a project for the current tenant automatically $project = Project::create([ 'name' => 'New Project' ]); // Queries are automatically scoped to the current tenant $projects = Project::all(); // Only returns current tenant's projects // Access the tenant relationship $tenant = $project->tenant;
Tenant-Aware Jobs
Make your job tenant-aware by implementing the TenantAware
interface:
use Illuminate\Contracts\Queue\ShouldQueue; use Revoltify\Tenantify\Job\TenantAware; class ProcessTenantData implements ShouldQueue, TenantAware { public function handle(): void { // Job will automatically run in the correct tenant context } }
Contributing
Please see CONTRIBUTING.md for details.
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.