-
-
Notifications
You must be signed in to change notification settings - Fork 27
Expand file tree
/
Copy pathFailureMiddlewareDispatcher.php
More file actions
106 lines (89 loc) · 3.61 KB
/
FailureMiddlewareDispatcher.php
File metadata and controls
106 lines (89 loc) · 3.61 KB
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
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
<?php
declare(strict_types=1);
namespace Yiisoft\Queue\Middleware\FailureHandling;
use Closure;
final class FailureMiddlewareDispatcher
{
public const DEFAULT_PIPELINE = 'failure-pipeline-default';
/**
* Contains a middleware pipeline handler.
*
* @var MiddlewareFailureStack[] The middleware stack.
*/
private array $stack = [];
/**
* @param array[][]|callable[][]|MiddlewareFailureInterface[][]|string[][] $middlewareDefinitions
*/
public function __construct(
private readonly MiddlewareFactoryFailureInterface $middlewareFactory,
private array $middlewareDefinitions,
) {
$this->init();
}
/**
* Dispatch request through middleware to get response.
*
* @param FailureHandlingRequest $request Request to pass to middleware.
* @param MessageFailureHandlerInterface $finishHandler Handler to use in case no middleware produced a response.
*/
public function dispatch(
FailureHandlingRequest $request,
MessageFailureHandlerInterface $finishHandler,
): FailureHandlingRequest {
$queueName = $request->getQueue()->getName();
if (!isset($this->middlewareDefinitions[$queueName]) || $this->middlewareDefinitions[$queueName] === []) {
$queueName = self::DEFAULT_PIPELINE;
}
$definitions = array_reverse($this->middlewareDefinitions[$queueName]);
if (!isset($this->stack[$queueName])) {
$this->stack[$queueName] = new MiddlewareFailureStack($this->buildMiddlewares(...$definitions), $finishHandler);
}
return $this->stack[$queueName]->handleFailure($request);
}
/**
* Returns new instance with middleware handlers replaced with the ones provided.
* The last specified handler will be executed first.
*
* @param array[][]|callable[][]|MiddlewareFailureInterface[][]|string[][] $middlewareDefinitions Each array element is:
*
* - A name of a middleware class. The middleware instance will be obtained from container executed.
* - A callable with `function(ServerRequestInterface $request, RequestHandlerInterface $handler):
* ResponseInterface` signature.
* - A "callable-like" array in format `[FooMiddleware::class, 'index']`. `FooMiddleware` instance will
* be created and `index()` method will be executed.
* - A function returning a middleware. The middleware returned will be executed.
*
* For callables typed parameters are automatically injected using dependency injection container.
*
* @return self New instance of the {@see FailureMiddlewareDispatcher}
*/
public function withMiddlewares(array $middlewareDefinitions): self
{
$instance = clone $this;
$instance->middlewareDefinitions = $middlewareDefinitions;
// Fixes a memory leak.
unset($instance->stack);
$instance->stack = [];
$instance->init();
return $instance;
}
private function init(): void
{
if (!isset($this->middlewareDefinitions[self::DEFAULT_PIPELINE])) {
$this->middlewareDefinitions[self::DEFAULT_PIPELINE] = [];
}
}
/**
* @return Closure[]
*/
private function buildMiddlewares(array|callable|string|MiddlewareFailureInterface ...$definitions): array
{
$middlewares = [];
$factory = $this->middlewareFactory;
foreach ($definitions as $middlewareDefinition) {
$middlewares[] = static fn(): MiddlewareFailureInterface
=> $factory->createFailureMiddleware($middlewareDefinition);
}
return $middlewares;
}
}