Learn how to use Eloquent observers to centralize model lifecycle logic. Stop cluttering services and keep your project board events clean and maintainable.
Previously in this course, we explored Asynchronous Processing with Queues in Laravel to handle heavy lifting in the background. While queues manage deferred tasks, sometimes you need to trigger logic immediately when a database record changes. This is where model observers come into play.
Observers allow you to group all the event listeners for a particular Eloquent model into a single class. Instead of scattering Task::created() hooks across your Service Layer, you centralize that orchestration in one place.
Eloquent models fire several events during their lifecycle: retrieved, creating, created, updating, updated, saving, saved, deleting, deleted, restoring, and restored.
When we talk about the lifecycle, we refer to the sequence of database operations. For instance, when you call $task->save(), Eloquent fires events that allow you to inject custom logic before the SQL is executed (e.g., creating or updating) or after the change is committed (e.g., created or updated).
Let’s evolve our project board. We want to ensure that every time a Task is deleted, we also clean up any associated activity logs or temporary file references. Instead of adding this to our Service-Oriented Task Management, we’ll use an observer.
Use the Artisan command to create the observer class:
Bashphp artisan make:observer TaskObserver --model=Task
This creates app/Observers/TaskObserver.php.
Open the file and add your logic. Observers are just plain classes where method names match the event they handle:
PHPnamespace App\Observers; use App\Models\Task; use Illuminate\Support\Facades\Log; class TaskObserver { public function created(Task $task): void { Log::info("Task {$task->id} was created by user {$task->user_id}"); } public function deleted(Task $task): void { #6A9955">// Clean up related assets $task->attachments()->delete(); } }
To make Laravel aware of your observer, you must register it in your AppServiceProvider.
PHP#6A9955">// app/Providers/AppServiceProvider.php use App\Models\Task; use App\Observers\TaskObserver; public function boot(): void { Task::observe(TaskObserver::class); }
Your project board needs an audit trail.
ProjectObserver to handle the updated event.updated method, log a message whenever a project's name attribute changes.$project->isDirty('name') to check if the column was modified before logging.AppServiceProvider and verify the log output in storage/logs/laravel.log after updating a project via your API.$task->delete() or $task->save()). If you use Task::where('status', 'pending')->update(['status' => 'done']), the updated observer will not fire. Eloquent performs these as direct database queries for performance.$task->save() inside the updated method, you will trigger the updated event again, causing an infinite loop. Always use updateQuietly() if you need to modify the model without triggering further events.We've moved from manual event hooking to a centralized, clean architecture using observers.
boot method of your Service Provider.By keeping your models clean and your side effects isolated, you ensure your project board remains maintainable as it scales.
Up next: We’ll look at Implementing Policies for Authorization to ensure users only interact with tasks they own.
Stop writing massive if-else chains for business logic. Learn how to implement the Strategy pattern in Laravel to keep your services clean and extensible.
Read moreLearn how to use Value Objects in Laravel to eliminate primitive obsession, encapsulate attribute logic, and improve domain clarity in your models.
Using Observers for Model Lifecycle Hooks