vendor/symfony/dependency-injection/Loader/PhpFileLoader.php line 146

  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\Component\DependencyInjection\Loader;
  11. use Symfony\Component\Config\Builder\ConfigBuilderGenerator;
  12. use Symfony\Component\Config\Builder\ConfigBuilderGeneratorInterface;
  13. use Symfony\Component\Config\Builder\ConfigBuilderInterface;
  14. use Symfony\Component\Config\FileLocatorInterface;
  15. use Symfony\Component\DependencyInjection\Attribute\When;
  16. use Symfony\Component\DependencyInjection\Container;
  17. use Symfony\Component\DependencyInjection\ContainerBuilder;
  18. use Symfony\Component\DependencyInjection\Exception\InvalidArgumentException;
  19. use Symfony\Component\DependencyInjection\Extension\ConfigurationExtensionInterface;
  20. use Symfony\Component\DependencyInjection\Extension\ExtensionInterface;
  21. use Symfony\Component\DependencyInjection\Loader\Configurator\ContainerConfigurator;
  22. /**
  23.  * PhpFileLoader loads service definitions from a PHP file.
  24.  *
  25.  * The PHP file is required and the $container variable can be
  26.  * used within the file to change the container.
  27.  *
  28.  * @author Fabien Potencier <fabien@symfony.com>
  29.  */
  30. class PhpFileLoader extends FileLoader
  31. {
  32.     protected $autoRegisterAliasesForSinglyImplementedInterfaces false;
  33.     private ?ConfigBuilderGeneratorInterface $generator;
  34.     public function __construct(ContainerBuilder $containerFileLocatorInterface $locatorstring $env nullConfigBuilderGeneratorInterface $generator null)
  35.     {
  36.         parent::__construct($container$locator$env);
  37.         $this->generator $generator;
  38.     }
  39.     public function load(mixed $resourcestring $type null): mixed
  40.     {
  41.         // the container and loader variables are exposed to the included file below
  42.         $container $this->container;
  43.         $loader $this;
  44.         $path $this->locator->locate($resource);
  45.         $this->setCurrentDir(\dirname($path));
  46.         $this->container->fileExists($path);
  47.         // the closure forbids access to the private scope in the included file
  48.         $load \Closure::bind(function ($path$env) use ($container$loader$resource$type) {
  49.             return include $path;
  50.         }, $thisProtectedPhpFileLoader::class);
  51.         try {
  52.             $callback $load($path$this->env);
  53.             if (\is_object($callback) && \is_callable($callback)) {
  54.                 $this->executeCallback($callback, new ContainerConfigurator($this->container$this$this->instanceof$path$resource$this->env), $path);
  55.             }
  56.         } finally {
  57.             $this->instanceof = [];
  58.             $this->registerAliasesForSinglyImplementedInterfaces();
  59.         }
  60.         return null;
  61.     }
  62.     public function supports(mixed $resourcestring $type null): bool
  63.     {
  64.         if (!\is_string($resource)) {
  65.             return false;
  66.         }
  67.         if (null === $type && 'php' === pathinfo($resource\PATHINFO_EXTENSION)) {
  68.             return true;
  69.         }
  70.         return 'php' === $type;
  71.     }
  72.     /**
  73.      * Resolve the parameters to the $callback and execute it.
  74.      */
  75.     private function executeCallback(callable $callbackContainerConfigurator $containerConfiguratorstring $path)
  76.     {
  77.         $callback $callback(...);
  78.         $arguments = [];
  79.         $configBuilders = [];
  80.         $r = new \ReflectionFunction($callback);
  81.         $attribute null;
  82.         foreach ($r->getAttributes(When::class, \ReflectionAttribute::IS_INSTANCEOF) as $attribute) {
  83.             if ($this->env === $attribute->newInstance()->env) {
  84.                 $attribute null;
  85.                 break;
  86.             }
  87.         }
  88.         if (null !== $attribute) {
  89.             return;
  90.         }
  91.         foreach ($r->getParameters() as $parameter) {
  92.             $reflectionType $parameter->getType();
  93.             if (!$reflectionType instanceof \ReflectionNamedType) {
  94.                 throw new \InvalidArgumentException(sprintf('Could not resolve argument "$%s" for "%s". You must typehint it (for example with "%s" or "%s").'$parameter->getName(), $pathContainerConfigurator::class, ContainerBuilder::class));
  95.             }
  96.             $type $reflectionType->getName();
  97.             switch ($type) {
  98.                 case ContainerConfigurator::class:
  99.                     $arguments[] = $containerConfigurator;
  100.                     break;
  101.                 case ContainerBuilder::class:
  102.                     $arguments[] = $this->container;
  103.                     break;
  104.                 case FileLoader::class:
  105.                 case self::class:
  106.                     $arguments[] = $this;
  107.                     break;
  108.                 case 'string':
  109.                     if (null !== $this->env && 'env' === $parameter->getName()) {
  110.                         $arguments[] = $this->env;
  111.                         break;
  112.                     }
  113.                     // no break
  114.                 default:
  115.                     try {
  116.                         $configBuilder $this->configBuilder($type);
  117.                     } catch (InvalidArgumentException|\LogicException $e) {
  118.                         throw new \InvalidArgumentException(sprintf('Could not resolve argument "%s" for "%s".'$type.' $'.$parameter->getName(), $path), 0$e);
  119.                     }
  120.                     $configBuilders[] = $configBuilder;
  121.                     $arguments[] = $configBuilder;
  122.             }
  123.         }
  124.         // Force load ContainerConfigurator to make env(), param() etc available.
  125.         class_exists(ContainerConfigurator::class);
  126.         $callback(...$arguments);
  127.         /** @var ConfigBuilderInterface $configBuilder */
  128.         foreach ($configBuilders as $configBuilder) {
  129.             $containerConfigurator->extension($configBuilder->getExtensionAlias(), $configBuilder->toArray());
  130.         }
  131.     }
  132.     /**
  133.      * @param string $namespace FQCN string for a class implementing ConfigBuilderInterface
  134.      */
  135.     private function configBuilder(string $namespace): ConfigBuilderInterface
  136.     {
  137.         if (!class_exists(ConfigBuilderGenerator::class)) {
  138.             throw new \LogicException('You cannot use the config builder as the Config component is not installed. Try running "composer require symfony/config".');
  139.         }
  140.         if (null === $this->generator) {
  141.             throw new \LogicException('You cannot use the ConfigBuilders without providing a class implementing ConfigBuilderGeneratorInterface.');
  142.         }
  143.         // If class exists and implements ConfigBuilderInterface
  144.         if (class_exists($namespace) && is_subclass_of($namespaceConfigBuilderInterface::class)) {
  145.             return new $namespace();
  146.         }
  147.         // If it does not start with Symfony\Config\ we don't know how to handle this
  148.         if (!str_starts_with($namespace'Symfony\\Config\\')) {
  149.             throw new InvalidArgumentException(sprintf('Could not find or generate class "%s".'$namespace));
  150.         }
  151.         // Try to get the extension alias
  152.         $alias Container::underscore(substr($namespace15, -6));
  153.         if (str_contains($alias'\\')) {
  154.             throw new InvalidArgumentException('You can only use "root" ConfigBuilders from "Symfony\\Config\\" namespace. Nested classes like "Symfony\\Config\\Framework\\CacheConfig" cannot be used.');
  155.         }
  156.         if (!$this->container->hasExtension($alias)) {
  157.             $extensions array_filter(array_map(function (ExtensionInterface $ext) { return $ext->getAlias(); }, $this->container->getExtensions()));
  158.             throw new InvalidArgumentException(sprintf('There is no extension able to load the configuration for "%s". Looked for namespace "%s", found "%s".'$namespace$alias$extensions implode('", "'$extensions) : 'none'));
  159.         }
  160.         $extension $this->container->getExtension($alias);
  161.         if (!$extension instanceof ConfigurationExtensionInterface) {
  162.             throw new \LogicException(sprintf('You cannot use the config builder for "%s" because the extension does not implement "%s".'$namespaceConfigurationExtensionInterface::class));
  163.         }
  164.         $configuration $extension->getConfiguration([], $this->container);
  165.         $loader $this->generator->build($configuration);
  166.         return $loader();
  167.     }
  168. }
  169. /**
  170.  * @internal
  171.  */
  172. final class ProtectedPhpFileLoader extends PhpFileLoader
  173. {
  174. }