DEV Community

Rishi Raj Jain for Neon

Posted on

Creating a Multi-Tenant Application with Laravel and Neon

Multi-tenancy is a software architecture where a single instance of an application serves multiple tenants or clients.

Each tenant's data is isolated and remains invisible to other tenants. This approach is commonly used in Software as a Service (SaaS) applications. In this tutorial, we'll build the foundation for a multi-tenant SaaS application using Laravel and Neon.

By the end of this tutorial, you'll have a fully functional multi-tenant SaaS application where tenants can manage their own books, users, and settings, all while maintaining data isolation between tenants.

Prerequisites

Before we start, make sure you have the following:

  • PHP 8.1 or higher installed on your system
  • Composer for managing PHP dependencies
  • Node.js and npm for managing front-end assets
  • A Neon account for database hosting
  • Basic knowledge of Laravel and Livewire

Setting up the Project

Let's start by creating a new Laravel project and setting up the necessary components.

Creating a New Laravel Project

Open your terminal and run the following command to create a new Laravel project:

composer create-project laravel/laravel laravel-multi-tenant-saas
cd laravel-multi-tenant-saas
Enter fullscreen mode Exit fullscreen mode

Installing Required Packages

For our multi-tenant SaaS application, we'll use the following package:

  • stancl/tenancy: A flexible multi-tenancy package for Laravel
  • Laravel Breeze: A minimal authentication starter kit for Laravel

Start by installing the stancl/tenancy package:

composer require stancl/tenancy
Enter fullscreen mode Exit fullscreen mode

After installing the package, let's set up the tenancy:

php artisan tenancy:install
Enter fullscreen mode Exit fullscreen mode

Register the TenancyServiceProvider in the bootstrap/providers.php file:

return [
    // ...
    App\Providers\TenancyServiceProvider::class,
],
Enter fullscreen mode Exit fullscreen mode

Let's install Laravel Breeze with the Blade views:

composer require laravel/breeze --dev
php artisan breeze:install blade
Enter fullscreen mode Exit fullscreen mode

Next, install the required NPM packages:

npm install
npm run dev
Enter fullscreen mode Exit fullscreen mode

Setting up the Database

Update your .env file with your Neon database credentials:

DB_CONNECTION=pgsql
DB_HOST=your-neon-hostname.neon.tech
DB_PORT=5432
DB_DATABASE=your_database_name
DB_USERNAME=your_username
DB_PASSWORD=your_password
Enter fullscreen mode Exit fullscreen mode

After updating the .env file, run the database migrations:

php artisan migrate
Enter fullscreen mode Exit fullscreen mode

Implementing Multi-Tenancy

Now that we have our basic setup, let's implement multi-tenancy in our application.

Creating the Tenant Model

Create a Tenant model:

php artisan make:model Tenant
Enter fullscreen mode Exit fullscreen mode

Update the app/Models/Tenant.php file:

<?php

namespace App\Models;

use Illuminate\Database\Eloquent\Factories\HasFactory;
use Stancl\Tenancy\Database\Models\Tenant as BaseTenant;
use Stancl\Tenancy\Contracts\TenantWithDatabase;
use Stancl\Tenancy\Database\Concerns\HasDatabase;
use Stancl\Tenancy\Database\Concerns\HasDomains;

class Tenant extends BaseTenant implements TenantWithDatabase
{
    use HasFactory, HasDatabase, HasDomains;

}
Enter fullscreen mode Exit fullscreen mode

This model extends the base Tenant model provided by the tenancy package and implements the TenantWithDatabase interface. We've also defined the fillable attributes and custom columns for our tenant.

The HasDatabase and HasDomains traits provided by the tenancy package allow us to manage tenant-specific databases and domains. This essentially means that each tenant will have its own database and domain providing data isolation between tenants.

To learn more about the tenancy package event system and how to customize the tenant model, refer to the stancl/tenancy documentation.

Configuring Tenancy

Update the config/tenancy.php file to use our custom Tenant model:

'tenant_model' => \App\Models\Tenant::class,
Enter fullscreen mode Exit fullscreen mode

Also, update the central domains configuration:

'central_domains' => [
    'laravel-multi-tenant-saas.test',
    'localhost',
    'example.com',
],
Enter fullscreen mode Exit fullscreen mode

Replace the default central domains with your own domain names.

This is an important part as this is how the tenancy package will determine which domain belongs to which tenant and load the tenant-specific data accordingly.

Feel free to review the other configuration options in the config/tenancy.php file to customize the tenancy behavior based on your requirements.

