vendor/symfony/framework-bundle/Routing/Router.php line 52

Open in your IDE?
  1. <?php
  2. /*
  3.  * This file is part of the Symfony package.
  4.  *
  5.  * (c) Fabien Potencier <fabien@symfony.com>
  6.  *
  7.  * For the full copyright and license information, please view the LICENSE
  8.  * file that was distributed with this source code.
  9.  */
  10. namespace Symfony\Bundle\FrameworkBundle\Routing;
  11. use Psr\Container\ContainerInterface;
  12. use Psr\Log\LoggerInterface;
  13. use Symfony\Component\Config\Loader\LoaderInterface;
  14. use Symfony\Component\DependencyInjection\Config\ContainerParametersResource;
  15. use Symfony\Component\DependencyInjection\ContainerInterface as SymfonyContainerInterface;
  16. use Symfony\Component\DependencyInjection\Exception\ParameterNotFoundException;
  17. use Symfony\Component\DependencyInjection\Exception\RuntimeException;
  18. use Symfony\Component\DependencyInjection\ServiceSubscriberInterface;
  19. use Symfony\Component\HttpKernel\CacheWarmer\WarmableInterface;
  20. use Symfony\Component\Routing\RequestContext;
  21. use Symfony\Component\Routing\RouteCollection;
  22. use Symfony\Component\Routing\Router as BaseRouter;
  23. /**
  24.  * This Router creates the Loader only when the cache is empty.
  25.  *
  26.  * @author Fabien Potencier <fabien@symfony.com>
  27.  */
  28. class Router extends BaseRouter implements WarmableInterfaceServiceSubscriberInterface
  29. {
  30.     private $container;
  31.     private $collectedParameters = [];
  32.     private $paramFetcher;
  33.     /**
  34.      * @param ContainerInterface      $container  A ContainerInterface instance
  35.      * @param mixed                   $resource   The main resource to load
  36.      * @param array                   $options    An array of options
  37.      * @param RequestContext          $context    The context
  38.      * @param ContainerInterface|null $parameters A ContainerInterface instance allowing to fetch parameters
  39.      * @param LoggerInterface|null    $logger
  40.      */
  41.     public function __construct(ContainerInterface $container$resource, array $options = [], RequestContext $context nullContainerInterface $parameters nullLoggerInterface $logger nullstring $defaultLocale null)
  42.     {
  43.         $this->container $container;
  44.         $this->resource $resource;
  45.         $this->context $context ?: new RequestContext();
  46.         $this->logger $logger;
  47.         $this->setOptions($options);
  48.         if ($parameters) {
  49.             $this->paramFetcher = [$parameters'get'];
  50.         } elseif ($container instanceof SymfonyContainerInterface) {
  51.             $this->paramFetcher = [$container'getParameter'];
  52.         } else {
  53.             throw new \LogicException(sprintf('You should either pass a "%s" instance or provide the $parameters argument of the "%s" method.'SymfonyContainerInterface::class, __METHOD__));
  54.         }
  55.         $this->defaultLocale $defaultLocale;
  56.     }
  57.     /**
  58.      * {@inheritdoc}
  59.      */
  60.     public function getRouteCollection()
  61.     {
  62.         if (null === $this->collection) {
  63.             $this->collection $this->container->get('routing.loader')->load($this->resource$this->options['resource_type']);
  64.             $this->resolveParameters($this->collection);
  65.             $this->collection->addResource(new ContainerParametersResource($this->collectedParameters));
  66.         }
  67.         return $this->collection;
  68.     }
  69.     /**
  70.      * {@inheritdoc}
  71.      */
  72.     public function warmUp($cacheDir)
  73.     {
  74.         $currentDir $this->getOption('cache_dir');
  75.         // force cache generation
  76.         $this->setOption('cache_dir'$cacheDir);
  77.         $this->getMatcher();
  78.         $this->getGenerator();
  79.         $this->setOption('cache_dir'$currentDir);
  80.     }
  81.     /**
  82.      * Replaces placeholders with service container parameter values in:
  83.      * - the route defaults,
  84.      * - the route requirements,
  85.      * - the route path,
  86.      * - the route host,
  87.      * - the route schemes,
  88.      * - the route methods.
  89.      */
  90.     private function resolveParameters(RouteCollection $collection)
  91.     {
  92.         foreach ($collection as $route) {
  93.             foreach ($route->getDefaults() as $name => $value) {
  94.                 $route->setDefault($name$this->resolve($value));
  95.             }
  96.             foreach ($route->getRequirements() as $name => $value) {
  97.                 $route->setRequirement($name$this->resolve($value));
  98.             }
  99.             $route->setPath($this->resolve($route->getPath()));
  100.             $route->setHost($this->resolve($route->getHost()));
  101.             $schemes = [];
  102.             foreach ($route->getSchemes() as $scheme) {
  103.                 $schemes array_merge($schemesexplode('|'$this->resolve($scheme)));
  104.             }
  105.             $route->setSchemes($schemes);
  106.             $methods = [];
  107.             foreach ($route->getMethods() as $method) {
  108.                 $methods array_merge($methodsexplode('|'$this->resolve($method)));
  109.             }
  110.             $route->setMethods($methods);
  111.             $route->setCondition($this->resolve($route->getCondition()));
  112.         }
  113.     }
  114.     /**
  115.      * Recursively replaces placeholders with the service container parameters.
  116.      *
  117.      * @param mixed $value The source which might contain "%placeholders%"
  118.      *
  119.      * @return mixed The source with the placeholders replaced by the container
  120.      *               parameters. Arrays are resolved recursively.
  121.      *
  122.      * @throws ParameterNotFoundException When a placeholder does not exist as a container parameter
  123.      * @throws RuntimeException           When a container value is not a string or a numeric value
  124.      */
  125.     private function resolve($value)
  126.     {
  127.         if (\is_array($value)) {
  128.             foreach ($value as $key => $val) {
  129.                 $value[$key] = $this->resolve($val);
  130.             }
  131.             return $value;
  132.         }
  133.         if (!\is_string($value)) {
  134.             return $value;
  135.         }
  136.         $escapedValue preg_replace_callback('/%%|%([^%\s]++)%/', function ($match) use ($value) {
  137.             // skip %%
  138.             if (!isset($match[1])) {
  139.                 return '%%';
  140.             }
  141.             if (preg_match('/^env\(\w+\)$/'$match[1])) {
  142.                 throw new RuntimeException(sprintf('Using "%%%s%%" is not allowed in routing configuration.'$match[1]));
  143.             }
  144.             $resolved = ($this->paramFetcher)($match[1]);
  145.             if (\is_string($resolved) || is_numeric($resolved)) {
  146.                 $this->collectedParameters[$match[1]] = $resolved;
  147.                 return (string) $resolved;
  148.             }
  149.             throw new RuntimeException(sprintf('The container parameter "%s", used in the route configuration value "%s", must be a string or numeric, but it is of type %s.'$match[1], $value, \gettype($resolved)));
  150.         }, $value);
  151.         return str_replace('%%''%'$escapedValue);
  152.     }
  153.     /**
  154.      * {@inheritdoc}
  155.      */
  156.     public static function getSubscribedServices()
  157.     {
  158.         return [
  159.             'routing.loader' => LoaderInterface::class,
  160.         ];
  161.     }
  162. }