123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125 |
- <?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\Http;
- use Symfony\Component\EventDispatcher\EventSubscriberInterface;
- use Symfony\Component\HttpKernel\Event\FinishRequestEvent;
- use Symfony\Component\HttpKernel\Event\RequestEvent;
- use Symfony\Component\HttpKernel\KernelEvents;
- use Symfony\Component\Security\Http\Firewall\FirewallListenerInterface;
- use Symfony\Contracts\EventDispatcher\EventDispatcherInterface;
- /**
- * Firewall uses a FirewallMap to register security listeners for the given
- * request.
- *
- * It allows for different security strategies within the same application
- * (a Basic authentication for the /api, and a web based authentication for
- * everything else for instance).
- *
- * @author Fabien Potencier <fabien@symfony.com>
- */
- class Firewall implements EventSubscriberInterface
- {
- private $map;
- private $dispatcher;
- private $exceptionListeners;
- public function __construct(FirewallMapInterface $map, EventDispatcherInterface $dispatcher)
- {
- $this->map = $map;
- $this->dispatcher = $dispatcher;
- $this->exceptionListeners = new \SplObjectStorage();
- }
- public function onKernelRequest(RequestEvent $event)
- {
- if (!$event->isMasterRequest()) {
- return;
- }
- // register listeners for this firewall
- $listeners = $this->map->getListeners($event->getRequest());
- $authenticationListeners = $listeners[0];
- $exceptionListener = $listeners[1];
- $logoutListener = $listeners[2];
- if (null !== $exceptionListener) {
- $this->exceptionListeners[$event->getRequest()] = $exceptionListener;
- $exceptionListener->register($this->dispatcher);
- }
- // Authentication listeners are pre-sorted by SortFirewallListenersPass
- $authenticationListeners = function () use ($authenticationListeners, $logoutListener) {
- if (null !== $logoutListener) {
- $logoutListenerPriority = $this->getListenerPriority($logoutListener);
- }
- foreach ($authenticationListeners as $listener) {
- $listenerPriority = $this->getListenerPriority($listener);
- // Yielding the LogoutListener at the correct position
- if (null !== $logoutListener && $listenerPriority < $logoutListenerPriority) {
- yield $logoutListener;
- $logoutListener = null;
- }
- yield $listener;
- }
- // When LogoutListener has the lowest priority of all listeners
- if (null !== $logoutListener) {
- yield $logoutListener;
- }
- };
- $this->callListeners($event, $authenticationListeners());
- }
- public function onKernelFinishRequest(FinishRequestEvent $event)
- {
- $request = $event->getRequest();
- if (isset($this->exceptionListeners[$request])) {
- $this->exceptionListeners[$request]->unregister($this->dispatcher);
- unset($this->exceptionListeners[$request]);
- }
- }
- /**
- * {@inheritdoc}
- */
- public static function getSubscribedEvents()
- {
- return [
- KernelEvents::REQUEST => ['onKernelRequest', 8],
- KernelEvents::FINISH_REQUEST => 'onKernelFinishRequest',
- ];
- }
- protected function callListeners(RequestEvent $event, iterable $listeners)
- {
- foreach ($listeners as $listener) {
- $listener($event);
- if ($event->hasResponse()) {
- break;
- }
- }
- }
- private function getListenerPriority(object $logoutListener): int
- {
- return $logoutListener instanceof FirewallListenerInterface ? $logoutListener->getPriority() : 0;
- }
- }
|