mahdi-mh / fallback-chain
A lightweight PHP package for building fallback chains with callable handlers and failure callbacks.
1.1.1
2026-01-04 12:05 UTC
Requires
- php: ^7.1 || ^8.0
Requires (Dev)
- phpunit/phpunit: ^7.5 || ^8.5 || ^9.5 || ^10.0 || ^11.0
README
A lightweight PHP package for building fallback chains with callable handlers and failure callbacks.
Features
- Fluent Interface: Build chains with a clean, readable syntax
- Automatic Fallback: Automatically continues to the next stage when one fails
- Failure Callbacks: Execute custom logic when a stage fails (logging, metrics, etc.)
- Context Sharing: Share data between all stages via a context object
- Exception Collection: Access all exceptions when the entire chain fails
- PHP 7.1+ Compatible: Works with PHP 7.1 through PHP 8.x
Installation
Install the package via Composer:
composer require mahdi-mh/fallback-chain
Usage
Basic Example
use MahdiMh\FallbackChain\FallbackChain; use MahdiMh\FallbackChain\Stage; use MahdiMh\FallbackChain\AllStagesFailedException; $context = [ 'to' => '09121234567', 'text' => 'Hello, this is a test message', ]; $chain = new FallbackChain($context); $chain ->add(Stage::handler(function ($ctx) { return SmsProvider1::send($ctx['to'], $ctx['text']); })->onFailure(function (\Throwable $e, $ctx) { error_log('SMS Provider 1 failed: ' . $e->getMessage()); })) ->add(Stage::handler(function ($ctx) { return SmsProvider2::send($ctx['to'], $ctx['text']); })) ->add(Stage::handler(function ($ctx) { return SmsProvider3::send($ctx['to'], $ctx['text']); })->onFailure(function (\Throwable $e, $ctx) { error_log('Provider 3 unavailable: ' . $e->getMessage()); })); try { $result = $chain->execute(); echo "Message sent successfully: " . $result; } catch (AllStagesFailedException $e) { echo "All providers failed."; // Access all exceptions that occurred foreach ($e->getExceptions() as $exception) { error_log($exception->getMessage()); } }
How It Works
- Create a chain with a shared context (any value - array, object, etc.)
- Add stages using the fluent
add()method - Execute the chain - it tries each stage in order:
- If a stage succeeds (no exception), its result is returned immediately
- If a stage fails (throws any
Throwable), the optionalonFailurecallback runs - Execution continues to the next stage
- Handle total failure - if all stages fail,
AllStagesFailedExceptionis thrown
Use Cases
- Multiple SMS/Email Providers: Try primary provider, fall back to secondary
- Payment Gateways: Attempt multiple payment processors
- API Endpoints: Try different API servers or mirrors
- Cache Layers: Check memory cache, then Redis, then database
- File Storage: Try local storage, then S3, then another cloud provider
Creating Stages
// Stage with just a handler $stage = Stage::handler(function ($context) { return doSomething($context); }); // Stage with handler and failure callback $stage = Stage::handler(function ($context) { return doSomething($context); })->onFailure(function (\Throwable $e, $context) { // Log, send metrics, cleanup, etc. });
Accessing Failed Exceptions
When all stages fail, you can inspect all the exceptions:
try { $chain->execute(); } catch (AllStagesFailedException $e) { $exceptions = $e->getExceptions(); foreach ($exceptions as $index => $exception) { echo "Stage {$index} failed: " . $exception->getMessage() . "\n"; } }
Testing
composer test
License
The MIT License (MIT). Please see License File for more information.