Opinionated message bus built on top of symfony/messenger largely based on the ideas and code base of damejidlo/message-bus, originally developed by Ondřej Bouda.
Via Composer:
$ composer require nepada/message-busWe define two types of messages and corresponding message buses - commands and events.
Command implementation must adhere to these rules:
- class must implement
Nepada\Commands\Commandinterface - class must be named
<command-name>Command - class must be final
- class must be readonly
- command name should be in imperative form ("do something")
- command must be a simple immutable DTO
- command must not contain entities, only references (i.e.
int $orderId, notOrder $order)
Examples of good command class names:
RejectOrderCommandCreateUserCommand
Command handler implementation must adhere to these rules:
- class must implement
Nepada\Commands\CommandHandlerinterface - class must be named
<command-name>Handler - class must be final
- class must implement method named
__invoke __invokemethod must have exactly one parameter named$command__invokemethod parameter must be typehinted with specific command class__invokemethod return type must bevoid__invokemethod must be annotated with@throwstags if specific exceptions can be thrown
Example:
final class DoSomethingHandler implements \Nepada\MessageBus\Commands\CommandHandler
{
/**
* @throws SomeException
*/
public function __invoke(DoSomethingCommand $command): void
{
// ...
}
}Every command must have exactly one handler.
Events must be dispatched during command handling only.
Event implementation must adhere to these rules:
- class must implement
Nepada\Events\Eventinterface - class must be named
<event-name>Event - class must be final
- class must be readonly
- event name should be in past tense ("something happened")
- event must be a simple immutable DTO
- event must not contain entities, only references (i.e.
int $orderId, notOrder $order)
Examples of good event class names:
OrderRejectedEventUserRegisteredEvent
Event subscriber implementation must adhere to these rules:
- class must implement
Nepada\Events\EventSubscriberinterface - class must be named
<do-something>On<event-name> - class must be final
- class must implement method named
__invoke __invokemethod must have exactly one parameter named$event__invokemethod parameter must be typehinted with specific event class__invokemethod return type must bevoid__invokemethod must be annotated with@throwstags if specific exceptions can be thrown
Example:
final class DoSomethingOnSomethingHappened implements \Nepada\MessageBus\Events\EventSubscriber
{
public function __invoke(SomethingHappenedEvent $event): void {}
}Every event may have any number of subscribers, or none at all.
Most of the conventions described above may be enforced by static analysis. The analysis should be run during the compilation of DI container, triggering it at application runtime is not recommended.
use Nepada\MessageBus\StaticAnalysis\ConfigurableHandlerValidator;
use Nepada\MessageBus\StaticAnalysis\HandlerType;
use Nepada\MessageBus\StaticAnalysis\MessageHandlerValidationConfiguration;
// Validate command handler
$commandHandlerType = HandlerType::fromString(DoSomethingHandler::class);
$commandHandlerConfiguration = MessageHandlerValidationConfiguration::command();
$commandHandlerValidator = new ConfigurableHandlerValidator($commandHandlerConfiguration);
$commandHandlerValidator->validate($commandHandlerType);
// Validate event subscriber
$eventSubscriberType = HandlerType::fromString(DoSomethingOnSomethingHappened::class);
$eventSubscriberConfiguration = MessageHandlerValidationConfiguration::event();
$eventSubscriberValidator = new ConfigurableHandlerValidator($eventSubscriberConfiguration);
$eventSubscriberValidator->validate($eventSubscriberType);To maintain backwards compatibility new rules are not enforced by default. They can be enabled by passing $bleedingEdge flag when creating validator configuration, e.g. MessageHandlerValidationConfiguration::command(true).
These rules will be enabled in the next major version: (none at the moment)
Use MessageTypeExtractor to retrieve the message type that a given command handler or event subscriber handles:
use Nepada\MessageBus\StaticAnalysis\HandlerType;
use Nepada\MessageBus\StaticAnalysis\MessageTypeExtractor;
// Extracting handled message type
$messageTypeExtractor = new MessageTypeExtractor();
$commandHandlerType = HandlerType::fromString(DoSomethingHandler::class);
$messageTypeExtractor->extract($commandHandlerType); // MessageType instance for DoSomethingCommandLoggingMiddleware implements logging into standard PSR-3 logger.
Start of message handling and its success or failure are logged separately.
Logging context is filled with the extracted attributes of command or event DTO.
Generally, it's not a good idea to execute commands from within another command handler.
You can completely forbid this behavior with PreventNestedHandlingMiddleware.
It is completely up to you to use the provided building blocks together with Symfony Messenger and configure one or more instances of command and/or event buses.
A minimal setup in pure PHP might look something like this:
use Nepada\MessageBus\Commands\CommandHandlerLocator;
use Nepada\MessageBus\Commands\MessengerCommandBus;
use Nepada\MessageBus\Events\EventSubscribersLocator;
use Nepada\MessageBus\Events\MessengerEventDispatcher;
use Nepada\MessageBus\Logging\LogMessageResolver;
use Nepada\MessageBus\Logging\MessageContextResolver;
use Nepada\MessageBus\Logging\PrivateClassPropertiesExtractor;
use Nepada\MessageBus\Middleware\LoggingMiddleware;
use Nepada\MessageBus\Middleware\PreventNestedHandlingMiddleware;
use Symfony\Component\Messenger\MessageBus;
use Symfony\Component\Messenger\Middleware\DispatchAfterCurrentBusMiddleware;
use Symfony\Component\Messenger\Middleware\HandleMessageMiddleware;
$dispatchAfterCurrentBusMiddleware = new DispatchAfterCurrentBusMiddleware();
$preventNestedHandlingMiddleware = new PreventNestedHandlingMiddleware();
$loggingMiddleware = new LoggingMiddleware(
new LogMessageResolver(),
new MessageContextResolver(
new PrivateClassPropertiesExtractor(),
),
$psrLogger,
);
$handleCommandMiddleware = new HandleMessageMiddleware(
new CommandHandlerLocator(
$psrContainer,
[
DoSomethingCommand::class => 'doSomethingHandlerServiceName',
],
),
);
$handleEventMiddleware = new HandleMessageMiddleware(
new EventSubscribersLocator(
$psrContainer,
[
SomethingHappenedEvent::class => [
'doSomethingOnSomethingHappenedServiceName',
'doSomethingElseOnSomethingHappenedServiceName',
],
],
),
);
$eventDispatcher = new MessengerEventDispatcher(
new MessageBus([
$dispatchAfterCurrentBusMiddleware,
$loggingMiddleware,
$handleEventMiddleware,
]),
);
$commandBus = new MessengerCommandBus(
new MessageBus([
$dispatchAfterCurrentBusMiddleware,
$loggingMiddleware,
$preventNestedHandlingMiddleware,
$handleCommandMiddleware,
]),
);Note the usage of DispatchAfterCurrentBusMiddleware - this is necessary to ensure that events produced during the handling of a command are handled only after the command handling successfully finishes.
For Nette Framework integration, consider using nepada/message-bus-nette.
- nepada/message-bus-doctrine Doctrine ORM integration - transaction handling, collecting and emitting domain events from entities, etc.
- nepada/message-bus-nette Nette Framework DI extension.
- nepada/phpstan-message-bus adding support for propagating checked exceptions thrown out of command handlers up to the command bus caller
Static analysis part of the code base and a lot of other core ideas are borrowed from damejidlo/message-bus, originally developed by Ondřej Bouda.