xerxes / laravel-rabbitmq-communication
A Laravel package for RabbitMQ messaging with outbox pattern, dead letter queues, and automatic retry mechanisms
Installs: 31
Dependents: 0
Suggesters: 0
Security: 0
Stars: 0
Watchers: 0
Forks: 0
Open Issues: 0
pkg:composer/xerxes/laravel-rabbitmq-communication
Requires
- php: ^8.1
- ext-json: *
- illuminate/contracts: ^9|^10|^11|^12
- php-amqplib/php-amqplib: ^3.0
Requires (Dev)
- nunomaduro/collision: ^7.0|^8.0
- nunomaduro/larastan: ^2.0
- orchestra/testbench: ^8.0|^9.0
- pestphp/pest: ^2.0|^3.0
- pestphp/pest-plugin-laravel: ^2.0|^3.0
- phpstan/extension-installer: ^1.1
- phpstan/phpstan-deprecation-rules: ^1.0
- phpstan/phpstan-phpunit: ^1.0
- phpunit/phpunit: ^10.0|^11.0
This package is auto-updated.
Last update: 2025-12-15 13:18:03 UTC
README
A Laravel package for RabbitMQ messaging with support for the outbox pattern, event consumers, dead letter queues, and automatic retry mechanisms.
Features
- Direct Publish with Outbox Fallback - Messages are published directly; if RabbitMQ is unavailable, they're stored for later retry
- Event Consumers - Consume messages from multiple queues with configurable handlers
- Dead Letter Queue (DLQ) - Automatic retry with exponential backoff, failed messages go to DLQ
- Type-Safe Enums - PHP 8.1+ enums for status and mode handling
- Fully Configurable - Environment-based configuration for all settings
Requirements
- PHP 8.1+
- Laravel 9, 10, 11, or 12
- RabbitMQ Server
- ext-json
Installation
composer require xerxes/laravel-rabbitmq-communication
Publish the configuration:
php artisan vendor:publish --tag=laravel-rabbitmq-communication-config
Run migrations:
php artisan migrate
Configuration
Environment Variables
# Connection Settings RABBITMQ_HOST=127.0.0.1 RABBITMQ_PORT=5672 RABBITMQ_USER=guest RABBITMQ_PASS=guest RABBITMQ_VHOST=/ # Outbox Pattern RABBITMQ_OUTBOX_ENABLED=true RABBITMQ_OUTBOX_CONNECTION=outbox # Dead Letter Queue RABBITMQ_DLQ_ENABLED=true RABBITMQ_DLQ_EXCHANGE=dlx.failed RABBITMQ_DLQ_MAX_RETRIES=3 # Consumer Settings RABBITMQ_CONSUMER_MODE=sync RABBITMQ_CONSUMER_PREFETCH=10 RABBITMQ_LOG_CHANNEL=stack
Publishing Messages
Basic Publishing
use Xerxes\RabbitMQ\RabbitMQ; app(RabbitMQ::class) ->message() ->viaExchange('my.exchange') ->route('my.routing.key') ->withPayload(['event' => 'user.created', 'data' => [...]]) ->persistent() ->publish();
Publishing with Headers
app(RabbitMQ::class) ->message() ->viaExchange('my.exchange') ->route('my.routing.key') ->withPayload($payload) ->withHeaders(['x-custom-header' => 'value']) ->persistent() ->publish();
Outbox Pattern (Guaranteed Delivery)
When outbox is enabled (default), messages follow this flow:
- Try direct publish - Message is sent immediately to RabbitMQ
- On failure - Message is stored in the database outbox table
- Outbox worker - Processes failed messages and retries publishing
// Uses outbox by default (configurable via RABBITMQ_OUTBOX_ENABLED) app(RabbitMQ::class) ->message() ->viaExchange('my.exchange') ->route('my.routing.key') ->withPayload($payload) ->publish(); // Explicitly disable outbox (direct publish only) ->withoutOutbox()
Process outbox messages:
php artisan outbox:work
Using Laravel Events
Implement the ShouldPublish interface to automatically publish events:
use Xerxes\RabbitMQ\Support\ShouldPublish; class UserCreated implements ShouldPublish { public string $exchange = 'user.events'; public function __construct( public User $user ) {} public function routingKey(): string { return 'user.created'; } } // Dispatch normally - automatically published to RabbitMQ event(new UserCreated($user));
Consuming Messages
Configuration
Define event consumers in config/rabbitmq.php:
'event-consumers' => [ [ 'exchange' => 'user.events', 'exchange_type' => 'topic', 'routing_key' => 'user.#', 'handler' => [\App\Handlers\UserHandler::class, 'handle'], 'queue' => 'my-app.users', ], ],
Handler Class
Your handler should implement the MessageHandler interface:
use Xerxes\RabbitMQ\Contracts\MessageHandler; class UserHandler implements MessageHandler { public function handle(array $payload, ?string $routingKey = null): void { // Process the message Log::info('Processing user event', [ 'routing_key' => $routingKey, 'payload' => $payload, ]); } }
Running the Consumer
php artisan rabbitmq:consume-events
Consumer Modes
Configure how consumed events are dispatched:
- sync: Events fired synchronously. Errors stop the consumer (unless DLQ is enabled).
- kind-sync: Events fired synchronously. Errors are logged but don't stop the consumer.
- job: Events dispatched via Laravel queue jobs.
Dead Letter Queue (DLQ)
How It Works
- Message fails processing
- Republished to retry queue with delay
- After TTL, routed back to original queue
- After max retries (default: 3), moved to dead letter queue
- Failed messages can be manually reprocessed
Retry Flow
Message fails → Retry Queue (5s delay) → Original Queue
↓ fails again
Retry Queue (30s delay) → Original Queue
↓ fails again
Retry Queue (2min delay) → Original Queue
↓ fails again (max retries exceeded)
Dead Letter Queue (.dlq)
Reprocessing Failed Messages
# Reprocess up to 100 messages php artisan rabbitmq:reprocess-dlq my-app.orders # Limit the number php artisan rabbitmq:reprocess-dlq my-app.orders --limit=50 # Dry run php artisan rabbitmq:reprocess-dlq my-app.orders --dry-run
Available Commands
| Command | Description |
|---|---|
rabbitmq:consume-events |
Start consuming messages from configured queues |
outbox:work |
Run outbox worker (continuous) |
outbox:process |
Process outbox messages once |
rabbitmq:declare-exchanges |
Declare configured exchanges |
rabbitmq:test-connection |
Test RabbitMQ connection |
rabbitmq:reprocess-dlq {queue} |
Reprocess messages from dead letter queue |
Testing
composer test
Changelog
Please see CHANGELOG for more information on what has changed recently.
Contributing
Please see CONTRIBUTING for details.
License
The MIT License (MIT). Please see License File for more information.