91 lines
2.3 KiB
PHP
91 lines
2.3 KiB
PHP
![]() |
<?php
|
||
|
|
||
|
namespace Engelsystem\Middleware;
|
||
|
|
||
|
use Psr\Http\Message\ResponseInterface;
|
||
|
use Psr\Http\Message\ServerRequestInterface;
|
||
|
use Psr\Http\Server\MiddlewareInterface;
|
||
|
use Psr\Http\Server\RequestHandlerInterface;
|
||
|
use Symfony\Component\HttpFoundation\Session\SessionInterface;
|
||
|
|
||
|
class VerifyCsrfToken implements MiddlewareInterface
|
||
|
{
|
||
|
/** @var SessionInterface */
|
||
|
protected $session;
|
||
|
|
||
|
/**
|
||
|
* @param SessionInterface $session
|
||
|
*/
|
||
|
public function __construct(SessionInterface $session)
|
||
|
{
|
||
|
$this->session = $session;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Verify csrf tokens
|
||
|
*
|
||
|
* @param ServerRequestInterface $request
|
||
|
* @param RequestHandlerInterface $handler
|
||
|
* @return ResponseInterface
|
||
|
*/
|
||
|
public function process(ServerRequestInterface $request, RequestHandlerInterface $handler): ResponseInterface
|
||
|
{
|
||
|
if (
|
||
|
$this->isReading($request)
|
||
|
|| $this->tokensMatch($request)
|
||
|
) {
|
||
|
return $handler->handle($request);
|
||
|
}
|
||
|
|
||
|
return $this->notAuthorizedResponse();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param ServerRequestInterface $request
|
||
|
* @return bool
|
||
|
*/
|
||
|
protected function isReading(ServerRequestInterface $request): bool
|
||
|
{
|
||
|
return in_array(
|
||
|
$request->getMethod(),
|
||
|
['GET', 'HEAD', 'OPTIONS']
|
||
|
);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param ServerRequestInterface $request
|
||
|
* @return bool
|
||
|
*/
|
||
|
protected function tokensMatch(ServerRequestInterface $request): bool
|
||
|
{
|
||
|
$token = null;
|
||
|
$body = $request->getParsedBody();
|
||
|
$header = $request->getHeader('X-CSRF-TOKEN');
|
||
|
|
||
|
if (is_array($body) && isset($body['_token'])) {
|
||
|
$token = $body['_token'];
|
||
|
}
|
||
|
|
||
|
if (!empty($header)) {
|
||
|
$header = array_shift($header);
|
||
|
}
|
||
|
|
||
|
$token = $token ?: $header;
|
||
|
$sessionToken = $this->session->get('_token');
|
||
|
|
||
|
return is_string($token)
|
||
|
&& is_string($sessionToken)
|
||
|
&& hash_equals($sessionToken, $token);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @return ResponseInterface
|
||
|
* @codeCoverageIgnore
|
||
|
*/
|
||
|
protected function notAuthorizedResponse(): ResponseInterface
|
||
|
{
|
||
|
// The 419 code is used as "Page Expired" to differentiate from a 401 (not authorized)
|
||
|
return response()->withStatus(419, 'Authentication Token Mismatch');
|
||
|
}
|
||
|
}
|