vendor/contao/core-bundle/src/Csrf/ContaoCsrfTokenManager.php line 98

Open in your IDE?
  1. <?php
  2. declare(strict_types=1);
  3. /*
  4.  * This file is part of Contao.
  5.  *
  6.  * (c) Leo Feyer
  7.  *
  8.  * @license LGPL-3.0-or-later
  9.  */
  10. namespace Contao\CoreBundle\Csrf;
  11. use Symfony\Component\HttpFoundation\Request;
  12. use Symfony\Component\HttpFoundation\RequestStack;
  13. use Symfony\Component\Security\Csrf\CsrfToken;
  14. use Symfony\Component\Security\Csrf\CsrfTokenManager;
  15. use Symfony\Component\Security\Csrf\TokenGenerator\TokenGeneratorInterface;
  16. use Symfony\Component\Security\Csrf\TokenStorage\TokenStorageInterface;
  17. use Symfony\Contracts\Service\ResetInterface;
  18. class ContaoCsrfTokenManager extends CsrfTokenManager implements ResetInterface
  19. {
  20.     private RequestStack $requestStack;
  21.     private string $csrfCookiePrefix;
  22.     private ?string $defaultTokenName;
  23.     /**
  24.      * @var array<int, string>
  25.      */
  26.     private array $usedTokenValues = [];
  27.     /**
  28.      * @param string|RequestStack|callable|null $namespace
  29.      */
  30.     public function __construct(RequestStack $requestStackstring $csrfCookiePrefixTokenGeneratorInterface $generator nullTokenStorageInterface $storage null$namespace nullstring $defaultTokenName null)
  31.     {
  32.         $this->requestStack $requestStack;
  33.         $this->csrfCookiePrefix $csrfCookiePrefix;
  34.         $this->defaultTokenName $defaultTokenName;
  35.         parent::__construct($generator$storage$namespace);
  36.     }
  37.     /**
  38.      * @return array<int, string>
  39.      */
  40.     public function getUsedTokenValues(): array
  41.     {
  42.         return $this->usedTokenValues;
  43.     }
  44.     public function getToken($tokenId): CsrfToken
  45.     {
  46.         $token parent::getToken($tokenId);
  47.         $this->usedTokenValues[] = $token->getValue();
  48.         return $token;
  49.     }
  50.     public function refreshToken($tokenId): CsrfToken
  51.     {
  52.         $token parent::refreshToken($tokenId);
  53.         $this->usedTokenValues[] = $token->getValue();
  54.         return $token;
  55.     }
  56.     public function isTokenValid(CsrfToken $token): bool
  57.     {
  58.         if (
  59.             ($request $this->requestStack->getMainRequest())
  60.             && 'POST' === $request->getRealMethod()
  61.             && $this->canSkipTokenValidation($request$this->csrfCookiePrefix.$token->getId())
  62.         ) {
  63.             return true;
  64.         }
  65.         return parent::isTokenValid($token);
  66.     }
  67.     /**
  68.      * Skip the CSRF token validation if the request has no cookies, no
  69.      * authenticated user and the session has not been started.
  70.      */
  71.     public function canSkipTokenValidation(Request $requeststring $tokenCookieName): bool
  72.     {
  73.         return
  74.             !$request->getUserInfo()
  75.             && (
  76.                 === $request->cookies->count()
  77.                 || [$tokenCookieName] === $request->cookies->keys()
  78.             )
  79.             && !($request->hasSession() && $request->getSession()->isStarted());
  80.     }
  81.     public function getDefaultTokenValue(): string
  82.     {
  83.         if (null === $this->defaultTokenName) {
  84.             throw new \RuntimeException('The Contao CSRF token manager was not initialized with a default token name.');
  85.         }
  86.         return $this->getToken($this->defaultTokenName)->getValue();
  87.     }
  88.     public function reset(): void
  89.     {
  90.         $this->usedTokenValues = [];
  91.     }
  92. }