* * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Symfony\Component\DependencyInjection\Compiler; use Symfony\Component\DependencyInjection\ContainerBuilder; use Symfony\Component\DependencyInjection\ContainerInterface; use Symfony\Component\DependencyInjection\Exception\ServiceNotFoundException; use Symfony\Component\DependencyInjection\Reference; /** * Checks that all references are pointing to a valid service. * * @author Johannes M. Schmitt */ class CheckExceptionOnInvalidReferenceBehaviorPass extends AbstractRecursivePass { protected bool $skipScalars = true; private array $serviceLocatorContextIds = []; /** * @return void */ public function process(ContainerBuilder $container) { $this->serviceLocatorContextIds = []; foreach ($container->findTaggedServiceIds('container.service_locator_context') as $id => $tags) { $this->serviceLocatorContextIds[$id] = $tags[0]['id']; $container->getDefinition($id)->clearTag('container.service_locator_context'); } try { parent::process($container); } finally { $this->serviceLocatorContextIds = []; } } protected function processValue(mixed $value, bool $isRoot = false): mixed { if (!$value instanceof Reference) { return parent::processValue($value, $isRoot); } if (ContainerInterface::EXCEPTION_ON_INVALID_REFERENCE < $value->getInvalidBehavior() || $this->container->has($id = (string) $value)) { return $value; } $currentId = $this->currentId; $graph = $this->container->getCompiler()->getServiceReferenceGraph(); if (isset($this->serviceLocatorContextIds[$currentId])) { $currentId = $this->serviceLocatorContextIds[$currentId]; $locator = $this->container->getDefinition($this->currentId)->getFactory()[0]; $this->throwServiceNotFoundException($value, $currentId, $locator->getArgument(0)); } if ('.' === $currentId[0] && $graph->hasNode($currentId)) { foreach ($graph->getNode($currentId)->getInEdges() as $edge) { if (!$edge->getValue() instanceof Reference || ContainerInterface::EXCEPTION_ON_INVALID_REFERENCE < $edge->getValue()->getInvalidBehavior()) { continue; } $sourceId = $edge->getSourceNode()->getId(); if ('.' !== $sourceId[0]) { $currentId = $sourceId; break; } if (isset($this->serviceLocatorContextIds[$sourceId])) { $currentId = $this->serviceLocatorContextIds[$sourceId]; $locator = $this->container->getDefinition($this->currentId); $this->throwServiceNotFoundException($value, $currentId, $locator->getArgument(0)); } } } $this->throwServiceNotFoundException($value, $currentId, $value); } private function throwServiceNotFoundException(Reference $ref, string $sourceId, $value): void { $id = (string) $ref; $alternatives = []; foreach ($this->container->getServiceIds() as $knownId) { if ('' === $knownId || '.' === $knownId[0] || $knownId === $this->currentId) { continue; } $lev = levenshtein($id, $knownId); if ($lev <= \strlen($id) / 3 || str_contains($knownId, $id)) { $alternatives[] = $knownId; } } $pass = new class() extends AbstractRecursivePass { public Reference $ref; public string $sourceId; public array $alternatives; public function processValue(mixed $value, bool $isRoot = false): mixed { if ($this->ref !== $value) { return parent::processValue($value, $isRoot); } $sourceId = $this->sourceId; if (null !== $this->currentId && $this->currentId !== (string) $value) { $sourceId = $this->currentId.'" in the container provided to "'.$sourceId; } throw new ServiceNotFoundException((string) $value, $sourceId, null, $this->alternatives); } }; $pass->ref = $ref; $pass->sourceId = $sourceId; $pass->alternatives = $alternatives; $pass->processValue($value, true); } }