126 lines
4.6 KiB
PHP
126 lines
4.6 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\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 <schmittjoh@gmail.com>
|
||
|
*/
|
||
|
class CheckExceptionOnInvalidReferenceBehaviorPass extends AbstractRecursivePass
|
||
|
{
|
||
|
protected bool $skipScalars = true;
|
||
|
|
||
|
private array $serviceLocatorContextIds = [];
|
||
|
|
||
|
public function process(ContainerBuilder $container): void
|
||
|
{
|
||
|
$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((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, mixed $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);
|
||
|
}
|
||
|
}
|