Kritim Yantra
Apr 03, 2025
Imagine you're building a house. Contracts are like the blueprints that define how different parts (like doors, windows, or electrical systems) should work, without specifying exactly which brand or model to use. In Laravel, contracts are interfaces that define how core services should behave.
Think of facades as shortcuts and contracts as formal agreements:
Feature | Facades | Contracts |
---|---|---|
Usage | Simple static calls | Type-hinted dependencies |
Flexibility | Less flexible | More flexible |
Testability | Good | Excellent |
Explicit Dependencies | No | Yes |
Example: Sending an email
// Using Facade
Mail::to('user@example.com')->send(new WelcomeEmail());
// Using Contract
public function __construct(
protected Illuminate\Contracts\Mail\Mailer $mailer
) {}
public function sendWelcomeEmail()
{
$this->mailer->to('user@example.com')->send(new WelcomeEmail());
}
use Illuminate\Contracts\Cache\Repository as Cache;
class UserRepository
{
public function __construct(
protected Cache $cache,
protected User $model
) {}
public function getActiveUsers()
{
return $this->cache->remember('active_users', 3600, function() {
return $this->model->where('active', true)->get();
});
}
}
use Illuminate\Contracts\Queue\Queue;
class ReportGenerator
{
public function __construct(
protected Queue $queue
) {}
public function generateAsync(Report $report)
{
$this->queue->push(new GenerateReportJob($report));
}
}
use Illuminate\Contracts\Filesystem\Filesystem;
class DocumentUploader
{
public function __construct(
protected Filesystem $storage
) {}
public function upload(User $user, UploadedFile $file)
{
$path = "documents/{$user->id}/".$file->hashName();
$this->storage->put($path, file_get_contents($file));
return $path;
}
}
Use Contracts When:
Use Facades When:
Contracts make testing much easier. Here's an example with the Mail contract:
// In your service
public function __construct(
protected Illuminate\Contracts\Mail\Mailer $mailer
) {}
public function notifyUser(User $user)
{
$this->mailer->to($user->email)->send(new NotificationMail());
}
// In your test
public function test_notification_sent()
{
$mailer = Mockery::mock(Illuminate\Contracts\Mail\Mailer::class);
$mailer->shouldReceive('send')->once();
$service = new MyService($mailer);
$service->notifyUser(new User(['email' => 'test@example.com']));
}
Here are some frequently used contracts:
Cache (Illuminate\Contracts\Cache\Repository
)
$this->cache->get('key')
Queue (Illuminate\Contracts\Queue\Queue
)
$this->queue->push(new Job())
Filesystem (Illuminate\Contracts\Filesystem\Filesystem
)
$this->filesystem->put('file.txt', 'content')
Mail (Illuminate\Contracts\Mail\Mailer
)
$this->mailer->send(new Mailable())
Events (Illuminate\Contracts\Events\Dispatcher
)
$this->events->dispatch(new OrderShipped($order))
When you type-hint a contract in a constructor, Laravel automatically resolves the appropriate implementation:
// Laravel will inject the default cache implementation
public function __construct(
Illuminate\Contracts\Cache\Repository $cache
) {
$this->cache = $cache;
}
This works because Laravel's service container knows which concrete class to use for each interface.
You can create contracts for your own services:
namespace App\Contracts;
interface PaymentProcessor
{
public function charge(float $amount, array $options);
public function refund(string $transactionId);
}
// Implementation
class StripePaymentProcessor implements PaymentProcessor
{
public function charge(float $amount, array $options) { /* ... */ }
public function refund(string $transactionId) { /* ... */ }
}
// Usage
public function __construct(
protected App\Contracts\PaymentProcessor $payment
) {}
You might wonder - do contracts impact performance? The answer is:
Laravel Contracts are powerful tools that:
While facades are great for quick development, contracts provide the structure and flexibility needed for larger, more complex applications. By understanding both, you can choose the right tool for each situation in your Laravel projects.
No comments yet. Be the first to comment!
Please log in to post a comment:
Continue with Google