Events
This is a domain events library with an included event logger. It's built with Udi Dahan's Domain Events pattern in mind.
Available via composer as mattferris/events
Event Handling
Implementation starts with creating a DomainEvents class in your domain that extends MattFerris\Events\AbstractDomainEvents.
namespace MyDomain;
class DomainEvents extends \MattFerris\Events\AbstractDomainEvents
{
}
Then you can create individual event classes either by extending MattFerris\Events\Event or implementing MattFerris\Events\EventInterface. Event is an empty class so your event class will still need to provide it's own logic.
namespace MyDomain;
class SomeEvent extends MattFerris\Events\Event
{
    protected $someArgument;
    public function __construct($someArgument)
    {
        $this->someArgument = $someArgument;
    }
    public function getSomeArgument()
    {
        return $this->someArgument;
    }
}
To raise/dispatch an event from within your domain entities, call DomainEvents::dispatch().
namespace MyDomain;
class SomeEntity
{
    public function doSomething()
    {
        // stuff happens
        // dispatch event
        DomainEvents::dispatch(new SomeEvent($argument));
    }
}
At the moment, there is no dispatcher configured so the event won't be handled by anything. Configuring a dispatcher can be done like so:
$dispatcher = new MattFerris\Events\Dispatcher();
MyDomain\DomainEvents::setDispatcher($dispatcher);
And now you can add events listeners to the dispatcher. Listeners can be any callable, and can listen for a specific event or pattern matched event. To match a specific event, use the fully qualified class name.
$dispatcher->addListener('MyDomain.SomeEvent', function ($event) { ... });
It's also possible to listen for events based on a prefix or suffix.
// listen for all events in the `MyDomain` namespace
$dispatcher->addListener('MyDomain.', $handler);
// listen for all SomeEvent events in any domain
$dispatcher->addListener('.SomeEvent', $handler);
Finally, you can listen for all events using an asterisk.
// listen for all events
$dispatcher->addListener('*', $handler);
Event Logging
Event logging is accomplished by using MattFerris\Events\Logger. The constructor takes an instance of Dispatcher and a callable that is passed the resulting log message and can then write it to a log file, stderr, etc.
$logger = new MattFerris\Events\Logger($dispatcher, function ($msg) { error_log($msg); });
That's it! All dispatched events will be logged to PHP's error log. By default, log messages are just the name of the event prefixed with event:. For example:
event: MattFerris\Events\Event
You can change the prefix using setPrefix().
$logger->setPrefix('another prefix: ');
You can use logging helpers to customize the messages for certain events. Create a class in your domain called DomainEventLoggerHelpers and have it extend MattFerris\Events\AbstractDomainEventLoggerHelpers. Then simply create static methods to be called for domain events, where each method returns the string to be logged. These methods should start with on followed by the event name e.g. onSomeEvent.
namespace MyDomain;
class DomainEventLoggerHelpers extends MattFerris\Events\AbstractDomainEventLoggerHelpers
{
    static public function onSomeEvent(SomeEvent $e)
    {
        $foo = $e->getFoo();
        $bar = $e->getBar();
        return "SomeEvent was dispatched with values foo=$foo, bar=$bar";
    }
}
To register the helpers with the logger:
MyDomain\DomainEventLoggerHelepers::addHelpers($logger);
Now, when SomeEvent is dispatched, the above helper will be called and the returned string will be logged.
event: SomeEvent was dispatched with values foo=blah, bar=bleh
In my opinion, it makes sense to have these logging helpers defined in MyDomain\DomainEventLoggerHelpers as the helpers are directly related to the events in the domain. When domain events are updated, the logging helpers can be updated and committed all together.
Comments