micro-module / saga
Micro module Saga common library
Requires
- php: ^8.0
- ext-json: *
- league/tactician: ^1.0
- micro-module/broadway: ^2.6
- micro-module/broadway-saga: ^0.6.0
Requires (Dev)
- mockery/mockery: ^1.3
- php-parallel-lint/php-console-highlighter: ^0.4
- php-parallel-lint/php-parallel-lint: ^1.0
- phpmd/phpmd: ^2.8
- phpstan/phpstan: ^0.12
- phpstan/phpstan-mockery: ^0.12
- phpstan/phpstan-phpunit: ^0.12
- phpunit/phpunit: ^9.0
- roave/security-advisories: dev-master
- symplify/easy-coding-standard: ^9.4
- vimeo/psalm: ^4.10
This package is auto-updated.
Last update: 2026-04-12 17:44:31 UTC
README
Micro-module saga extension library. Provides abstract saga base class, DBAL saga repository, and command dispatching infrastructure following the Dependency Inversion Principle.
Installation
composer require micro-module/saga
Dispatcher Interfaces
SagaCommandDispatcherInterface
Abstraction for dispatching commands from within sagas. Two implementations are provided:
SyncSagaCommandDispatcher— dispatches synchronously via TacticianCommandBusOutboxSagaCommandDispatcher— enqueues asynchronously viaSagaCommandQueueInterface
SagaCommandQueueInterface
Abstract queue contract. The saga package does NOT depend on outbox-bundle or any concrete queue implementation. The outbox package (or any other package) implements this interface and is wired via DI.
Dependency Inversion Diagram
saga package
┌───────────────────────────────────────────────┐
│ │
│ SagaCommandDispatcherInterface │
│ ▲ ▲ │
│ │ │ │
│ OutboxSaga- SyncSaga- │
│ CommandDispatcher CommandDispatcher │
│ │ │ │
│ SagaCommandQueueInterface CommandBus │
│ │ │ │
└──────────┼──────────────┼─────────────────────┘
│ │
┌──────────┼──────────────┼─────────────────────┐
│ │ │ outbox-bundle / │
│ OutboxAwareTaskProducer tactician-bundle │
│ implements SagaCommandQueueInterface │
└───────────────────────────────────────────────┘
Key rule: The arrow from OutboxAwareTaskProducer points TO the saga package interface — the saga package has zero dependency on the outbox package. This is the Dependency Inversion Principle in action.
Usage
Sync dispatch (Tactician)
// DI wiring $dispatcher = new SyncSagaCommandDispatcher($commandBus); $dispatcher->dispatch(new CreateNewsCommand($id));
Async dispatch via outbox (DIP)
// The queue implementation (OutboxAwareTaskProducer) lives in outbox-bundle // and implements SagaCommandQueueInterface. // Wire via DI alias — never hard-code the concrete class in your saga. $dispatcher = new OutboxSagaCommandDispatcher($queue); // $queue is SagaCommandQueueInterface $dispatcher->dispatch(new PublishNewsCommand($id)); // command must implement Serializable
Requirements
- PHP 8.0+
league/tactician^1.0micro-module/broadway^2.6 (providesBroadway\Serializer\Serializable)