Creating Tenant Migrations

The tenancy package has built-in event listeners that automatically run tenant-specific migrations when a tenant is created. For this we need to make sure that all of the tenant-specific migrations are in the database/migrations/tenant directory.

As each tenant will have its own database, the migrations in the tenant directory will be used to create tenant-specific tables in the tenant's database.

Start by copying the default User migration to the database/migrations/tenant directory:

cp database/migrations/0001_01_01_000000_create_users_table.php database/migrations/tenant
Enter fullscreen mode Exit fullscreen mode

This will be the base migration for tenant-specific tables.

Implementing Tenant Routes

The tenancy package provides middleware to handle tenant-specific routes. This allows you to define routes that are accessible only to tenants and not to central domains.

Start by creating a new file routes/tenant.php for tenant-specific routes with the following content:

<?php

use Illuminate\Support\Facades\Route;
use Stancl\Tenancy\Middleware\InitializeTenancyByDomain;
use Stancl\Tenancy\Middleware\PreventAccessFromCentralDomains;

/*
|--------------------------------------------------------------------------
| Tenant Routes
|--------------------------------------------------------------------------
|
| Here you can register the tenant routes for your application.
| These routes are loaded by the TenantRouteServiceProvider.
|
| Feel free to customize them however you want. Good luck!
|
*/

Route::middleware([
    'web',
    InitializeTenancyByDomain::class,
    PreventAccessFromCentralDomains::class,
])->group(function () {
    Route::get('/', function () {
        return 'This is your multi-tenant application. The id of the current tenant is ' . tenant('id');
    });

    // Here you can add more tenant-specific routes
});
Enter fullscreen mode Exit fullscreen mode

These routes will be loaded by the TenantRouteServiceProvider and will be accessible only to tenants. The InitializeTenancyByDomain middleware will set the current tenant based on the domain, and the PreventAccessFromCentralDomains middleware will prevent access from central domains.

For more information on how to customize the tenancy routes, refer to the stancl/tenancy documentation.

Implementing Tenant Creation

Create a controller for tenant registration, this would usually be done by the admin users of the application:

php artisan make:controller TenantController
Enter fullscreen mode Exit fullscreen mode

Update the app/Http/Controllers/TenantController.php controller and implement the tenant registration process:

<?php

namespace App\Http\Controllers;

use App\Models\Tenant;
use Illuminate\Http\Request;

class TenantController extends Controller
{
    public function showRegistrationForm()
    {
        return view('tenant.register');
    }

    public function register(Request $request)
    {
        $request->validate([
            'domain' => 'required|string|max:255|unique:domains,domain',
        ]);

        $tenant = Tenant::create();
        $tenant->domains()->create(['domain' => $request->domain]);

        return redirect()->route('tenant.registered', $request->domain);
    }

    public function registered($domain)
    {
        return view('tenant.registered', compact('domain'));
    }
}
Enter fullscreen mode Exit fullscreen mode

This controller handles tenant registration, creates a new tenant in the database, and sets up the tenant's domain. The TenancyServiceProvider will automatically map the tenancy events to the listener, which will create the tenant's database and run the tenant-specific migrations inside the database/migrations/tenant directory for the new tenant.

In a nutshell, the controller has three methods:

  • showRegistrationForm(): Displays the tenant registration form
  • register(): Registers a new tenant, which creates a new tenant record and domain
  • registered(): Displays a success message after registration

This controller will be used to manage tenant registration in our application. Allowing new tenants to register and create their own subdomain and database for their account.

Add routes for tenant registration in routes/web.php:

use App\Http\Controllers\TenantController;

Route::get('/register', [TenantController::class, 'showRegistrationForm'])->name('tenant.register');
Route::post('/register', [TenantController::class, 'register']);
Route::get('/registered/{domain}', [TenantController::class, 'registered'])->name('tenant.registered');
Enter fullscreen mode Exit fullscreen mode

Create the corresponding views for tenant registration starting by creating the resources/views/tenant/register.blade.php file:

<x-guest-layout>
  <form method="POST" action="{{ route('tenant.register') }}">
    @csrf
    <div class="mt-4">
      <x-input-label for="domain" :value="__('Subdomain')" />
      <div class="flex">
        <x-text-input
          id="domain"
          class="mt-1 block w-full"
          type="text"
          name="domain"
          :value="old('domain')"
          required
        />
        <span class="text-gray-600 ml-2 mt-1">.example.com</span>
      </div>
    </div>

    <div class="mt-4 flex items-center justify-end">
      <x-primary-button class="ml-4"> {{ __('Register Tenant') }} </x-primary-button>
    </div>
  </form>
