409 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
		
		
			
		
	
	
			409 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| 
								 | 
							
								<?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\DependencyInjection\Loader;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Config\Exception\FileLocatorFileNotFoundException;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Config\Exception\LoaderLoadException;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Config\FileLocatorInterface;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Config\Loader\FileLoader as BaseFileLoader;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Config\Loader\Loader;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Config\Resource\GlobResource;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\Alias;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\Attribute\AsAlias;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\Attribute\Exclude;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\Attribute\When;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\Attribute\WhenNot;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\ChildDefinition;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\Compiler\RegisterAutoconfigureAttributesPass;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\ContainerBuilder;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\Definition;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\Exception\InvalidArgumentException;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\Exception\LogicException;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * FileLoader is the abstract class used by all built-in loaders that are file based.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * @author Fabien Potencier <fabien@symfony.com>
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								abstract class FileLoader extends BaseFileLoader
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    public const ANONYMOUS_ID_REGEXP = '/^\.\d+_[^~]*+~[._a-zA-Z\d]{7}$/';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    protected bool $isLoadingInstanceof = false;
							 | 
						||
| 
								 | 
							
								    protected array $instanceof = [];
							 | 
						||
| 
								 | 
							
								    protected array $interfaces = [];
							 | 
						||
| 
								 | 
							
								    protected array $singlyImplemented = [];
							 | 
						||
| 
								 | 
							
								    /** @var array<string, Alias> */
							 | 
						||
| 
								 | 
							
								    protected array $aliases = [];
							 | 
						||
| 
								 | 
							
								    protected bool $autoRegisterAliasesForSinglyImplementedInterfaces = true;
							 | 
						||
| 
								 | 
							
								    protected array $extensionConfigs = [];
							 | 
						||
