sheum / laravel-auto-transaction
Automatic database transaction management for Laravel applications
Requires
- php: ^8.0
- illuminate/database: ^9.0|^10.0|^11.0
- illuminate/http: ^9.0|^10.0|^11.0
- illuminate/support: ^9.0|^10.0|^11.0
Requires (Dev)
- laravel/pint: ^1.0
- mockery/mockery: ^1.5
- orchestra/testbench: ^8.0|^9.0
- phpunit/phpunit: ^10.0
This package is auto-updated.
Last update: 2025-10-04 23:57:11 UTC
README
Automatic database transaction management for Laravel applications. No more manually calling DB::beginTransaction()
, DB::commit()
, and DB::rollBack()
!
Features
- 🛡️ Middleware support - Automatic transactions for entire routes
- 🔧 Trait integration - Easy to use in services and repositories
- 🎨 Helper functions - Simple wrapper functions
- 🔄 Retry mechanism - Handle deadlocks automatically
- 🎛️ Multiple connections - Support for different database connections
Requirements
- PHP 8.1 or higher
- Laravel 10.x or higher
Installation
composer require sheum/laravel-auto-transaction
The package will automatically register itself.
Publish Configuration (Optional)
php artisan vendor:publish --tag=auto-transaction-config
Usage
Method 1: Using Trait with runInTransaction (Recommended for Services)
use Sheum\AutoTransaction\Traits\HandlesTransactions; class UserService { use HandlesTransactions; public function createUser(array $data) { return $this->runInTransaction(function () use ($data) { $user = User::create($data); $user->profile()->create($data['profile']); // Automatically commits on success // Automatically rolls back on exception return $user; }); } public function updateUserWithRetry(User $user, array $data) { return $this->runInTransaction( callback: fn() => $user->update($data), attempts: 3, connection: 'mysql' ); } }
Method 2: Using Middleware (For Controllers)
Apply to specific routes:
// In routes/web.php or routes/api.php Route::middleware(['transaction'])->group(function () { Route::post('/orders', [OrderController::class, 'store']); Route::put('/orders/{order}', [OrderController::class, 'update']); Route::delete('/orders/{order}', [OrderController::class, 'destroy']); }); // Or on a single route Route::post('/users', [UserController::class, 'store']) ->middleware('transaction'); // With custom connection Route::post('/reports', [ReportController::class, 'generate']) ->middleware('transaction:reporting_db');
Apply to controller methods:
class OrderController extends Controller { public function __construct() { $this->middleware('transaction')->only(['store', 'update', 'destroy']); } public function store(Request $request) { $order = Order::create($request->validated()); // Automatically commits on successful response (2xx) // Automatically rolls back on error or exception return response()->json($order, 201); } }
Method 3: Using Helper Functions
// Simple usage $user = transactional(function () use ($data) { $user = User::create($data); $user->profile()->create($data['profile']); return $user; }); // With retry attempts $order = transactional(function () use ($orderData) { return Order::create($orderData); }, attempts: 3); // With custom connection $report = transactional(function () use ($data) { return Report::create($data); }, connection: 'reporting_db'); // Using auto_transaction helper $result = auto_transaction(function () { // Your database operations }, [ 'attempts' => 3, 'connection' => 'mysql' ]);
Method 4: Using Attributes (Advanced)
For advanced usage, you can use attributes with the trait:
use Sheum\AutoTransaction\Attributes\Transactional; use Sheum\AutoTransaction\Traits\HandlesTransactions; class OrderService { use HandlesTransactions; public function placeOrder(array $orderData) { return $this->executeWithTransactionIfNeeded('placeOrderWithTransaction', [$orderData]); } #[Transactional(attempts: 3)] protected function placeOrderWithTransaction(array $orderData) { $order = Order::create($orderData); foreach ($orderData['items'] as $item) { $order->items()->create($item); } return $order; } }
Configuration Options
runInTransaction Parameters
$this->runInTransaction( callback: fn() => /* your code */, connection: 'mysql', // Database connection name (default: null) attempts: 3, // Number of retry attempts (default: 1) throwOnFailure: true // Throw exception on failure (default: true) );
Attribute Parameters
#[Transactional( connection: 'mysql', // Database connection name (default: null) attempts: 3, // Number of retry attempts (default: 1) throwOnFailure: true // Throw exception on failure (default: true) )]
Best Practices
- Use
runInTransaction()
for service methods - Clean and explicit - Use middleware for API endpoints - Automatic transaction per request
- Use helper functions for quick operations - Simple and straightforward
- Use traits in repositories - Consistent transaction handling
- Handle nested transactions carefully - Laravel uses savepoints
- Don't mix transaction methods - Choose one approach per layer
Error Handling
The package throws TransactionException
on failure:
use Sheum\AutoTransaction\Exceptions\TransactionException; try { $user = $userService->createUser($data); } catch (TransactionException $e) { Log::error('User creation failed: ' . $e->getMessage()); return response()->json(['error' => 'Failed to create user'], 500); }
Testing
composer test
Examples
E-commerce Order Processing
use Sheum\AutoTransaction\Traits\HandlesTransactions; class OrderService { use HandlesTransactions; public function completeOrder(Cart $cart, array $paymentData) { return $this->runInTransaction(function () use ($cart, $paymentData) { // Create order $order = Order::create([ 'user_id' => $cart->user_id, 'total' => $cart->total, ]); // Transfer cart items to order foreach ($cart->items as $item) { $order->items()->create([ 'product_id' => $item->product_id, 'quantity' => $item->quantity, 'price' => $item->price, ]); } // Reduce inventory foreach ($cart->items as $item) { Product::find($item->product_id) ->decrement('stock', $item->quantity); } // Process payment $payment = Payment::create([ 'order_id' => $order->id, 'amount' => $cart->total, 'method' => $paymentData['method'], ]); // Clear cart $cart->items()->delete(); // All or nothing - automatic commit/rollback return $order; }, attempts: 3); } }
User Registration with Profile
use Sheum\AutoTransaction\Traits\HandlesTransactions; class UserService { use HandlesTransactions; public function registerUser(array $data) { return $this->runInTransaction(function () use ($data) { $user = User::create([ 'name' => $data['name'], 'email' => $data['email'], 'password' => Hash::make($data['password']), ]); $user->profile()->create([ 'bio' => $data['bio'], 'avatar' => $data['avatar'], ]); $user->settings()->create([ 'notifications' => true, 'newsletter' => $data['newsletter'] ?? false, ]); return $user; }); } }
Banking Transfer Service
use Sheum\AutoTransaction\Traits\HandlesTransactions; class BankingService { use HandlesTransactions; public function transfer(Account $from, Account $to, float $amount) { return $this->runInTransaction( callback: function () use ($from, $to, $amount) { if ($from->balance < $amount) { throw new \Exception('Insufficient funds'); } $from->decrement('balance', $amount); $to->increment('balance', $amount); return Transaction::create([ 'from_account_id' => $from->id, 'to_account_id' => $to->id, 'amount' => $amount, ]); }, attempts: 5, connection: 'banking' ); } }
Additional Examples
Please see Additional Examples for more examples.
License
The MIT License (MIT). Please see License File for more information.
Changelog
Please see CHANGELOG for more information on what has changed recently.
Contributing
Contributions are welcome! Please feel free to submit a Pull Request.
Credits
Support
If you discover any issues, please email nazmulhasansheum@gmail.com or create an issue on GitHub.