</x-guest-layout>
Enter fullscreen mode Exit fullscreen mode

Then create the resources/views/tenant/registered.blade.php file to display the success message after registration:

<x-guest-layout>
  <div class="text-gray-600 mb-4 text-sm">
    {{ __('Your tenant has been registered successfully!') }}
  </div>

  <div class="mt-4 flex items-center justify-between">
    <div>
      Your tenant URL:
      <a
        href="https://{{ $domain }}.example.com"
        class="text-gray-600 hover:text-gray-900 text-sm underline"
        target="_blank"
        >https://{{ $domain }}.example.com</a
      >
    </div>
  </div>
</x-guest-layout>
Enter fullscreen mode Exit fullscreen mode

This completes the tenant registration process. Tenants can now register and create their own subdomain and database for their account. In a real-world scenario, you would protect the registration routes with authentication middleware to ensure that only authorized admin users can create new tenants.

Verifying Tenant Registration

To verify that the registration process works, visit http://laravel-multi-tenant-saas.test/register and register a new tenant. After registration, you should see the success message with the tenant's domain.

Next go to your Neon dashboard and verify that the new tenant's database has been created:

SELECT * FROM tenants;
Enter fullscreen mode Exit fullscreen mode

You should see the newly created tenant in the tenants table. You can also check the domains table to verify that the tenant's domain has been added:

SELECT * FROM domains;
Enter fullscreen mode Exit fullscreen mode

And to verify that you actually have a separate database for the new tenant, use the \l command in the psql console to list all databases or the following SQL query:

SELECT datname FROM pg_database WHERE datistemplate = false;
Enter fullscreen mode Exit fullscreen mode

The tenant's database should be listed in the results and it should be named tenant{tenant_id}.

The tenancy package allows you to configure the database naming convention for tenants. By default, the database name is tenant{tenant_id} where {tenant_id} is the ID of the tenant. You can also configure the package to use separate schemas instead of separate databases for tenants.

With that done, you've successfully implemented tenant registration in your multi-tenant SaaS application. Next let's implement the tenant onboarding process.

Implementing Tenant Onboarding

Now that you can register new tenants, let's create an onboarding process.

Each tenant will need to create an account to access their dashboard. The domain will be used to identify the tenant, so we'll use the domain as the tenant's subdomain, e.g., tenant1.example.com.

Create a new controller for tenant onboarding:

php artisan make:controller Tenant/OnboardingController
Enter fullscreen mode Exit fullscreen mode

Update the app/Http/Controllers/Tenant/OnboardingController.php to handle the onboarding process:

<?php

namespace App\Http\Controllers\Tenant;

use App\Http\Controllers\Controller;
use App\Models\User;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Hash;

class OnboardingController extends Controller
{
    public function show()
    {
        if (User::count() > 0) {
            return redirect()->route('tenant.dashboard');
        }

        return view('tenant.onboarding');
    }

    public function store(Request $request)
    {
        $validated = $request->validate([
            'name' => 'required|string|max:255',
            'email' => 'required|string|email|max:255|unique:users',
            'password' => 'required|string|min:8|confirmed',
        ]);

        $user = User::create([
            'name' => $validated['name'],
            'email' => $validated['email'],
            'password' => Hash::make($validated['password']),
        ]);

        auth()->login($user);

        return redirect()->route('tenant.dashboard')->with('success', 'Welcome to your new account!');
    }
}
Enter fullscreen mode Exit fullscreen mode

Add routes for the onboarding process in routes/tenant.php inside the Route::middleware group for tenant routes:

use App\Http\Controllers\Tenant\OnboardingController;

Route::middleware([
    'web',
    InitializeTenancyByDomain::class,
    PreventAccessFromCentralDomains::class,
])->group(function () {
    // Existing routes
    // ...

    Route::get('/onboarding', [OnboardingController::class, 'show'])->name('tenant.onboarding');
    Route::post('/onboarding', [OnboardingController::class, 'store'])->name('tenant.onboarding.store');

});
Enter fullscreen mode Exit fullscreen mode

Create the onboarding view in resources/views/tenant/onboarding.blade.php:

