predatorstudio / laravel-attribute-code-generators
Package info
github.com/PredatorStudio/LaravelAttributeCodeGenerators
pkg:composer/predatorstudio/laravel-attribute-code-generators
Requires
- php: ^8.1
- illuminate/console: ^10.0|^11.0
- illuminate/filesystem: ^10.0|^11.0
- illuminate/support: ^10.0|^11.0
- nikic/php-parser: ^5.0
Requires (Dev)
- orchestra/testbench: ^8.0|^9.0
- phpunit/phpunit: ^10.0|^11.0
This package is not auto-updated.
Last update: 2026-05-08 19:54:56 UTC
README
v1.0
A Laravel package that generates a full CRUD scaffold from PHP 8.2 Attributes placed directly on Eloquent models. One command reads every model in your app, inspects its attributes, and writes controllers, services, repositories, DTOs, resources, migrations, policies, observers, factories, seeders, tests, and routes — only what you asked for, nothing more.
Requirements
- PHP 8.2+
- Laravel 11, 12, or 13
Installation
composer require predatorstudio/laravel-attribute-code-generators
The service provider is auto-discovered via Laravel's package auto-discovery.
Claude Code integration (optional)
Install a /describe-attributes slash command into your project's Claude Code environment:
php artisan crud:install --ai
This copies a prompt file to .claude/commands/describe-attributes.md in your project root. After restarting Claude Code the command is available as a slash command:
| Command | Description |
|---|---|
/describe-attributes |
Full reference — every attribute, its parameters, generated artifacts, and interaction map |
/describe-attributes Seeder |
Reference for a single attribute (case-insensitive) |
The skill covers:
- What each attribute generates
- All constructor parameters with types, defaults, and descriptions
- Usage examples
- Non-obvious behaviours (idempotency, PATCH semantics,
hiddenflag, ALTER migrations, …) fields()key reference- Attribute interaction map (which attributes depend on or complement each other)
Usage
Annotate your model with the attributes that describe what you want generated, then run:
php artisan crud:sync
| Option | Description |
|---|---|
--dry-run |
Preview what would be generated without writing any files |
--force |
Overwrite existing files (asks for confirmation per file) |
Available Attributes
#[Crud]
Enables CRUD generation for the model. Optionally restricts which HTTP methods are scaffolded. The generated controller and resource routes will contain only the listed methods.
#[Crud] // all methods #[Crud(methods: ['index', 'store', 'show', 'update', 'destroy'])]
#[Route]
Registers the resource route under the given path. Optionally applies middleware to the route group.
#[Route(path: 'users')] #[Route(path: 'users', middleware: ['auth:sanctum', 'verified'])]
#[Resource]
Generates an API resource class. Specify which fields to expose, or omit fields to auto-generate from the model's visible fields() columns (respecting hidden: true).
#[Resource] #[Resource(fields: ['id', 'name', 'email'])]
#[Service]
Generates a service class. Pass interface: true to also generate a contract and bind it in the service container.
#[Service]
#[Service(interface: true)]
#[Repository]
Generates a repository class. Pass interface: true to also generate a contract and bind it.
#[Repository]
#[Repository(interface: true)]
#[DTO]
Generates a Data Transfer Object with readonly properties, a fromArray() factory, and a toArray() method. Fields marked hidden: true in fields() are excluded.
#[DTO]
#[Policy]
Generates a policy class for the model.
#[Policy]
#[Observer]
Generates an observer class and registers it automatically.
#[Observer]
#[Factory]
Generates an Eloquent factory for the model with sensible faker defaults per column type.
#[Factory]
#[Seeder]
Generates a database seeder that uses the model's factory to create records. The default count is 10.
#[Seeder]
#[Seeder(count: 50)]
#[SoftDeletes]
Adds soft-delete support: appends $table->softDeletes() to the generated migration and injects the SoftDeletes trait into the model.
#[SoftDeletes]
#[GenerateMigration]
Generates a create_*_table migration based on the model's fields() method. On subsequent runs the package compares saved column names against the current fields() list — if new columns are found it generates an add_columns_to_*_table ALTER migration instead of recreating the original.
#[GenerateMigration]
#[ValidateFromMigration]
Generates StoreRequest and UpdateRequest validation rules derived from the migration columns. The UpdateRequest automatically prepends sometimes to every rule, making it suitable for PATCH requests (only fields present in the payload are validated).
Fields marked hidden: true in fields() are excluded from both requests.
#[ValidateFromMigration]
#[BackedEnum]
Generates a backed enum for a field. Repeatable — add one per enum field.
#[BackedEnum(field: 'status', values: ['active', 'inactive'])] #[BackedEnum(field: 'role', values: ['admin', 'editor', 'viewer'], type: 'string')]
#[Action]
Generates single-purpose action classes (Create, Update, Delete) for the model.
#[Action]
#[GenerateTest]
Generates a feature test for the model's CRUD endpoints.
#[GenerateTest]
The fields() method
Define the model's schema by implementing a fields() method. It drives migration generation, validation rules, factories, DTOs, and resource output.
public function fields(): array { return [ ['type' => 'id'], ['name' => 'title', 'type' => 'string'], ['name' => 'body', 'type' => 'text', 'nullable' => true], ['name' => 'status', 'type' => 'string', 'default' => 'draft'], ['name' => 'user_id', 'type' => 'foreignId'], ['name' => 'secret_hash', 'type' => 'string', 'hidden' => true], ['type' => 'timestamps'], ]; }
| Key | Type | Description |
|---|---|---|
name |
string | Column name |
type |
string | Column type (string, text, integer, boolean, foreignId, json, id, timestamps, …) |
nullable |
bool | Adds nullable() to the migration column and nullable to validation rules |
unique |
bool | Adds unique() to the migration column |
default |
mixed | Adds ->default(value) to the migration column |
hidden |
bool | Excludes the field from requests, resources, and DTOs — useful for internal columns like hashed tokens or audit fields |
Full example
<?php namespace App\Models; use Illuminate\Database\Eloquent\Model; use Vendor\LaravelAttributeCodeGenerators\Attributes\Action; use Vendor\LaravelAttributeCodeGenerators\Attributes\BackedEnum; use Vendor\LaravelAttributeCodeGenerators\Attributes\Crud; use Vendor\LaravelAttributeCodeGenerators\Attributes\DTO; use Vendor\LaravelAttributeCodeGenerators\Attributes\Factory; use Vendor\LaravelAttributeCodeGenerators\Attributes\GenerateTest; use Vendor\LaravelAttributeCodeGenerators\Attributes\Observer; use Vendor\LaravelAttributeCodeGenerators\Attributes\Policy; use Vendor\LaravelAttributeCodeGenerators\Attributes\Repository; use Vendor\LaravelAttributeCodeGenerators\Attributes\Resource; use Vendor\LaravelAttributeCodeGenerators\Attributes\Route; use Vendor\LaravelAttributeCodeGenerators\Attributes\Seeder; use Vendor\LaravelAttributeCodeGenerators\Attributes\Service; use Vendor\LaravelAttributeCodeGenerators\Attributes\SoftDeletes; use Vendor\LaravelAttributeCodeGenerators\Attributes\ValidateFromMigration; #[Crud(methods: ['index', 'store', 'show', 'update', 'destroy'])] #[Route(path: 'users', middleware: ['auth:sanctum'])] #[Resource(fields: ['id', 'name', 'email'])] #[Service(interface: true)] #[Repository(interface: true)] #[Policy] #[ValidateFromMigration] #[DTO] #[SoftDeletes] #[BackedEnum(field: 'status', values: ['active', 'inactive'])] #[Observer] #[Action] #[Factory] #[Seeder(count: 20)] #[GenerateTest] class User extends Model { public function fields(): array { return [ ['type' => 'id'], ['name' => 'name', 'type' => 'string'], ['name' => 'email', 'type' => 'string', 'unique' => true], ['name' => 'bio', 'type' => 'text', 'nullable' => true], ['name' => 'password', 'type' => 'string', 'hidden' => true], ['type' => 'timestamps'], ]; } }
Running php artisan crud:sync on this model generates:
UserControllerwith the five resource methodsUserService+UserServiceInterface(bound in the container)UserRepository+UserRepositoryInterface(bound in the container)UserResourceexposingid,name,email(password excluded viahidden)UserDTOwith readonly properties (password excluded viahidden)UserStoreRequestwith validation rules from the migrationUserUpdateRequestwith the same rules prefixed withsometimesfor PATCH semanticsUserPolicyUserObserverUserFactorywith faker values per column typeUserSeedercreating 20 records via the factoryUserStatusEnumbacked enum- A feature test
UserTest - A migration file with
softDeletes() - Resource routes registered under
/usersbehind theauth:sanctummiddleware
On the next run, already-generated artifacts are skipped. If new columns are added to fields(), only an ALTER migration is generated for the new columns.
License
MIT