-
Notifications
You must be signed in to change notification settings - Fork 5
/
AsyncEventDispatcherMethods.php
84 lines (75 loc) · 2.13 KB
/
AsyncEventDispatcherMethods.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
<?php
/*
* This file is part of the DriftPHP Project
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*
* Feel free to edit as you please, and have fun.
*
* @author Marc Morera <[email protected]>
*/
declare(strict_types=1);
namespace Drift\HttpKernel;
use Drift\HttpKernel\Event\DomainEventEnvelope;
use React\Promise\PromiseInterface;
use function React\Promise\resolve;
use Symfony\Contracts\EventDispatcher\Event;
/**
* Trait AsyncEventDispatcherMethods.
*/
trait AsyncEventDispatcherMethods
{
/**
* @param object $event
* @param string|null $eventName
*
* @return PromiseInterface
*/
public function asyncDispatch(
$event,
string $eventName = null
): PromiseInterface {
$eventName = $eventName ?? \get_class($event);
$dispatchableEvent = $event instanceof Event
? $event
: new DomainEventEnvelope($event);
if ($listeners = $this->getListeners($eventName)) {
return $this
->doAsyncDispatch($listeners, $eventName, $dispatchableEvent)
->then(function () use ($event) {
return $event;
});
}
return resolve($event);
}
/**
* Triggers the listeners of an event.
*
* This method can be overridden to add functionality that is executed
* for each listener.
*
* @param callable[] $listeners
* @param string $eventName
* @param Event $event
*
* @return PromiseInterface
*/
private function doAsyncDispatch(
array $listeners,
string $eventName,
Event $event
) {
$promise = resolve(null);
foreach ($listeners as $listener) {
$promise = $promise->then(function () use ($event, $eventName, $listener) {
return
resolve($event->isPropagationStopped()
? null
: $listener($event, $eventName, $this)
);
});
}
return $promise;
}
}