vendor/symfony/dependency-injection/Compiler/AutowirePass.php line 356

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\Component\DependencyInjection\Compiler;
  11. use Symfony\Component\Config\Resource\ClassExistenceResource;
  12. use Symfony\Component\DependencyInjection\ContainerBuilder;
  13. use Symfony\Component\DependencyInjection\Definition;
  14. use Symfony\Component\DependencyInjection\Exception\AutowiringFailedException;
  15. use Symfony\Component\DependencyInjection\Exception\RuntimeException;
  16. use Symfony\Component\DependencyInjection\LazyProxy\ProxyHelper;
  17. use Symfony\Component\DependencyInjection\TypedReference;
  18. /**
  19.  * Inspects existing service definitions and wires the autowired ones using the type hints of their classes.
  20.  *
  21.  * @author Kévin Dunglas <dunglas@gmail.com>
  22.  * @author Nicolas Grekas <p@tchwork.com>
  23.  */
  24. class AutowirePass extends AbstractRecursivePass
  25. {
  26.     private $types;
  27.     private $ambiguousServiceTypes;
  28.     private $lastFailure;
  29.     private $throwOnAutowiringException;
  30.     private $decoratedClass;
  31.     private $decoratedId;
  32.     private $methodCalls;
  33.     private $defaultArgument;
  34.     private $getPreviousValue;
  35.     private $decoratedMethodIndex;
  36.     private $decoratedMethodArgumentIndex;
  37.     private $typesClone;
  38.     public function __construct(bool $throwOnAutowireException true)
  39.     {
  40.         $this->throwOnAutowiringException $throwOnAutowireException;
  41.         $this->defaultArgument = new class() {
  42.             public $value;
  43.             public $names;
  44.         };
  45.     }
  46.     /**
  47.      * {@inheritdoc}
  48.      */
  49.     public function process(ContainerBuilder $container)
  50.     {
  51.         try {
  52.             $this->typesClone = clone $this;
  53.             parent::process($container);
  54.         } finally {
  55.             $this->decoratedClass null;
  56.             $this->decoratedId null;
  57.             $this->methodCalls null;
  58.             $this->defaultArgument->names null;
  59.             $this->getPreviousValue null;
  60.             $this->decoratedMethodIndex null;
  61.             $this->decoratedMethodArgumentIndex null;
  62.             $this->typesClone null;
  63.         }
  64.     }
  65.     /**
  66.      * {@inheritdoc}
  67.      */
  68.     protected function processValue($value$isRoot false)
  69.     {
  70.         try {
  71.             return $this->doProcessValue($value$isRoot);
  72.         } catch (AutowiringFailedException $e) {
  73.             if ($this->throwOnAutowiringException) {
  74.                 throw $e;
  75.             }
  76.             $this->container->getDefinition($this->currentId)->addError($e->getMessageCallback() ?? $e->getMessage());
  77.             return parent::processValue($value$isRoot);
  78.         }
  79.     }
  80.     /**
  81.      * @return mixed
  82.      */
  83.     private function doProcessValue($valuebool $isRoot false)
  84.     {
  85.         if ($value instanceof TypedReference) {
  86.             if ($ref $this->getAutowiredReference($value)) {
  87.                 return $ref;
  88.             }
  89.             if (ContainerBuilder::RUNTIME_EXCEPTION_ON_INVALID_REFERENCE === $value->getInvalidBehavior()) {
  90.                 $message $this->createTypeNotFoundMessageCallback($value'it');
  91.                 // since the error message varies by referenced id and $this->currentId, so should the id of the dummy errored definition
  92.                 $this->container->register($id sprintf('.errored.%s.%s'$this->currentId, (string) $value), $value->getType())
  93.                     ->addError($message);
  94.                 return new TypedReference($id$value->getType(), $value->getInvalidBehavior(), $value->getName());
  95.             }
  96.         }
  97.         $value parent::processValue($value$isRoot);
  98.         if (!$value instanceof Definition || !$value->isAutowired() || $value->isAbstract() || !$value->getClass()) {
  99.             return $value;
  100.         }
  101.         if (!$reflectionClass $this->container->getReflectionClass($value->getClass(), false)) {
  102.             $this->container->log($thissprintf('Skipping service "%s": Class or interface "%s" cannot be loaded.'$this->currentId$value->getClass()));
  103.             return $value;
  104.         }
  105.         $this->methodCalls $value->getMethodCalls();
  106.         try {
  107.             $constructor $this->getConstructor($valuefalse);
  108.         } catch (RuntimeException $e) {
  109.             throw new AutowiringFailedException($this->currentId$e->getMessage(), 0$e);
  110.         }
  111.         if ($constructor) {
  112.             array_unshift($this->methodCalls, [$constructor$value->getArguments()]);
  113.         }
  114.         $this->methodCalls $this->autowireCalls($reflectionClass$isRoot);
  115.         if ($constructor) {
  116.             [, $arguments] = array_shift($this->methodCalls);
  117.             if ($arguments !== $value->getArguments()) {
  118.                 $value->setArguments($arguments);
  119.             }
  120.         }
  121.         if ($this->methodCalls !== $value->getMethodCalls()) {
  122.             $value->setMethodCalls($this->methodCalls);
  123.         }
  124.         return $value;
  125.     }
  126.     private function autowireCalls(\ReflectionClass $reflectionClassbool $isRoot): array
  127.     {
  128.         $this->decoratedId null;
  129.         $this->decoratedClass null;
  130.         $this->getPreviousValue null;
  131.         if ($isRoot && ($definition $this->container->getDefinition($this->currentId)) && null !== ($this->decoratedId $definition->innerServiceId) && $this->container->has($this->decoratedId)) {
  132.             $this->decoratedClass $this->container->findDefinition($this->decoratedId)->getClass();
  133.         }
  134.         $patchedIndexes = [];
  135.         foreach ($this->methodCalls as $i => $call) {
  136.             [$method$arguments] = $call;
  137.             if ($method instanceof \ReflectionFunctionAbstract) {
  138.                 $reflectionMethod $method;
  139.             } else {
  140.                 $definition = new Definition($reflectionClass->name);
  141.                 try {
  142.                     $reflectionMethod $this->getReflectionMethod($definition$method);
  143.                 } catch (RuntimeException $e) {
  144.                     if ($definition->getFactory()) {
  145.                         continue;
  146.                     }
  147.                     throw $e;
  148.                 }
  149.             }
  150.             $arguments $this->autowireMethod($reflectionMethod$arguments$i);
  151.             if ($arguments !== $call[1]) {
  152.                 $this->methodCalls[$i][1] = $arguments;
  153.                 $patchedIndexes[] = $i;
  154.             }
  155.         }
  156.         // use named arguments to skip complex default values
  157.         foreach ($patchedIndexes as $i) {
  158.             $namedArguments null;
  159.             $arguments $this->methodCalls[$i][1];
  160.             foreach ($arguments as $j => $value) {
  161.                 if ($namedArguments && !$value instanceof $this->defaultArgument) {
  162.                     unset($arguments[$j]);
  163.                     $arguments[$namedArguments[$j]] = $value;
  164.                 }
  165.                 if ($namedArguments || !$value instanceof $this->defaultArgument) {
  166.                     continue;
  167.                 }
  168.                 if (\PHP_VERSION_ID >= 80100 && (\is_array($value->value) ? $value->value : \is_object($value->value))) {
  169.                     unset($arguments[$j]);
  170.                     $namedArguments $value->names;
  171.                 } else {
  172.                     $arguments[$j] = $value->value;
  173.                 }
  174.             }
  175.             $this->methodCalls[$i][1] = $arguments;
  176.         }
  177.         return $this->methodCalls;
  178.     }
  179.     /**
  180.      * Autowires the constructor or a method.
  181.      *
  182.      * @return array The autowired arguments
  183.      *
  184.      * @throws AutowiringFailedException
  185.      */
  186.     private function autowireMethod(\ReflectionFunctionAbstract $reflectionMethod, array $argumentsint $methodIndex): array
  187.     {
  188.         $class $reflectionMethod instanceof \ReflectionMethod $reflectionMethod->class $this->currentId;
  189.         $method $reflectionMethod->name;
  190.         $parameters $reflectionMethod->getParameters();
  191.         if ($reflectionMethod->isVariadic()) {
  192.             array_pop($parameters);
  193.         }
  194.         $this->defaultArgument->names = new \ArrayObject();
  195.         foreach ($parameters as $index => $parameter) {
  196.             $this->defaultArgument->names[$index] = $parameter->name;
  197.             if (\array_key_exists($index$arguments) && '' !== $arguments[$index]) {
  198.                 continue;
  199.             }
  200.             $type ProxyHelper::getTypeHint($reflectionMethod$parametertrue);
  201.             if (!$type) {
  202.                 if (isset($arguments[$index])) {
  203.                     continue;
  204.                 }
  205.                 // no default value? Then fail
  206.                 if (!$parameter->isDefaultValueAvailable()) {
  207.                     // For core classes, isDefaultValueAvailable() can
  208.                     // be false when isOptional() returns true. If the
  209.                     // argument *is* optional, allow it to be missing
  210.                     if ($parameter->isOptional()) {
  211.                         --$index;
  212.                         break;
  213.                     }
  214.                     $type ProxyHelper::getTypeHint($reflectionMethod$parameterfalse);
  215.                     $type $type sprintf('is type-hinted "%s"'ltrim($type'\\')) : 'has no type-hint';
  216.                     throw new AutowiringFailedException($this->currentIdsprintf('Cannot autowire service "%s": argument "$%s" of method "%s()" %s, you should configure its value explicitly.'$this->currentId$parameter->name$class !== $this->currentId $class.'::'.$method $method$type));
  217.                 }
  218.                 // specifically pass the default value
  219.                 $arguments[$index] = clone $this->defaultArgument;
  220.                 $arguments[$index]->value $parameter->getDefaultValue();
  221.                 continue;
  222.             }
  223.             $getValue = function () use ($type$parameter$class$method) {
  224.                 if (!$value $this->getAutowiredReference($ref = new TypedReference($type$typeContainerBuilder::EXCEPTION_ON_INVALID_REFERENCE$parameter->name))) {
  225.                     $failureMessage $this->createTypeNotFoundMessageCallback($refsprintf('argument "$%s" of method "%s()"'$parameter->name$class !== $this->currentId $class.'::'.$method $method));
  226.                     if ($parameter->isDefaultValueAvailable()) {
  227.                         $value = clone $this->defaultArgument;
  228.                         $value->value $parameter->getDefaultValue();
  229.                     } elseif (!$parameter->allowsNull()) {
  230.                         throw new AutowiringFailedException($this->currentId$failureMessage);
  231.                     }
  232.                 }
  233.                 return $value;
  234.             };
  235.             if ($this->decoratedClass && $isDecorated is_a($this->decoratedClass$typetrue)) {
  236.                 if ($this->getPreviousValue) {
  237.                     // The inner service is injected only if there is only 1 argument matching the type of the decorated class
  238.                     // across all arguments of all autowired methods.
  239.                     // If a second matching argument is found, the default behavior is restored.
  240.                     $getPreviousValue $this->getPreviousValue;
  241.                     $this->methodCalls[$this->decoratedMethodIndex][1][$this->decoratedMethodArgumentIndex] = $getPreviousValue();
  242.                     $this->decoratedClass null// Prevent further checks
  243.                 } else {
  244.                     $arguments[$index] = new TypedReference($this->decoratedId$this->decoratedClass);
  245.                     $this->getPreviousValue $getValue;
  246.                     $this->decoratedMethodIndex $methodIndex;
  247.                     $this->decoratedMethodArgumentIndex $index;
  248.                     continue;
  249.                 }
  250.             }
  251.             $arguments[$index] = $getValue();
  252.         }
  253.         if ($parameters && !isset($arguments[++$index])) {
  254.             while (<= --$index) {
  255.                 if (!$arguments[$index] instanceof $this->defaultArgument) {
  256.                     break;
  257.                 }
  258.                 unset($arguments[$index]);
  259.             }
  260.         }
  261.         // it's possible index 1 was set, then index 0, then 2, etc
  262.         // make sure that we re-order so they're injected as expected
  263.         ksort($arguments);
  264.         return $arguments;
  265.     }
  266.     /**
  267.      * Returns a reference to the service matching the given type, if any.
  268.      */
  269.     private function getAutowiredReference(TypedReference $reference): ?TypedReference
  270.     {
  271.         $this->lastFailure null;
  272.         $type $reference->getType();
  273.         if ($type !== (string) $reference) {
  274.             return $reference;
  275.         }
  276.         if (null !== $name $reference->getName()) {
  277.             if ($this->container->has($alias $type.' $'.$name) && !$this->container->findDefinition($alias)->isAbstract()) {
  278.                 return new TypedReference($alias$type$reference->getInvalidBehavior());
  279.             }
  280.             if ($this->container->has($name) && !$this->container->findDefinition($name)->isAbstract()) {
  281.                 foreach ($this->container->getAliases() as $id => $alias) {
  282.                     if ($name === (string) $alias && str_starts_with($id$type.' $')) {
  283.                         return new TypedReference($name$type$reference->getInvalidBehavior());
  284.                     }
  285.                 }
  286.             }
  287.         }
  288.         if ($this->container->has($type) && !$this->container->findDefinition($type)->isAbstract()) {
  289.             return new TypedReference($type$type$reference->getInvalidBehavior());
  290.         }
  291.         return null;
  292.     }
  293.     /**
  294.      * Populates the list of available types.
  295.      */
  296.     private function populateAvailableTypes(ContainerBuilder $container)
  297.     {
  298.         $this->types = [];
  299.         $this->ambiguousServiceTypes = [];
  300.         foreach ($container->getDefinitions() as $id => $definition) {
  301.             $this->populateAvailableType($container$id$definition);
  302.         }
  303.     }
  304.     /**
  305.      * Populates the list of available types for a given definition.
  306.      */
  307.     private function populateAvailableType(ContainerBuilder $containerstring $idDefinition $definition)
  308.     {
  309.         // Never use abstract services
  310.         if ($definition->isAbstract()) {
  311.             return;
  312.         }
  313.         if ('' === $id || '.' === $id[0] || $definition->isDeprecated() || !$reflectionClass $container->getReflectionClass($definition->getClass(), false)) {
  314.             return;
  315.         }
  316.         foreach ($reflectionClass->getInterfaces() as $reflectionInterface) {
  317.             $this->set($reflectionInterface->name$id);
  318.         }
  319.         do {
  320.             $this->set($reflectionClass->name$id);
  321.         } while ($reflectionClass $reflectionClass->getParentClass());
  322.     }
  323.     /**
  324.      * Associates a type and a service id if applicable.
  325.      */
  326.     private function set(string $typestring $id)
  327.     {
  328.         // is this already a type/class that is known to match multiple services?
  329.         if (isset($this->ambiguousServiceTypes[$type])) {
  330.             $this->ambiguousServiceTypes[$type][] = $id;
  331.             return;
  332.         }
  333.         // check to make sure the type doesn't match multiple services
  334.         if (!isset($this->types[$type]) || $this->types[$type] === $id) {
  335.             $this->types[$type] = $id;
  336.             return;
  337.         }
  338.         // keep an array of all services matching this type
  339.         if (!isset($this->ambiguousServiceTypes[$type])) {
  340.             $this->ambiguousServiceTypes[$type] = [$this->types[$type]];
  341.             unset($this->types[$type]);
  342.         }
  343.         $this->ambiguousServiceTypes[$type][] = $id;
  344.     }
  345.     private function createTypeNotFoundMessageCallback(TypedReference $referencestring $label): \Closure
  346.     {
  347.         if (null === $this->typesClone->container) {
  348.             $this->typesClone->container = new ContainerBuilder($this->container->getParameterBag());
  349.             $this->typesClone->container->setAliases($this->container->getAliases());
  350.             $this->typesClone->container->setDefinitions($this->container->getDefinitions());
  351.             $this->typesClone->container->setResourceTracking(false);
  352.         }
  353.         $currentId $this->currentId;
  354.         return (function () use ($reference$label$currentId) {
  355.             return $this->createTypeNotFoundMessage($reference$label$currentId);
  356.         })->bindTo($this->typesClone);
  357.     }
  358.     private function createTypeNotFoundMessage(TypedReference $referencestring $labelstring $currentId): string
  359.     {
  360.         if (!$r $this->container->getReflectionClass($type $reference->getType(), false)) {
  361.             // either $type does not exist or a parent class does not exist
  362.             try {
  363.                 $resource = new ClassExistenceResource($typefalse);
  364.                 // isFresh() will explode ONLY if a parent class/trait does not exist
  365.                 $resource->isFresh(0);
  366.                 $parentMsg false;
  367.             } catch (\ReflectionException $e) {
  368.                 $parentMsg $e->getMessage();
  369.             }
  370.             $message sprintf('has type "%s" but this class %s.'$type$parentMsg sprintf('is missing a parent class (%s)'$parentMsg) : 'was not found');
  371.         } else {
  372.             $alternatives $this->createTypeAlternatives($this->container$reference);
  373.             $message $this->container->has($type) ? 'this service is abstract' 'no such service exists';
  374.             $message sprintf('references %s "%s" but %s.%s'$r->isInterface() ? 'interface' 'class'$type$message$alternatives);
  375.             if ($r->isInterface() && !$alternatives) {
  376.                 $message .= ' Did you create a class that implements this interface?';
  377.             }
  378.         }
  379.         $message sprintf('Cannot autowire service "%s": %s %s'$currentId$label$message);
  380.         if (null !== $this->lastFailure) {
  381.             $message $this->lastFailure."\n".$message;
  382.             $this->lastFailure null;
  383.         }
  384.         return $message;
  385.     }
  386.     private function createTypeAlternatives(ContainerBuilder $containerTypedReference $reference): string
  387.     {
  388.         // try suggesting available aliases first
  389.         if ($message $this->getAliasesSuggestionForType($container$type $reference->getType())) {
  390.             return ' '.$message;
  391.         }
  392.         if (null === $this->ambiguousServiceTypes) {
  393.             $this->populateAvailableTypes($container);
  394.         }
  395.         $servicesAndAliases $container->getServiceIds();
  396.         if (!$container->has($type) && false !== $key array_search(strtolower($type), array_map('strtolower'$servicesAndAliases))) {
  397.             return sprintf(' Did you mean "%s"?'$servicesAndAliases[$key]);
  398.         } elseif (isset($this->ambiguousServiceTypes[$type])) {
  399.             $message sprintf('one of these existing services: "%s"'implode('", "'$this->ambiguousServiceTypes[$type]));
  400.         } elseif (isset($this->types[$type])) {
  401.             $message sprintf('the existing "%s" service'$this->types[$type]);
  402.         } else {
  403.             return '';
  404.         }
  405.         return sprintf(' You should maybe alias this %s to %s.'class_exists($typefalse) ? 'class' 'interface'$message);
  406.     }
  407.     private function getAliasesSuggestionForType(ContainerBuilder $containerstring $type): ?string
  408.     {
  409.         $aliases = [];
  410.         foreach (class_parents($type) + class_implements($type) as $parent) {
  411.             if ($container->has($parent) && !$container->findDefinition($parent)->isAbstract()) {
  412.                 $aliases[] = $parent;
  413.             }
  414.         }
  415.         if ($len = \count($aliases)) {
  416.             $message 'Try changing the type-hint to one of its parents: ';
  417.             for ($i 0, --$len$i $len; ++$i) {
  418.                 $message .= sprintf('%s "%s", 'class_exists($aliases[$i], false) ? 'class' 'interface'$aliases[$i]);
  419.             }
  420.             $message .= sprintf('or %s "%s".'class_exists($aliases[$i], false) ? 'class' 'interface'$aliases[$i]);
  421.             return $message;
  422.         }
  423.         if ($aliases) {
  424.             return sprintf('Try changing the type-hint to "%s" instead.'$aliases[0]);
  425.         }
  426.         return null;
  427.     }
  428. }