Laravel's schema builder and migration system are great for managing tables and indexes---but offer no built-in support for other SQL entities, such as (materialized) views, procedures, functions, and triggers. These often get handled via raw SQL in migrations, making them hard to manage, prone to unknown conflicts, and difficult to track over time.
laravel-sql-entities
solves this by offering:
- π¦ Class-based definitions: bringing views, functions, triggers, and more into your application code.
- π§ First-class source control: you can easily track changes, review diffs, and resolve conflicts.
- π§± Decoupled grammars: letting you support multiple drivers without needing dialect-specific SQL.
- π Lifecycle hooks: run logic at various points, enabling logging, auditing, and more.
- π Batch operations: easily create or drop all entities in a single command or lifecycle event.
- π§ͺ Testability: definitions are just code so theyβre easy to test, validate, and keep consistent.
Whether you're managing reporting views, business logic functions, or automation triggers, this package helps you treat SQL entities like real, versioned parts of your codebase---no more scattered SQL in migrations!
Note
Migration rollbacks are not supported since the definitions always reflect the latest state.
"We're never going backwards. You only go forward." -Taylor Otwell
First pull in the package using Composer:
composer require calebdw/laravel-sql-entities
The package looks for SQL entities under database/entities/
so you might need to add
a namespace to your composer.json
file, for example:
{
"autoload": {
"psr-4": {
"App\\": "app/",
+ "Database\\Entities\\": "database/entities/",
"Database\\Factories\\": "database/factories/",
"Database\\Seeders\\": "database/seeders/"
}
}
}
Tip
This package looks for any files matching database/entities
in the application's
base path. This means it should automatically work for a modular setup where
the entities might be spread across multiple directories.
To get started, create a new class in a database/entities/
directory
(structure is up to you) and extend the appropriate entity class (e.g. View
, etc.).
For example, to create a view for recent orders, you might create the following class:
<?php
namespace Database\Entities\Views;
use App\Models\Order;
use CalebDW\SqlEntities\View;
use Illuminate\Database\Query\Builder;
use Override;
// will create a view named `recent_orders_view`
class RecentOrdersView extends View
{
#[Override]
public function definition(): Builder|string
{
return Order::query()
->select(['id', 'customer_id', 'status', 'created_at'])
->where('created_at', '>=', now()->subDays(30))
->toBase();
// could also use raw SQL
return <<<'SQL'
SELECT id, customer_id, status, created_at
FROM orders
WHERE created_at >= NOW() - INTERVAL '30 days'
SQL;
}
}
You can also override the name and connection:
<?php
class RecentOrdersView extends View
{
protected ?string $name = 'other_name';
// also supports schema
protected ?string $name = 'other_schema.other_name';
protected ?string $connection = 'other_connection';
}
You can also use the provided lifecycle hooks to run logic before or after an entity is created or dropped.
Returning false
from the creating
or dropping
methods will prevent the entity from being created or dropped, respectively.
<?php
use Illuminate\Database\Connection;
class RecentOrdersView extends View
{
// ...
#[Override]
public function creating(Connection $connection): bool
{
if (/** should not create */) {
return false;
}
/** other logic */
return true;
}
#[Override]
public function created(Connection $connection): void
{
$this->connection->statement(<<<SQL
GRANT SELECT ON TABLE {$this->name()} TO other_user;
SQL);
}
#[Override]
public function dropping(Connection $connection): bool
{
if (/** should not drop */) {
return false;
}
/** other logic */
return true;
}
#[Override]
public function dropped(Connection $connection): void
{
/** logic */
}
}
The SqlEntityManager
singleton is responsible for creating and dropping SQL entities at runtime.
You can interact with it directly, or use the SqlEntity
facade for convenience.
<?php
use CalebDW\SqlEntities\Facades\SqlEntity;
use CalebDW\SqlEntities\SqlEntityManager;
use CalebDW\SqlEntities\View;
// Create a single entity by name, class, or instance
SqlEntity::create('recent_orders_view');
resolve(SqlEntityManager::class)->create(RecentOrdersView::class);
resolve('sql-entities')->create(new RecentOrdersView());
// Similarly, you can drop a single entity using the name, class, or instance
SqlEntity::drop(RecentOrdersView::class);
// Create or drop all entities
SqlEntity::createAll();
SqlEntity::dropAll();
// You can also filter by type or connection
SqlEntity::createAll(type: View::class, connection: 'reporting');
SqlEntity::dropAll(type: View::class, connection: 'reporting');
You may want to automatically drop all SQL entities before migrating, and then recreate them after the migrations are complete. This is helpful when the entities depend on schema changes. To do this, register the built-in subscriber in a service provider:
<?php
use CalebDW\SqlEntities\Listeners\SyncSqlEntities;
use Illuminate\Support\Facades\Event;
use Illuminate\Support\ServiceProvider;
class AppServiceProvider extends ServiceProvider
{
public function boot(): void
{
Event::subscribe(SyncSqlEntities::class);
}
}
Thank you for considering contributing! You can read the contribution guide here.
This is open-sourced software licensed under the MIT license.