toramanlis/laravel-implicit-migrations

This package is a tool that creates Laravel migration files by inspecting the application's models.

v1.0.0 2025-04-02 19:29 UTC

This package is auto-updated.

Last update: 2025-04-02 19:29:49 UTC


README

codecov Latest Version on Packagist Software License Total Downloads

Overview

What It Is

This package is a tool that creates Laravel migration files by inspecting the application's models with the command php artisan implicit-migrations:generate. Even after you change the model classes, you can run the command and generate a migration with the necessary update operations.

How It Works

With the most basic configuration, the implicit-migrations:generate artisan command looks at a Eloquent model and finds necessary information about the table properties such as the table name, primary key etc. Then it goes over the properties of the model and collects the name, type and default value information if provided. With the information collected, it creates a migration file and populates the up() and down() methods with the appropriate definitions.

Implications

For further details, the generator refers to some additional data in the model class which we call "Implications". These can be specified with either annotations or attributes on the class, its properties and methods.

Annotations

Annotations in DocBlocks with the format @<implication-name>(<parameters>) are recognized and interpreted as implications. For example, an annotation like this tells the generator that this integer property corresponds to an UNSIGNED INT column named product_id in the order_items table:

<?php

namespace App\Models;

use Illuminate\Database\Eloquent\Model;

class OrderItem extends Model
{
    /**
     * @Column(unsigned: true)
     */
    public int $product_id;
}

In turn, the generator produces a migration like this:

<?php

use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
use App\Models\OrderItem as Source;

return new class extends Migration
{
    public const TABLE_NAME = 'order_items';

    public function getSource(): string
    {
        return Source::class;
    }

    public function tableUp(Blueprint $table): void
    {
        $table->id();
        $table->integer('product_id')->unsigned();
        $table->timestamps();
    }

    public function up(): void
    {
        Schema::create(static::TABLE_NAME, function (Blueprint $table) {
            $this->tableUp($table);
        });
    }

    public function down(): void
    {
        Schema::drop(static::TABLE_NAME);
    }
};

You can find out more on other implications in the Implication Reference section.

PHP Attributes

Another way of specifying implications is using PHP attributes. The very same implications are avaliable as attributes with the same notation. This is the same model definition as above as far as the generator is concerned:

<?php

namespace App\Models;

use Illuminate\Database\Eloquent\Model;
use Toramanlis\ImplicitMigrations\Attributes\Column;

class OrderItem extends Model
{
    #[Column(unsigned: true)]
    public int $product_id;
}

The advantage of this option is that your IDE/Editor will recognize the attributes as the classes they are and provide autocompletion and descriptions. The down side is that the classes have to be referenced in the models and now they need to be existent in the production environment too.

The obvious approach to tackling this is just adding the implicit-migrations package to the require instead of require-dev. The neat approach, on the other hand, is to get the attribute classes to the database/migrations/attributes directory of the project by publishing them with the php artisan vendor:publish --tag=implication-attributes command and add "database/attributes/composer.json" to the composer.json file like this:

...
"extra": {
    "merge-plugin": {
        "include": [
            "database/attributes/composer.json"
        ]
    }
}
...

This way, you can have the package in the require-dev section of your composer.json and still have the attribute classes available in production.

Updates

This tool doesn't only work for creating a table for a model. If you change your model and run implicit-migrations:generate again, it will resolve the changes by referring to the already generated migrations (Only the generated migrations that is. See: Manual Migrations) and generate a new migration that applies the changes to the table structure.

For example if you update the above model and add another property to it:

<?php

namespace App\Models;

use Illuminate\Database\Eloquent\Model;

class OrderItem extends Model
{
    /**
     * @Column(unsigned: true)
     */
    public int $product_id;

    public int $order_id;
}

After you run php artisan implicit-migrations:generate having the initial migration above already in place, you will get another migration like this:

<?php

use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
use App\Models\OrderItem as Source;

return new class extends Migration
{
    public const TABLE_NAME = 'order_items';

    public function getSource(): string
    {
        return Source::class;
    }

    public function tableUp(Blueprint $table): void
    {
        $table->integer('order_id');
    }

    public function tableDown(Blueprint $table): void
    {
        $table->dropColumn('order_id');
    }

    public function up(): void
    {
        Schema::table(static::TABLE_NAME, function (Blueprint $table) {
            $this->tableUp($table);
        });
    }

    public function down(): void
    {
        Schema::table(static::TABLE_NAME, function (Blueprint $table) {
            $this->tableDown($table);
        });
    }
};

Installation

The recommended installation is using composer require --dev toramanlis/laravel-implicit-migrations command. Since this will make the implication attributes unavailable in production, if you want to use attributes for implications you have to take one of the following approaches:

Publishing The Attributes

You can publish the attribute classes with php artisan vendor:publish --tag=implication-attributes and add the line "database/attributes/composer.json" in the composer.json file like this:

...
"extra": {
    "merge-plugin": {
        "include": [
            "database/attributes/composer.json"
        ]
    }
}
...

Opting Out From Attributes

Each and every one of the implications are available as both attributes and annotations. You can completely give up using attributes and switch to the annotation notation with no missing functionality.

Installing To Production

Alternatively, you can always install the package with composer install toramanlis/laravel-implicit-migrations without the --dev option. Having a tool like this in production sure is unnecessary, but it's just that, unnecessary.

Configuration

database.model_paths

Type: array
Default: ['app/Models']

An array of paths relative to the project directory where application models reside. If there are multiple model and migration paths in a project, the migration files are created in the migration path that is closest to the source model in the directory tree (complicit with nWidart/laravel-modules).

