<?php
declare(strict_types=1);
/*
* This file is part of Contao.
*
* (c) Leo Feyer
*
* @license LGPL-3.0-or-later
*/
namespace Contao\CoreBundle\Csrf;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpFoundation\RequestStack;
use Symfony\Component\Security\Csrf\CsrfToken;
use Symfony\Component\Security\Csrf\CsrfTokenManager;
use Symfony\Component\Security\Csrf\TokenGenerator\TokenGeneratorInterface;
use Symfony\Component\Security\Csrf\TokenStorage\TokenStorageInterface;
use Symfony\Contracts\Service\ResetInterface;
class ContaoCsrfTokenManager extends CsrfTokenManager implements ResetInterface
{
private RequestStack $requestStack;
private string $csrfCookiePrefix;
private ?string $defaultTokenName;
/**
* @var array<int, string>
*/
private array $usedTokenValues = [];
/**
* @param string|RequestStack|callable|null $namespace
*/
public function __construct(RequestStack $requestStack, string $csrfCookiePrefix, TokenGeneratorInterface $generator = null, TokenStorageInterface $storage = null, $namespace = null, string $defaultTokenName = null)
{
$this->requestStack = $requestStack;
$this->csrfCookiePrefix = $csrfCookiePrefix;
$this->defaultTokenName = $defaultTokenName;
parent::__construct($generator, $storage, $namespace);
}
/**
* @return array<int, string>
*/
public function getUsedTokenValues(): array
{
return $this->usedTokenValues;
}
public function getToken($tokenId): CsrfToken
{
$token = parent::getToken($tokenId);
$this->usedTokenValues[] = $token->getValue();
return $token;
}
public function refreshToken($tokenId): CsrfToken
{
$token = parent::refreshToken($tokenId);
$this->usedTokenValues[] = $token->getValue();
return $token;
}
public function isTokenValid(CsrfToken $token): bool
{
if (
($request = $this->requestStack->getMainRequest())
&& 'POST' === $request->getRealMethod()
&& $this->canSkipTokenValidation($request, $this->csrfCookiePrefix.$token->getId())
) {
return true;
}
return parent::isTokenValid($token);
}
/**
* Skip the CSRF token validation if the request has no cookies, no
* authenticated user and the session has not been started.
*/
public function canSkipTokenValidation(Request $request, string $tokenCookieName): bool
{
return
!$request->getUserInfo()
&& (
0 === $request->cookies->count()
|| [$tokenCookieName] === $request->cookies->keys()
)
&& !($request->hasSession() && $request->getSession()->isStarted());
}
public function getDefaultTokenValue(): string
{
if (null === $this->defaultTokenName) {
throw new \RuntimeException('The Contao CSRF token manager was not initialized with a default token name.');
}
return $this->getToken($this->defaultTokenName)->getValue();
}
public function reset(): void
{
$this->usedTokenValues = [];
}
}