Transaction-aware Event Dispatcher for Laravel
ttlove/laravel-transactional-events is a Laravel package for transaction-aware event dispatcher for laravel.
It currently has 0 GitHub stars and 8 downloads on Packagist (latest version 1.8.9).
Install it with composer require ttlove/laravel-transactional-events.
Discover more Laravel packages by ttlove
or browse all Laravel packages to compare alternatives.
Last updated
This Laravel package introduces Transaction-aware Event Dispatcher.
It ensures the events dispatched within a database transaction are dispatched only if the outer transaction successfully commits. Otherwise, the events are discarded and never dispatched.
Consider the following example of ordering tickets that involves changes to the database.
The orderTickets dispatches the custom OrderCreated event.
In turn, its listener sends an email to the user with the order details.
DB::transaction(function() {
...
$order = $concert->orderTickets($user, 3); // internally dispatches 'OrderCreated' event
PaymentService::registerOrder($order);
});
In the case of transaction failure, due to an exception in the orderTickets method or even a deadlock, the database changes are completely discarded.
Unfortunately, this is not true for the already dispatched OrderCreated event.
This results in sending the order confirmation email to the user, even after the order failure.
The purpose of this package is thus to hold events dispatched within a database transaction until it successfully commits.
In the above example the OrderCreated event would never be dispatched in the case of transaction failure.
Laravel | Package :---------|:---------- 5.5.x-5.7.x | 1.4.x 5.8.x-7.x | 1.8.x
composer:composer require fntneves/laravel-transactional-events
transactional-events.php configuration file:php artisan vendor:publish --provider="Neves\Events\EventServiceProvider"
composer:composer require fntneves/laravel-transactional-events
transactional-events.php configuration file to the config folder:cp vendor/fntneves/laravel-transactional-events/src/config/transactional-events.php config/transactional-events.php
bootstrap/app.php:// Ensure the original EventServiceProvider is registered first, otherwise your event listeners are overriden.
$app->register(App\Providers\EventServiceProvider::class);
$app->configure('transactional-events');
$app->register(Neves\Events\EventServiceProvider::class);
The transaction-aware layer is enabled out of the box for the events under the App\Events namespace.
This package offers three distinct ways to dispatch transaction-aware events:
Neves\Events\Contracts\TransactionalEvent contract;TransactionalClosureEvent event;The simplest way to mark events as transaction-aware events is implementing the Neves\Events\Contracts\TransactionalEvent contract:
namespace App\Events;
use Illuminate\Queue\SerializesModels;
use Illuminate\Foundation\Events\Dispatchable;
...
use Neves\Events\Contracts\TransactionalEvent;
class TicketsOrdered implements TransactionalEvent
{
use Dispatchable, InteractsWithSockets, SerializesModels;
...
}
And that's it. There are no further changes required.
This package provides a generic TransactionalClosureEvent event for bringing the transaction-aware behavior to custom behavior without requiring specific events.
One relevant use case is to ensure that Jobs are dispatched only after the transaction successfully commits:
DB::transaction(function () {
...
Event::dispatch(new TransactionalClosureEvent(function () {
// Job will be dispatched only if the transaction commits.
ProcessOrderShippingJob::dispatch($order);
});
...
});
The configuration file includes the following parameters:
Enable or disable the transaction-aware behavior:
'enable' => true
By default, the transaction-aware behavior will be applied to all events under the App\Events namespace.
Feel free to use patterns and namespaces.
'transactional' => [
'App\Events'
]
Choose the events that should always bypass the transaction-aware layer, i.e., should be handled by the original event dispatcher. By default, all *ed Eloquent events are excluded. The main reason for this default value is to avoid interference with your already existing event listeners for Eloquent events.
'excluded' => [
// 'eloquent.*',
'eloquent.booted',
'eloquent.retrieved',
'eloquent.saved',
'eloquent.updated',
'eloquent.created',
'eloquent.deleted',
'eloquent.restored',
],
Yes. As mentioned in Usage, you can use the generic TransactionalClosureEvent(Closure $callable) event to trigger jobs only after the transaction commits.
This issue is fixed for Laravel 5.6.16+ (see #23832).
For previous versions, it is associated with the RefreshDatabase or DatabaseTransactions trait, namely when it uses database transactions to reset database after each test.
This package relies on events dispached when transactions begin/commit/rollback and as each test is executed within a transaction that is rolled back when test finishes, the dispatched application events are never actually dispatched. In order to get the expected behavior, use the Neves\Testing\RefreshDatabase or Neves\Testing\DatabaseTransactions trait in your tests instead of the ones originally provided by Laravel.
This package is open-sourced software licensed under the MIT license.