foxws / laravel-scout-relations
Automatically re-index Scout-searchable related models.
Fund package maintenance!
Requires
- php: ^8.4
- illuminate/contracts: ^12.0|^13.0
- laravel/scout: ^10.0|^11.0
- spatie/laravel-package-tools: ^1.16
Requires (Dev)
- larastan/larastan: ^3.0
- laravel/pint: ^1.14
- nunomaduro/collision: ^8.8
- orchestra/testbench: ^10.0|^11.0
- pestphp/pest: ^4.0
- pestphp/pest-plugin-arch: ^4.0
- pestphp/pest-plugin-laravel: ^4.0
- phpstan/extension-installer: ^1.4
- phpstan/phpstan-deprecation-rules: ^2.0
- phpstan/phpstan-phpunit: ^2.0
README
Automatically re-index Scout-searchable related models when an Eloquent model is saved or deleted.
When a parent model changes (e.g. an Author), its related Searchable models (e.g. Post) are automatically queued for re-indexing, keeping your search index consistent without any manual intervention.
Requirements
- PHP 8.4+
- Laravel 12+
- Laravel Scout
Installation
Install the package via Composer:
composer require foxws/laravel-scout-relations
Usage
Add the HasSearchableRelations trait to any Eloquent model whose changes should trigger re-indexing of related models. Then override searchableRelations() to return the relationship names to watch.
use Foxws\ScoutRelations\Concerns\HasSearchableRelations; use Illuminate\Database\Eloquent\Model; use Illuminate\Database\Eloquent\Relations\HasMany; class Author extends Model { use HasSearchableRelations; /** * Relationships whose models should be re-indexed when this model changes. * * @return array<int, string> */ public function searchableRelations(): array { return ['posts']; } public function posts(): HasMany { return $this->hasMany(Post::class); } }
The related Post model must use Laravel Scout's Searchable trait:
use Illuminate\Database\Eloquent\Model; use Laravel\Scout\Searchable; class Post extends Model { use Searchable; public function toSearchableArray(): array { return [ 'id' => $this->id, 'title' => $this->title, 'author_name' => $this->author->name, // kept fresh on every re-index ]; } }
Now whenever an Author is saved with changes or deleted, all of its Post records are automatically re-indexed.
How it works
The trait hooks into Eloquent's saved and deleted model events:
saved— re-indexes relations only whenwasChanged()istrue, avoiding unnecessary indexing on no-op saves.deleted— re-indexes relations unconditionally so the search index reflects the parent's removal.
Re-indexing is performed in chunks via chunkById. If the related model defines makeAllSearchableUsing(), it is applied to the chunk query, preventing N+1 queries.
A per-class re-entry guard prevents infinite cascades when mutual relationships exist.
Configuration
Publish the config file with:
php artisan vendor:publish --tag="scout-relations-config"
Available options in config/scout-relations.php:
| Key | Env variable | Default | Description |
|---|---|---|---|
enabled |
SCOUT_RELATIONS_ENABLED |
true |
Disable all automatic relation syncing |
chunk.searchable |
SCOUT_RELATIONS_CHUNK_SEARCHABLE |
500 |
Chunk size for searchable() calls |
chunk.unsearchable |
SCOUT_RELATIONS_CHUNK_UNSEARCHABLE |
500 |
Chunk size reserved for future use |
Artisan command
To manually force re-indexing of all relations for a model (ignoring the enabled flag), use:
php artisan scout:index-relations "App\Models\Author"
This chunks through every record of the given model and calls reindexSearchableRelations() on each, with a progress bar.
Testing
composer test
Changelog
Please see CHANGELOG for more information on what has changed recently.
Contributing
Please see CONTRIBUTING for details.
Security Vulnerabilities
Please review our security policy on how to report security vulnerabilities.
Credits
License
The MIT License (MIT). Please see License File for more information.