eloquent-best-practices
π―Skillfrom iserter/laravel-claude-agents
Optimizes Laravel Eloquent queries by implementing best practices for performance, relationship management, and code efficiency.
Installation
npx skills add https://github.com/iserter/laravel-claude-agents --skill eloquent-best-practicesSkill Details
Best practices for Laravel Eloquent ORM including query optimization, relationship management, and avoiding common pitfalls like N+1 queries.
Overview
# Eloquent Best Practices
Query Optimization
Always Eager Load Relationships
```php
// β N+1 Query Problem
$posts = Post::all();
foreach ($posts as $post) {
echo $post->user->name; // N additional queries
}
// β Eager Loading
$posts = Post::with('user')->get();
foreach ($posts as $post) {
echo $post->user->name; // No additional queries
}
```
Select Only Needed Columns
```php
// β Fetches all columns
$users = User::all();
// β Only needed columns
$users = User::select(['id', 'name', 'email'])->get();
// β With relationships
$posts = Post::with(['user:id,name'])->select(['id', 'title', 'user_id'])->get();
```
Use Query Scopes
```php
// β Define reusable query logic
class Post extends Model
{
public function scopePublished($query)
{
return $query->where('status', 'published')
->whereNotNull('published_at');
}
public function scopePopular($query, $threshold = 100)
{
return $query->where('views', '>', $threshold);
}
}
// Usage
$posts = Post::published()->popular()->get();
```
Relationship Best Practices
Define Return Types
```php
use Illuminate\Database\Eloquent\Relations\BelongsTo;
use Illuminate\Database\Eloquent\Relations\HasMany;
class Post extends Model
{
public function user(): BelongsTo
{
return $this->belongsTo(User::class);
}
public function comments(): HasMany
{
return $this->hasMany(Comment::class);
}
}
```
Use withCount for Counts
```php
// β Triggers additional queries
foreach ($posts as $post) {
echo $post->comments()->count();
}
// β Load counts efficiently
$posts = Post::withCount('comments')->get();
foreach ($posts as $post) {
echo $post->comments_count;
}
```
Mass Assignment Protection
```php
class Post extends Model
{
// β Whitelist fillable attributes
protected $fillable = ['title', 'content', 'status'];
// Or blacklist guarded attributes
protected $guarded = ['id', 'user_id'];
// β Never do this
// protected $guarded = [];
}
```
Use Casts for Type Safety
```php
class Post extends Model
{
protected $casts = [
'published_at' => 'datetime',
'metadata' => 'array',
'is_featured' => 'boolean',
'views' => 'integer',
];
}
```
Chunking for Large Datasets
```php
// β Process in chunks to save memory
Post::chunk(200, function ($posts) {
foreach ($posts as $post) {
// Process each post
}
});
// β Or use lazy collections
Post::lazy()->each(function ($post) {
// Process one at a time
});
```
Database-Level Operations
```php
// β Slow - loads into memory first
$posts = Post::where('status', 'draft')->get();
foreach ($posts as $post) {
$post->update(['status' => 'archived']);
}
// β Fast - single query
Post::where('status', 'draft')->update(['status' => 'archived']);
// β Increment/decrement
Post::where('id', $id)->increment('views');
```
Use Model Events Wisely
```php
class Post extends Model
{
protected static function booted()
{
static::creating(function ($post) {
$post->slug = Str::slug($post->title);
});
static::deleting(function ($post) {
$post->comments()->delete();
});
}
}
```
Common Pitfalls to Avoid
Don't Query in Loops
```php
// β Bad
foreach ($userIds as $id) {
$user = User::find($id);
}
// β Good
$users = User::whereIn('id', $userIds)->get();
```
Don't Forget Indexes
```php
// Migration
Schema::create('posts', function (Blueprint $table) {
$table->id();
$table->foreignId('user_id')->constrained()->index();
$table->string('slug')->unique();
$table->string('status')->index();
$table->timestamp('published_at')->nullable()->index();
// Composite index for common queries
$table->index(['status', 'published_at']);
});
```
Prevent Lazy Loading in Development
```php
// In AppServiceProvider boot method
Model::preventLazyLoading(!app()->isProduction());
```
Checklist
- [ ] Relationships eagerly loaded where needed
- [ ] Only selecting required columns
- [ ] Using query scopes for reusability
- [ ] Mass assignment protection configured
- [ ] Appropriate casts defined
- [ ] Indexes on foreign keys and query columns
- [ ] Using database-level operations when possible
- [ ] Chunking for large datasets
- [ ] Model events used appropriately
- [ ] Lazy loading prevented in development
More from this repository4
Provides a structured, step-by-step diagnostic approach for systematically identifying, analyzing, and resolving complex debugging challenges in Laravel applications, focusing on methodical problem...
Facilitates collaborative feature ideation and design planning for Laravel projects by guiding developers through structured brainstorming techniques and generating comprehensive feature specificat...
Transforms Laravel API resources with advanced patterns for conditional attributes, relationship handling, and dynamic data presentation.
Guides Laravel developers through test-driven development using Pest PHP, writing tests first and implementing minimal code to pass.