123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167 |
- <?php
- /*
- * This file is part of the Symfony package.
- *
- * (c) Fabien Potencier <fabien@symfony.com>
- *
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- */
- namespace Symfony\Component\Security\Guard\Provider;
- use Symfony\Component\Security\Core\Authentication\Provider\AuthenticationProviderInterface;
- use Symfony\Component\Security\Core\Authentication\Token\TokenInterface;
- use Symfony\Component\Security\Core\Encoder\UserPasswordEncoderInterface;
- use Symfony\Component\Security\Core\Exception\AuthenticationException;
- use Symfony\Component\Security\Core\Exception\AuthenticationExpiredException;
- use Symfony\Component\Security\Core\Exception\BadCredentialsException;
- use Symfony\Component\Security\Core\Exception\UsernameNotFoundException;
- use Symfony\Component\Security\Core\User\PasswordUpgraderInterface;
- use Symfony\Component\Security\Core\User\UserCheckerInterface;
- use Symfony\Component\Security\Core\User\UserInterface;
- use Symfony\Component\Security\Core\User\UserProviderInterface;
- use Symfony\Component\Security\Guard\AuthenticatorInterface;
- use Symfony\Component\Security\Guard\PasswordAuthenticatedInterface;
- use Symfony\Component\Security\Guard\Token\GuardTokenInterface;
- use Symfony\Component\Security\Guard\Token\PreAuthenticationGuardToken;
- /**
- * Responsible for accepting the PreAuthenticationGuardToken and calling
- * the correct authenticator to retrieve the authenticated token.
- *
- * @author Ryan Weaver <ryan@knpuniversity.com>
- */
- class GuardAuthenticationProvider implements AuthenticationProviderInterface
- {
- /**
- * @var AuthenticatorInterface[]
- */
- private $guardAuthenticators;
- private $userProvider;
- private $providerKey;
- private $userChecker;
- private $passwordEncoder;
- /**
- * @param iterable|AuthenticatorInterface[] $guardAuthenticators The authenticators, with keys that match what's passed to GuardAuthenticationListener
- * @param string $providerKey The provider (i.e. firewall) key
- */
- public function __construct(iterable $guardAuthenticators, UserProviderInterface $userProvider, string $providerKey, UserCheckerInterface $userChecker, UserPasswordEncoderInterface $passwordEncoder = null)
- {
- $this->guardAuthenticators = $guardAuthenticators;
- $this->userProvider = $userProvider;
- $this->providerKey = $providerKey;
- $this->userChecker = $userChecker;
- $this->passwordEncoder = $passwordEncoder;
- }
- /**
- * Finds the correct authenticator for the token and calls it.
- *
- * @param GuardTokenInterface $token
- *
- * @return TokenInterface
- */
- public function authenticate(TokenInterface $token)
- {
- if (!$token instanceof GuardTokenInterface) {
- throw new \InvalidArgumentException('GuardAuthenticationProvider only supports GuardTokenInterface.');
- }
- if (!$token instanceof PreAuthenticationGuardToken) {
- /*
- * The listener *only* passes PreAuthenticationGuardToken instances.
- * This means that an authenticated token (e.g. PostAuthenticationGuardToken)
- * is being passed here, which happens if that token becomes
- * "not authenticated" (e.g. happens if the user changes between
- * requests). In this case, the user should be logged out, so
- * we will return an AnonymousToken to accomplish that.
- */
- // this should never happen - but technically, the token is
- // authenticated... so it could just be returned
- if ($token->isAuthenticated()) {
- return $token;
- }
- // this causes the user to be logged out
- throw new AuthenticationExpiredException();
- }
- $guardAuthenticator = $this->findOriginatingAuthenticator($token);
- if (null === $guardAuthenticator) {
- throw new AuthenticationException(sprintf('Token with provider key "%s" did not originate from any of the guard authenticators of provider "%s".', $token->getGuardProviderKey(), $this->providerKey));
- }
- return $this->authenticateViaGuard($guardAuthenticator, $token);
- }
- private function authenticateViaGuard(AuthenticatorInterface $guardAuthenticator, PreAuthenticationGuardToken $token): GuardTokenInterface
- {
- // get the user from the GuardAuthenticator
- $user = $guardAuthenticator->getUser($token->getCredentials(), $this->userProvider);
- if (null === $user) {
- $e = new UsernameNotFoundException(sprintf('Null returned from "%s::getUser()".', get_debug_type($guardAuthenticator)));
- $e->setUsername($token->getUsername());
- throw $e;
- }
- if (!$user instanceof UserInterface) {
- throw new \UnexpectedValueException(sprintf('The "%s::getUser()" method must return a UserInterface. You returned "%s".', get_debug_type($guardAuthenticator), get_debug_type($user)));
- }
- $this->userChecker->checkPreAuth($user);
- if (true !== $checkCredentialsResult = $guardAuthenticator->checkCredentials($token->getCredentials(), $user)) {
- if (false !== $checkCredentialsResult) {
- throw new \TypeError(sprintf('"%s::checkCredentials()" must return a boolean value.', get_debug_type($guardAuthenticator)));
- }
- throw new BadCredentialsException(sprintf('Authentication failed because "%s::checkCredentials()" did not return true.', get_debug_type($guardAuthenticator)));
- }
- if ($this->userProvider instanceof PasswordUpgraderInterface && $guardAuthenticator instanceof PasswordAuthenticatedInterface && null !== $this->passwordEncoder && (null !== $password = $guardAuthenticator->getPassword($token->getCredentials())) && method_exists($this->passwordEncoder, 'needsRehash') && $this->passwordEncoder->needsRehash($user)) {
- $this->userProvider->upgradePassword($user, $this->passwordEncoder->encodePassword($user, $password));
- }
- $this->userChecker->checkPostAuth($user);
- // turn the UserInterface into a TokenInterface
- $authenticatedToken = $guardAuthenticator->createAuthenticatedToken($user, $this->providerKey);
- if (!$authenticatedToken instanceof TokenInterface) {
- throw new \UnexpectedValueException(sprintf('The "%s::createAuthenticatedToken()" method must return a TokenInterface. You returned "%s".', get_debug_type($guardAuthenticator), get_debug_type($authenticatedToken)));
- }
- return $authenticatedToken;
- }
- private function findOriginatingAuthenticator(PreAuthenticationGuardToken $token): ?AuthenticatorInterface
- {
- // find the *one* GuardAuthenticator that this token originated from
- foreach ($this->guardAuthenticators as $key => $guardAuthenticator) {
- // get a key that's unique to *this* guard authenticator
- // this MUST be the same as GuardAuthenticationListener
- $uniqueGuardKey = $this->providerKey.'_'.$key;
- if ($uniqueGuardKey === $token->getGuardProviderKey()) {
- return $guardAuthenticator;
- }
- }
- // no matching authenticator found - but there will be multiple GuardAuthenticationProvider
- // instances that will be checked if you have multiple firewalls.
- return null;
- }
- public function supports(TokenInterface $token)
- {
- if ($token instanceof PreAuthenticationGuardToken) {
- return null !== $this->findOriginatingAuthenticator($token);
- }
- return $token instanceof GuardTokenInterface;
- }
- }
|