database.auto_infer_migrations

Type: bool
Default: true

This is a boolean value that controls, you guessed it, whether or not to infer the migration information automatically. What this means is basically, unless specified otherwise with an implication, none of the models, properties or methods are going to be inspected for migration information. However, if a property or method of a model has an implication, that model will be inspected. The default is true.

database.implications.<implication_name_in_snake_case>

Type: bool
Default: true

These are boolean values that can be used to enable or disable each implication. The implication names have to be in snake case as per Laravel's convention for configuration keys e.g. database.implications.foreign_key. This set to true by default for all the implications.

Manual Migrations

It's always a good idea to have a backup plan. You might come accross some more intricate or complicated requirements from a migration. For this reason, this tool doesn't take into account any migrations that does not have a getSource() method. This way, you can add your own custom migrations that are processed by Laravel's migrate command, but completely invisible to implicit-migrations:generate.

If a manual migration happens to have a method named getSource, the Off implication can be utilized to indicate that it is in fact a manual migration.

Implication Reference

All the PHP attributes for the implications reside in the namespace Toramanlis\ImplicitMigrations\Attributes. If you choose to utilize them, make sure they're available in your production environment as well. See the installation section for details.

Generally, the parameters of the implications are optional as they often have default values or can possibly be inferred from the rest of the information available in the application, such as the native PHP definitions of models, properties and methods or other implications' details.

Best to keep in mind that these details still might not be sufficient to make a definition and some of the optional parameters might, in fact, be required.

Table

Target: class

Table(?string $name = null, ?string $engine = null, ?string $charset = null, ?string $collation = null)

Used with classes for specifying the table details. When the database.auto_infer_migrations configuration option is set to true, using this implication lets the class get processed.

Column

Target: class, property

Column(?string $type = null, ?string $name = null, ?bool $nullable = null, $default = null, ?int $length = null, ?bool $unsigned = null, ?bool $autoIncrement = null, ?int $precision = null, ?int $total = null, ?int $places = null, ?array $allowed = null, ?bool $fixed = null, ?string $subtype = null, ?int $srid = null, ?string $expression = null, ?string $collation = null, ?string $comment = null, ?string $virtualAs = null, ?string $storedAs = null, ?string $after = null)

Can be used both on classes and properties to define columns. The name parameter is mandatory when used on classes as it won't be able to infer the column name. In contrast, when used on a property, column name defaults to the name of the property. Either by using it on a property or providing a name that matches a property allows it to infer whatever information available in the definition of said property.

Index

Target: class, property

Index(null|array|string $column = null, string $type = 'index', ?string $name = null, ?string $algorithm = null, ?string $language = null)

Just like Column, this can also be used both on classes and properties. The column parameter is optional when used on a property and defaults to the column name associated with that property even if the property doesn't have a Column implication of its own. When used on a class, the column parameter is mandatory.

When Index is associated with a single column name by either using it on a property or having given a value to the column parameter, it will try and ensure the existence of a column with that name, using any information available in the model definition.

Unique

Target: class, property

Unique(null|array|string $column = null, ?string $name = null, ?string $algorithm = null, ?string $language = null)

Alias for Index($column, type: 'unique', ...$args)

Primary

Target: class, property

Primary(null|array|string $column = null, ?string $name = null, ?string $algorithm = null, ?string $language = null)

Alias for Index($column, type: 'primary', ...$args)

Relationship

Target: method

Relationship()

Specifies that a method is a Laravel relationship. What kind of relationship it is will always be inferred by the return type of the method. This implication is redundant if the database.auto_infer_migrations configuration option is set to true, as the return type of a public method is already taken as an implication of whether or not it's a relationship method.

If the type of relationship requires tables and columns that are not defined, Relationship will try to ensure them in the migration using whatever information is available.

ForeignKey

Target: class, property

ForeignKey(string $on, null|array|string $column = null, null|array|string $references = null, ?string $onUpdate = null, ?string $onDelete = null)

Similar to Index, this can be used both on classes and properties, but with classes, it's mandatory to provide the column parameter.

The on parameter can be a table name or a class name of a model.

PivotTable

Target: method

PivotTable(?string $name = null, ?string $engine = null, ?string $charset = null, ?string $collation = null)

Specifies the details of a pivot table of a relationship. Even if no Relationship implication is present, having this implication lets the generator know it's a relationship method.

PivotColumn

Target: method

PivotColumn(?string $name, protected ?string $type = null, ?bool $nullable = null, $default = null, ?int $length = null, ?bool $unsigned = null, ?bool $autoIncrement = null, ?int $precision = null, ?int $total = null, ?int $places = null, ?array $allowed = null, ?bool $fixed = null, ?string $subtype = null, ?int $srid = null, ?string $expression = null, ?string $collation = null, ?string $comment = null, ?string $virtualAs = null, ?string $storedAs = null, ?string $after = null)

Defines a column on a pivot table of a relationship. Just like PivotTable, having this implication lets the generator know it's a relationship method. Since pivot tables typically don't have models of their own, we define any extra columns on the relationship method they are required by. Only the columns other than the foreign keys need this implicaiton, foreign keys are already covered with the relationship. It's still allowed to use this implication to fine tune them, though.

Off

Target: class, property, method

Off()

Lets the generator know that the given class, property or method should be ignored. This includes getSource method a migration. If you have a manually written migration that happens to have a method named getSource, you can add this implication to that method to keep the generator off of that migration.