This package is an extension to the MessageBus package by @matthiasnoback intended to build query buses.
It allows you to create a message bus that will catch the return value of the handling of the given message.
To install the latest stable version of this component, open a console and execute the following command:
$ composer require ajgl/simple-bus-query-bus
The classes and interfaces from this package can be used to set up a query bus. The characteristics of a query bus are:
- It handles queries, i.e. interrogative messages
- Queries are handled by exactly one query handler
- The behavior of the query bus is extensible: middlewares are allowed to do things before or after handling a query
- To get the query result back, a second parameter has to be passed to the query bus to fill it with the query result.
At least we need an instance of CatchReturnMessageBusSupportingMiddleware
:
use Ajgl\SimpleBus\Message\Bus\Middleware\CatchReturnMessageBusSupportingMiddleware;
$query = new CatchReturnMessageBusSupportingMiddleware();
Now we also want queries to be handled by exactly one query handler (which can be any callable). We first need to define the collection of handlers that are available in the application. We should make this query handler map lazy-loading, or every query handler will be fully loaded, even though it is not going to be used:
use SimpleBus\Message\CallableResolver\CallableMap;
use SimpleBus\Message\CallableResolver\ServiceLocatorAwareCallableResolver;
// Provide a map of query names to callables. You can provide actual callables, or lazy-loading ones.
$queryHandlersByQueryName = [
'Fully\Qualified\Class\Name\Of\Query' => ... // a "callable"
];
Each of the provided "callables" can be one of the following things:
- An actual PHP callable,
- A service id (string) which the service locator (see below) can resolve to a PHP callable,
- An array of which the first value is a service id (string), which the service locator can resolve to a regular object, and the second value is a method name.
For backwards compatibility an object with a handle()
method also counts as a "callable".
// Provide a service locator callable. It will be used to instantiate a handler service whenever requested.
$serviceLocator = function ($serviceId) {
$handler = ...;
return $handler;
}
$queryHandlerMap = new CallableMap(
$queryHandlersByQueryName,
new ServiceLocatorAwareCallableResolver($serviceLocator)
);
First we need a way to resolve the name of a query. You can use the fully-qualified class name (FQCN) of a query object as its name:
use SimpleBus\Message\Name\ClassBasedNameResolver;
$queryNameResolver = new ClassBasedNameResolver();
Or you can ask query objects what their name is:
use SimpleBus\Message\Name\NamedMessageNameResolver;
$queryNameResolver = new NamedMessageNameResolver();
In that case your queries have to implement NamedMessage
:
use SimpleBus\Message\Name\NamedMessage;
class YourQuery implements NamedMessage
{
public static function messageName()
{
return 'your_query';
}
}
If you want to use another rule to determine the name of a query, create a class that implements
SimpleBus\Message\Name\MessageNameResolver
.
Using the MessageNameResolver
of your choice, you can now let the query handler resolver find the right query
handler for a given query.
use SimpleBus\Message\Handler\Resolver\NameBasedMessageHandlerResolver;
$queryHandlerResolver = new NameBasedMessageHandlerResolver(
$queryNameResolver,
$queryHandlerMap
);
Finally, we should add some middleware to the query bus that calls the resolved query handler and catchs the handler result:
use Ajgl\SimpleBus\Message\Handler\DelegatesToMessageHandlerAndCatchReturnMiddleware;
$queryBus->appendMiddleware(
new DelegatesToMessageHandlerAndCatchReturnMiddleware(
$queryHandlerResolver
)
);
Consider the following query:
class FindUserByEmailAddress
{
private $emailAddress;
public function __construct($emailAddress)
{
$this->emailAddress = $emailAddress;
}
public function emailAddress()
{
return $this->emailAddress;
}
}
This query communicates the intention to "find an user by email address". The message data consists of an email address. This information is required to execute the desired behavior.
The handler for this query looks like this:
class FindUserByEmailAddressQueryHandler
{
...
public function handle(FindUserByEmailAddress $query)
{
$user = $this->userRepository->findOneByEmailAddress(
$query->emailAddress()
);
return $user;
}
}
We should register this handler as a service and add the service id to the query handler map.
Since we have already fully configured the query bus, we can just start creating a new query object and let the
query bus handle it. Eventually the query will be passed as a message to the FindUserByEmailAddressQueryHandler
:
$query = new FindUserByEmailAddress(
'aj@garcialagar.es'
);
$queryBus->handle($query, $queryResult);
Once handled the query, the $queryResult
variable will contain the query result (the user found).
This component is under the MIT license. See the complete license in the LICENSE file.
Issues and feature requests are tracked in the Github issue tracker.
Developed with ♥ by Antonio J. García Lagar.
If you find this component useful, please add a ★ in the GitHub repository page and/or the Packagist package page.