| 
								 | 
							
								    protected int $importing = 0;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * @param bool $prepend Whether to prepend extension config instead of appending them
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public function __construct(
							 | 
						||
| 
								 | 
							
								        protected ContainerBuilder $container,
							 | 
						||
| 
								 | 
							
								        FileLocatorInterface $locator,
							 | 
						||
| 
								 | 
							
								        ?string $env = null,
							 | 
						||
| 
								 | 
							
								        protected bool $prepend = false,
							 | 
						||
| 
								 | 
							
								    ) {
							 | 
						||
| 
								 | 
							
								        parent::__construct($locator, $env);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * @param bool|string $ignoreErrors Whether errors should be ignored; pass "not_found" to ignore only when the loaded resource is not found
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public function import(mixed $resource, ?string $type = null, bool|string $ignoreErrors = false, ?string $sourceResource = null, $exclude = null): mixed
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $args = \func_get_args();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if ($ignoreNotFound = 'not_found' === $ignoreErrors) {
							 | 
						||
| 
								 | 
							
								            $args[2] = false;
							 | 
						||
| 
								 | 
							
								        } elseif (!\is_bool($ignoreErrors)) {
							 | 
						||
| 
								 | 
							
								            throw new \TypeError(\sprintf('Invalid argument $ignoreErrors provided to "%s::import()": boolean or "not_found" expected, "%s" given.', static::class, get_debug_type($ignoreErrors)));
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        ++$this->importing;
							 | 
						||
| 
								 | 
							
								        try {
							 | 
						||
| 
								 | 
							
								            return parent::import(...$args);
							 | 
						||
| 
								 | 
							
								        } catch (LoaderLoadException $e) {
							 | 
						||
| 
								 | 
							
								            if (!$ignoreNotFound || !($prev = $e->getPrevious()) instanceof FileLocatorFileNotFoundException) {
							 | 
						||
| 
								 | 
							
								                throw $e;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            foreach ($prev->getTrace() as $frame) {
							 | 
						||
| 
								 | 
							
								                if ('import' === ($frame['function'] ?? null) && is_a($frame['class'] ?? '', Loader::class, true)) {
							 | 
						||
| 
								 | 
							
								                    break;
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if (__FILE__ !== $frame['file']) {
							 | 
						||
| 
								 | 
							
								                throw $e;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        } finally {
							 | 
						||
| 
								 | 
							
								            --$this->importing;
							 | 
						||
| 
								 | 
							
								            $this->loadExtensionConfigs();
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return null;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Registers a set of classes as services using PSR-4 for discovery.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * @param Definition           $prototype A definition to use as template
							 | 
						||
| 
								 | 
							
								     * @param string               $namespace The namespace prefix of classes in the scanned directory
							 | 
						||
| 
								 | 
							
								     * @param string               $resource  The directory to look for classes, glob-patterns allowed
							 | 
						||
| 
								 | 
							
								     * @param string|string[]|null $exclude   A globbed path of files to exclude or an array of globbed paths of files to exclude
							 | 
						||
| 
								 | 
							
								     * @param string|null          $source    The path to the file that defines the auto-discovery rule
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public function registerClasses(Definition $prototype, string $namespace, string $resource, string|array|null $exclude = null, ?string $source = null): void
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if (!str_ends_with($namespace, '\\')) {
							 | 
						||
| 
								 | 
							
								            throw new InvalidArgumentException(\sprintf('Namespace prefix must end with a "\\": "%s".', $namespace));
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        if (!preg_match('/^(?:[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*+\\\\)++$/', $namespace)) {
							 | 
						||
| 
								 | 
							
								            throw new InvalidArgumentException(\sprintf('Namespace is not a valid PSR-4 prefix: "%s".', $namespace));
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        // This can happen with YAML files
							 | 
						||
| 
								 | 
							
								        if (\is_array($exclude) && \in_array(null, $exclude, true)) {
							 | 
						||
| 
								 | 
							
								            throw new InvalidArgumentException('The exclude list must not contain a "null" value.');
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        // This can happen with XML files
							 | 
						||
| 
								 | 
							
								        if (\is_array($exclude) && \in_array('', $exclude, true)) {
							 | 
						||
| 
								 | 
							
								            throw new InvalidArgumentException('The exclude list must not contain an empty value.');
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $autoconfigureAttributes = new RegisterAutoconfigureAttributesPass();
							 | 
						||
| 
								 | 
							
								        $autoconfigureAttributes = $autoconfigureAttributes->accept($prototype) ? $autoconfigureAttributes : null;
							 | 
						||
| 
								 | 
							
								        $classes = $this->findClasses($namespace, $resource, (array) $exclude, $source);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $getPrototype = static fn () => clone $prototype;
							 | 
						||
| 
								 | 
							
								        $serialized = serialize($prototype);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        // avoid deep cloning if no definitions are nested
							 | 
						||
| 
								 | 
							
								        if (strpos($serialized, 'O:48:"Symfony\Component\DependencyInjection\Definition"', 55)
							 | 
						||
| 
								 | 
							
								            || strpos($serialized, 'O:53:"Symfony\Component\DependencyInjection\ChildDefinition"', 55)
							 | 
						||
| 
								 | 
							
								        ) {
							 | 
						||
| 
								 | 
							
								            // prepare for deep cloning
							 | 
						||
| 
								 | 
							
								            foreach (['Arguments', 'Properties', 'MethodCalls', 'Configurator', 'Factory', 'Bindings'] as $key) {
							 | 
						||
| 
								 | 
							
								                $serialized = serialize($prototype->{'get'.$key}());
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                if (strpos($serialized, 'O:48:"Symfony\Component\DependencyInjection\Definition"')
							 | 
						||
| 
								 | 
							
								                    || strpos($serialized, 'O:53:"Symfony\Component\DependencyInjection\ChildDefinition"')
							 | 
						||
| 
								 | 
							
								                ) {
							 | 
						||
| 
								 | 
							
								                    $getPrototype = static fn () => $getPrototype()->{'set'.$key}(unserialize($serialized));
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        unset($serialized);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        foreach ($classes as $class => $errorMessage) {
							 | 
						||
| 
								 | 
							
								            if (null === $errorMessage && $autoconfigureAttributes) {
							 | 
						||
| 
								 | 
							
								                $r = $this->container->getReflectionClass($class);
							 | 
						||
| 
								 | 
							
								                if ($r->getAttributes(Exclude::class)[0] ?? null) {
							 | 
						||
| 
								 | 
							
								                    $this->addContainerExcludedTag($class, $source);
							 | 
						||
| 
								 | 
							
								                    continue;
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                if ($this->env) {
							 | 
						||
| 
								 | 
							
								                    $excluded = true;
							 | 
						||
| 
								 | 
							
								                    $whenAttributes = $r->getAttributes(When::class, \ReflectionAttribute::IS_INSTANCEOF);
							 | 
						||
| 
								 | 
							
								                    $notWhenAttributes = $r->getAttributes(WhenNot::class, \ReflectionAttribute::IS_INSTANCEOF);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                    if ($whenAttributes && $notWhenAttributes) {
							 | 
						||
| 
								 | 
							
								                        throw new LogicException(\sprintf('The "%s" class cannot have both #[When] and #[WhenNot] attributes.', $class));
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                    if (!$whenAttributes && !$notWhenAttributes) {
							 | 
						||
| 
								 | 
							
								                        $excluded = false;
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                    foreach ($whenAttributes as $attribute) {
							 | 
						||
| 
								 | 
							
								                        if ($this->env === $attribute->newInstance()->env) {
							 | 
						||
| 
								 | 
							
								                            $excluded = false;
							 | 
						||
| 
								 | 
							
								                            break;
							 | 
						||
| 
								 | 
							
								                        }
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                    foreach ($notWhenAttributes as $attribute) {
							 | 
						||
| 
								 | 
							
								                        if ($excluded = $this->env === $attribute->newInstance()->env) {
							 | 
						||
| 
								 | 
							
								                            break;
							 | 
						||
| 
								 | 
							
								                        }
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                    if ($excluded) {
							 | 
						||
| 
								 | 
							
								                        $this->addContainerExcludedTag($class, $source);
							 | 
						||
| 
								 | 
							
								                        continue;
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            $r = null === $errorMessage ? $this->container->getReflectionClass($class) : null;
							 | 
						||
| 
								 | 
							
								            if ($r?->isAbstract() || $r?->isInterface()) {
							 | 
						||
| 
								 | 
							
								                if ($r->isInterface()) {
							 | 
						||
| 
								 | 
							
								                    $this->interfaces[] = $class;
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                $autoconfigureAttributes?->processClass($this->container, $r);
							 | 
						||
| 
								 | 
							
								                continue;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            $this->setDefinition($class, $definition = $getPrototype());
							 | 
						||
| 
								 | 
							
								            if (null !== $errorMessage) {
							 | 
						||
| 
								 | 
							
								                $definition->addError($errorMessage);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                continue;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            $definition->setClass($class);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            $interfaces = [];
							 | 
						||
| 
								 | 
							
								            foreach (class_implements($class, false) as $interface) {
							 | 
						||
| 
								 | 
							
								                $this->singlyImplemented[$interface] = ($this->singlyImplemented[$interface] ?? $class) !== $class ? false : $class;
							 | 
						||
| 
								 | 
							
								                $interfaces[] = $interface;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if (!$autoconfigureAttributes) {
							 | 
						||
| 
								 | 
							
								                continue;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            $r = $this->container->getReflectionClass($class);
							 | 
						||
| 
								 | 
							
								            $defaultAlias = 1 === \count($interfaces) ? $interfaces[0] : null;
							 | 
						||
| 
								 | 
							
								            foreach ($r->getAttributes(AsAlias::class) as $attr) {
							 | 
						||
| 
								 | 
							
								                /** @var AsAlias $attribute */
							 | 
						||
| 
								 | 
							
								                $attribute = $attr->newInstance();
							 | 
						||
| 
								 | 
							
								                $alias = $attribute->id ?? $defaultAlias;
							 | 
						||
| 
								 | 
							
								                $public = $attribute->public;
							 | 
						||
| 
								 | 
							
								                if (null === $alias) {
							 | 
						||
| 
								 | 
							
								                    throw new LogicException(\sprintf('Alias cannot be automatically determined for class "%s". If you have used the #[AsAlias] attribute with a class implementing multiple interfaces, add the interface you want to alias to the first parameter of #[AsAlias].', $class));
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                if (!$attribute->when || \in_array($this->env, $attribute->when, true)) {
							 | 
						||
| 
								 | 
							
								                    if (isset($this->aliases[$alias])) {
							 | 
						||
| 
								 | 
							
								                        throw new LogicException(\sprintf('The "%s" alias has already been defined with the #[AsAlias] attribute in "%s".', $alias, $this->aliases[$alias]));
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                    $this->aliases[$alias] = new Alias($class, $public);
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        foreach ($this->aliases as $alias => $aliasDefinition) {
							 | 
						||
| 
								 | 
							
								            $this->container->setAlias($alias, $aliasDefinition);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if ($this->autoRegisterAliasesForSinglyImplementedInterfaces) {
							 | 
						||
| 
								 | 
							
								            $this->registerAliasesForSinglyImplementedInterfaces();
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    public function registerAliasesForSinglyImplementedInterfaces(): void
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        foreach ($this->interfaces as $interface) {
							 | 
						||
| 
								 | 
							
								            if (!empty($this->singlyImplemented[$interface]) && !isset($this->aliases[$interface]) && !$this->container->has($interface)) {
							 | 
						||
| 
								 | 
							
								                $this->container->setAlias($interface, $this->singlyImplemented[$interface]);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $this->interfaces = $this->singlyImplemented = $this->aliases = [];
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    final protected function loadExtensionConfig(string $namespace, array $config): void
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if (!$this->prepend) {
							 | 
						||
| 
								 | 
							
								            $this->container->loadFromExtension($namespace, $config);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            return;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if ($this->importing) {
							 | 
						||
| 
								 | 
							
								            if (!isset($this->extensionConfigs[$namespace])) {
							 | 
						||
| 
								 | 
							
								                $this->extensionConfigs[$namespace] = [];
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            array_unshift($this->extensionConfigs[$namespace], $config);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            return;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $this->container->prependExtensionConfig($namespace, $config);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    final protected function loadExtensionConfigs(): void
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if ($this->importing || !$this->extensionConfigs) {
							 | 
						||
| 
								 | 
							
								            return;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        foreach ($this->extensionConfigs as $namespace => $configs) {
							 | 
						||
| 
								 | 
							
								            foreach ($configs as $config) {
							 | 
						||
| 
								 | 
							
								                $this->container->prependExtensionConfig($namespace, $config);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $this->extensionConfigs = [];
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Registers a definition in the container with its instanceof-conditionals.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    protected function setDefinition(string $id, Definition $definition): void
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $this->container->removeBindings($id);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        foreach ($definition->getTag('container.error') as $error) {
							 | 
						||
| 
								 | 
							
								            if (isset($error['message'])) {
							 | 
						||
| 
								 | 
							
								                $definition->addError($error['message']);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if ($this->isLoadingInstanceof) {
							 | 
						||
| 
								 | 
							
								            if (!$definition instanceof ChildDefinition) {
							 | 
						||
| 
								 | 
							
								                throw new InvalidArgumentException(\sprintf('Invalid type definition "%s": ChildDefinition expected, "%s" given.', $id, get_debug_type($definition)));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            $this->instanceof[$id] = $definition;
							 | 
						||
| 
								 | 
							
								        } else {
							 | 
						||
| 
								 | 
							
								            $this->container->setDefinition($id, $definition->setInstanceofConditionals($this->instanceof));
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private function findClasses(string $namespace, string $pattern, array $excludePatterns, ?string $source): array
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $parameterBag = $this->container->getParameterBag();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $excludePaths = [];
							 | 
						||
| 
								 | 
							
								        $excludePrefix = null;
							 | 
						||
| 
								 | 
							
								        $excludePatterns = $parameterBag->unescapeValue($parameterBag->resolveValue($excludePatterns));
							 | 
						||
| 
								 | 
							
								        foreach ($excludePatterns as $excludePattern) {
							 | 
						||
| 
								 | 
							
								            foreach ($this->glob($excludePattern, true, $resource, true, true) as $path => $info) {
							 | 
						||
| 
								 | 
							
								                $excludePrefix ??= $resource->getPrefix();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                // normalize Windows slashes and remove trailing slashes
							 | 
						||
| 
								 | 
							
								                $excludePaths[rtrim(str_replace('\\', '/', $path), '/')] = true;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $pattern = $parameterBag->unescapeValue($parameterBag->resolveValue($pattern));
							 | 
						||
| 
								 | 
							
								        $classes = [];
							 | 
						||
| 
								 | 
							
								        $prefixLen = null;
							 | 
						||
| 
								 | 
							
								        foreach ($this->glob($pattern, true, $resource, false, false, $excludePaths) as $path => $info) {
							 | 
						||
| 
								 | 
							
								            if (null === $prefixLen) {
							 | 
						||
| 
								 | 
							
								                $prefixLen = \strlen($resource->getPrefix());
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                if ($excludePrefix && !str_starts_with($excludePrefix, $resource->getPrefix())) {
							 | 
						||
| 
								 | 
							
								                    throw new InvalidArgumentException(\sprintf('Invalid "exclude" pattern when importing classes for "%s": make sure your "exclude" pattern (%s) is a subset of the "resource" pattern (%s).', $namespace, $excludePattern, $pattern));
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if (isset($excludePaths[str_replace('\\', '/', $path)])) {
							 | 
						||
| 
								 | 
							
								                continue;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if (!str_ends_with($path, '.php')) {
							 | 
						||
| 
								 | 
							
								                continue;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            $class = $namespace.ltrim(str_replace('/', '\\', substr($path, $prefixLen, -4)), '\\');
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if (!preg_match('/^[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*+(?:\\\\[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*+)*+$/', $class)) {
							 | 
						||
| 
								 | 
							
								                continue;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            try {
							 | 
						||
| 
								 | 
							
								                $r = $this->container->getReflectionClass($class);
							 | 
						||
| 
								 | 
							
								            } catch (\ReflectionException $e) {
							 | 
						||
| 
								 | 
							
								                $classes[$class] = $e->getMessage();
							 | 
						||
| 
								 | 
							
								                continue;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            // check to make sure the expected class exists
							 | 
						||
| 
								 | 
							
								            if (!$r) {
							 | 
						||
| 
								 | 
							
								                throw new InvalidArgumentException(\sprintf('Expected to find class "%s" in file "%s" while importing services from resource "%s", but it was not found! Check the namespace prefix used with the resource.', $class, $path, $pattern));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if (!$r->isTrait()) {
							 | 
						||
| 
								 | 
							
								                $classes[$class] = null;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        // track only for new & removed files
							 | 
						||
| 
								 | 
							
								        if ($resource instanceof GlobResource) {
							 | 
						||
| 
								 | 
							
								            $this->container->addResource($resource);
							 | 
						||
| 
								 | 
							
								        } else {
							 | 
						||
| 
								 | 
							
								            foreach ($resource as $path) {
							 | 
						||
| 
								 | 
							
								                $this->container->fileExists($path, false);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (null !== $prefixLen) {
							 | 
						||
| 
								 | 
							
								            foreach ($excludePaths as $path => $_) {
							 | 
						||
| 
								 | 
							
								                $class = $namespace.ltrim(str_replace('/', '\\', substr($path, $prefixLen, str_ends_with($path, '.php') ? -4 : null)), '\\');
							 | 
						||
| 
								 | 
							
								                $this->addContainerExcludedTag($class, $source);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return $classes;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private function addContainerExcludedTag(string $class, ?string $source): void
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if ($this->container->has($class)) {
							 | 
						||
| 
								 | 
							
								            return;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        static $attributes = [];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (null !== $source && !isset($attributes[$source])) {
							 | 
						||
| 
								 | 
							
								            $attributes[$source] = ['source' => \sprintf('in "%s/%s"', basename(\dirname($source)), basename($source))];
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $this->container->register($class, $class)
							 | 
						||
| 
								 | 
							
								            ->setAbstract(true)
							 | 
						||
| 
								 | 
							
								            ->addTag('container.excluded', null !== $source ? $attributes[$source] : []);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								}
							 |