1
|
|
|
<?php |
2
|
|
|
|
3
|
|
|
/** |
4
|
|
|
* It's free open-source software released under the MIT License. |
5
|
|
|
* |
6
|
|
|
* @author Anatoly Nekhay <[email protected]> |
7
|
|
|
* @copyright Copyright (c) 2018, Anatoly Nekhay |
8
|
|
|
* @license https://github.com/sunrise-php/http-router/blob/master/LICENSE |
9
|
|
|
* @link https://github.com/sunrise-php/http-router |
10
|
|
|
*/ |
11
|
|
|
|
12
|
|
|
declare(strict_types=1); |
13
|
|
|
|
14
|
|
|
namespace Sunrise\Http\Router\Middleware; |
15
|
|
|
|
16
|
|
|
use Psr\Http\Message\ResponseFactoryInterface; |
17
|
|
|
use Psr\Http\Message\ResponseInterface; |
18
|
|
|
use Psr\Http\Message\ServerRequestInterface; |
19
|
|
|
use Psr\Http\Message\StreamFactoryInterface; |
20
|
|
|
use Psr\Http\Server\MiddlewareInterface; |
21
|
|
|
use Psr\Http\Server\RequestHandlerInterface; |
22
|
|
|
use Psr\Log\LoggerInterface; |
23
|
|
|
use Sunrise\Coder\CodecManagerInterface; |
24
|
|
|
use Sunrise\Coder\MediaTypeInterface; |
25
|
|
|
use Sunrise\Http\Router\Dictionary\HeaderName; |
26
|
|
|
use Sunrise\Http\Router\Exception\HttpException; |
27
|
|
|
use Sunrise\Http\Router\Exception\HttpExceptionFactory; |
28
|
|
|
use Sunrise\Http\Router\LanguageInterface; |
29
|
|
|
use Sunrise\Http\Router\ServerRequest; |
30
|
|
|
use Sunrise\Http\Router\Validation\ConstraintViolationInterface; |
31
|
|
|
use Sunrise\Http\Router\View\ErrorView; |
32
|
|
|
use Sunrise\Http\Router\View\ViolationView; |
33
|
|
|
use Sunrise\Translator\TranslatorManagerInterface; |
34
|
|
|
use Throwable; |
35
|
|
|
|
36
|
|
|
use function sprintf; |
37
|
|
|
|
38
|
|
|
final class ErrorHandlingMiddleware implements MiddlewareInterface |
39
|
|
|
{ |
40
|
6 |
|
public function __construct( |
41
|
|
|
private readonly ResponseFactoryInterface $responseFactory, |
42
|
|
|
private readonly StreamFactoryInterface $streamFactory, |
43
|
|
|
private readonly CodecManagerInterface $codecManager, |
44
|
|
|
/** @var array<array-key, mixed> */ |
45
|
|
|
private readonly array $codecContext, |
46
|
|
|
/** @var list<MediaTypeInterface> */ |
47
|
|
|
private readonly array $producedMediaTypes, |
48
|
|
|
private readonly MediaTypeInterface $defaultMediaType, |
49
|
|
|
private readonly TranslatorManagerInterface $translatorManager, |
50
|
|
|
/** @var list<LanguageInterface> */ |
51
|
|
|
private readonly array $producedLanguages, |
52
|
|
|
private readonly LanguageInterface $defaultLanguage, |
53
|
|
|
private readonly LoggerInterface $logger, |
54
|
|
|
private readonly ?int $fatalErrorStatusCode = null, |
55
|
|
|
private readonly ?string $fatalErrorMessage = null, |
56
|
|
|
) { |
57
|
6 |
|
} |
58
|
|
|
|
59
|
|
|
/** |
60
|
|
|
* @inheritDoc |
61
|
|
|
*/ |
62
|
6 |
|
public function process(ServerRequestInterface $request, RequestHandlerInterface $handler): ResponseInterface |
63
|
|
|
{ |
64
|
|
|
try { |
65
|
6 |
|
return $handler->handle($request); |
66
|
5 |
|
} catch (HttpException $e) { |
67
|
1 |
|
return $this->handleHttpError($e, $request); |
68
|
4 |
|
} catch (Throwable $e) { |
69
|
4 |
|
return $this->handleFatalError($e, $request); |
70
|
|
|
} |
71
|
|
|
} |
72
|
|
|
|
73
|
5 |
|
private function handleHttpError(HttpException $error, ServerRequestInterface $request): ResponseInterface |
74
|
|
|
{ |
75
|
5 |
|
$clientPreferredMediaType = ServerRequest::create($request) |
76
|
5 |
|
->getClientPreferredMediaType(...$this->producedMediaTypes) |
77
|
5 |
|
?? $this->defaultMediaType; |
78
|
|
|
|
79
|
5 |
|
$clientPreferredLanguage = ServerRequest::create($request) |
80
|
5 |
|
->getClientPreferredLanguage(...$this->producedLanguages) |
81
|
5 |
|
?? $this->defaultLanguage; |
82
|
|
|
|
83
|
5 |
|
return $this->createErrorResponse($error, $clientPreferredMediaType, $clientPreferredLanguage); |
84
|
|
|
} |
85
|
|
|
|
86
|
4 |
|
private function handleFatalError(Throwable $error, ServerRequestInterface $request): ResponseInterface |
87
|
|
|
{ |
88
|
4 |
|
$this->logger->error($error->getMessage(), [ |
89
|
4 |
|
'error' => $error, |
90
|
4 |
|
'request' => $request, |
91
|
4 |
|
]); |
92
|
|
|
|
93
|
4 |
|
$httpError = HttpExceptionFactory::internalServerError( |
94
|
4 |
|
message: $this->fatalErrorMessage, |
95
|
4 |
|
code: $this->fatalErrorStatusCode, |
96
|
4 |
|
previous: $error, |
97
|
4 |
|
); |
98
|
|
|
|
99
|
4 |
|
return $this->handleHttpError($httpError, $request); |
100
|
|
|
} |
101
|
|
|
|
102
|
5 |
|
private function createErrorResponse( |
103
|
|
|
HttpException $error, |
104
|
|
|
MediaTypeInterface $mediaType, |
105
|
|
|
LanguageInterface $language, |
106
|
|
|
): ResponseInterface { |
107
|
5 |
|
$response = $this->responseFactory->createResponse($error->getCode()); |
108
|
5 |
|
foreach ($error->getHeaderFields() as [$fieldName, $fieldValue]) { |
109
|
1 |
|
$response = $response->withHeader($fieldName, $fieldValue); |
110
|
|
|
} |
111
|
|
|
|
112
|
5 |
|
$errorView = $this->createErrorView($error, $language); |
113
|
5 |
|
$responseContent = $this->codecManager->encode($mediaType, $errorView, $this->codecContext); |
114
|
5 |
|
$responseContentType = sprintf('%s; charset=UTF-8', $mediaType->getIdentifier()); |
115
|
|
|
|
116
|
5 |
|
return $response->withHeader(HeaderName::CONTENT_TYPE, $responseContentType) |
|
|
|
|
117
|
5 |
|
->withBody($this->streamFactory->createStream($responseContent)); |
118
|
|
|
} |
119
|
|
|
|
120
|
5 |
|
private function createErrorView( |
121
|
|
|
HttpException $error, |
122
|
|
|
LanguageInterface $language, |
123
|
|
|
): ErrorView { |
124
|
5 |
|
$message = $this->translatorManager->translate( |
125
|
5 |
|
domain: $error->getTranslationDomain(), |
126
|
5 |
|
locale: $language->getCode(), |
127
|
5 |
|
template: $error->getMessageTemplate(), |
128
|
5 |
|
placeholders: $error->getMessagePlaceholders(), |
129
|
5 |
|
); |
130
|
|
|
|
131
|
5 |
|
$violationViews = []; |
132
|
5 |
|
foreach ($error->getConstraintViolations() as $violation) { |
133
|
1 |
|
$violationViews[] = $this->createViolationView($violation, $language); |
134
|
|
|
} |
135
|
|
|
|
136
|
5 |
|
return new ErrorView( |
137
|
5 |
|
message: $message, |
138
|
5 |
|
violations: $violationViews, |
139
|
5 |
|
); |
140
|
|
|
} |
141
|
|
|
|
142
|
1 |
|
private function createViolationView( |
143
|
|
|
ConstraintViolationInterface $violation, |
144
|
|
|
LanguageInterface $language, |
145
|
|
|
): ViolationView { |
146
|
1 |
|
$message = $this->translatorManager->translate( |
147
|
1 |
|
domain: $violation->getTranslationDomain(), |
148
|
1 |
|
locale: $language->getCode(), |
149
|
1 |
|
template: $violation->getMessageTemplate(), |
150
|
1 |
|
placeholders: $violation->getMessagePlaceholders(), |
151
|
1 |
|
); |
152
|
|
|
|
153
|
1 |
|
return new ViolationView( |
154
|
1 |
|
source: $violation->getPropertyPath(), |
155
|
1 |
|
message: $message, |
156
|
1 |
|
code: $violation->getCode(), |
157
|
1 |
|
); |
158
|
|
|
} |
159
|
|
|
} |
160
|
|
|
|