<x-guest-layout>
  <form method="POST" action="{{ route('tenant.onboarding.store') }}">
    @csrf

    <div>
      <x-input-label for="name" :value="__('Name')" />
      <x-text-input
        id="name"
        class="mt-1 block w-full"
        type="text"
        name="name"
        :value="old('name')"
        required
        autofocus
        autocomplete="name"
      />
    </div>

    <div class="mt-4">
      <x-input-label for="email" :value="__('Email')" />
      <x-text-input
        id="email"
        class="mt-1 block w-full"
        type="email"
        name="email"
        :value="old('email')"
        required
        autocomplete="username"
      />
    </div>

    <div class="mt-4">
      <x-input-label for="password" :value="__('Password')" />
      <x-text-input
        id="password"
        class="mt-1 block w-full"
        type="password"
        name="password"
        required
        autocomplete="new-password"
      />
    </div>

    <div class="mt-4">
      <x-input-label for="password_confirmation" :value="__('Confirm Password')" />
      <x-text-input
        id="password_confirmation"
        class="mt-1 block w-full"
        type="password"
        name="password_confirmation"
        required
        autocomplete="new-password"
      />
    </div>

    <div class="mt-4 flex items-center justify-end">
      <x-primary-button class="ml-4"> {{ __('Complete Setup') }} </x-primary-button>
    </div>
  </form>
</x-guest-layout>
Enter fullscreen mode Exit fullscreen mode

For simplicity, we're extending the Breeze guest layout for the onboarding form. But you can customize the layout to match your application's design and even have different layouts for the onboarding process based on each tenant's requirements.

To test the onboarding process, visit http://tenant1.example.com/onboarding and complete the onboarding form. After submitting the form, you should be redirected to the tenant dashboard which we'll implement next.

Implementing Tenant Dashboard

Create a new controller for the tenant dashboard:

php artisan make:controller Tenant/DashboardController
Enter fullscreen mode Exit fullscreen mode

Update the app/Http/Controllers/Tenant/DashboardController.php to display the tenant dashboard:

<?php

namespace App\Http\Controllers\Tenant;

use App\Http\Controllers\Controller;
use Illuminate\Http\Request;

class DashboardController extends Controller
{
    public function index()
    {
        return view('tenant.dashboard');
    }
}
Enter fullscreen mode Exit fullscreen mode

Create the dashboard view in resources/views/tenant/dashboard.blade.php:

<x-app-layout>
  <x-slot name="header">
    <h2 class="text-gray-800 text-xl font-semibold leading-tight">{{ __('Dashboard') }}</h2>
  </x-slot>

  <div class="py-12">
    <div class="mx-auto max-w-7xl lg:px-8 sm:px-6">
      <div class="overflow-hidden bg-white shadow-sm sm:rounded-lg">
        <div class="text-gray-900 p-6">{{ __("You're logged in!") }}</div>
      </div>
    </div>
  </div>
</x-app-layout>
Enter fullscreen mode Exit fullscreen mode

Add a route for the tenant dashboard in routes/tenant.php inside the Route::middleware group for tenant routes:

use App\Http\Controllers\Tenant\DashboardController;

Route::middleware([
    'web',
    InitializeTenancyByDomain::class,
    PreventAccessFromCentralDomains::class,
])->group(function () {
    // Existing routes
    // ...
    Route::get('/dashboard', [DashboardController::class, 'index'])->name('tenant.dashboard');

});
Enter fullscreen mode Exit fullscreen mode

To test the tenant dashboard, visit http://tenant1.example.com/dashboard after completing the onboarding process. You should see the dashboard view with a welcome message.

You can also check the users table in the tenant's database to verify that the user account created during onboarding has been added:

SELECT * FROM users;
Enter fullscreen mode Exit fullscreen mode

This will show you the user account created during the onboarding process for that specific tenant in the tenant's database rather than the central database.

Conclusion

In this tutorial, we've built a simple multi-tenant application using Laravel and Neon. We've covered:

  1. Setting up the project and implementing multi-tenancy
  2. Creating a tenant registration process
  3. Implementing tenant onboarding
  4. Adding a tenant dashboard for individual tenants

This implementation provides a foundation for building more complex SaaS applications with Laravel and Neon. You can further expand on this system by:

  • Adding more features to the tenant dashboard
  • Implementing billing and subscription management
  • Enhancing security with two-factor authentication
  • Adding more tenant-specific customizations

Using the stancl/tenancy package along with Neon, each tenant will have its own database. Thanks to Neon's autoscaling feature, you can easily scale your application as you onboard more tenants.

There are other packages and tools available to help you build multi-tenant applications with Laravel. You can explore these options based on your requirements and choose the one that best fits your needs. Some of the popular packages include:

Additional Resources

Top comments (0)