230 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
		
		
			
		
	
	
			230 lines
		
	
	
		
			7.9 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\Argument\ArgumentInterface;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\ContainerBuilder;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\Definition;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\Exception\ServiceCircularReferenceException;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\Reference;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Inline service definitions where this is possible.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * @author Johannes M. Schmitt <schmittjoh@gmail.com>
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								class InlineServiceDefinitionsPass extends AbstractRecursivePass
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    protected bool $skipScalars = true;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private array $cloningIds = [];
							 | 
						||
| 
								 | 
							
								    private array $connectedIds = [];
							 | 
						||
| 
								 | 
							
								    private array $notInlinedIds = [];
							 | 
						||
| 
								 | 
							
								    private array $inlinedIds = [];
							 | 
						||
| 
								 | 
							
								    private array $notInlinableIds = [];
							 | 
						||
| 
								 | 
							
								    private ?ServiceReferenceGraph $graph = null;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    public function __construct(
							 | 
						||
| 
								 | 
							
								        private ?AnalyzeServiceReferencesPass $analyzingPass = null,
							 | 
						||
| 
								 | 
							
								    ) {
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    public function process(ContainerBuilder $container): void
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $this->container = $container;
							 | 
						||
| 
								 | 
							
								        if ($this->analyzingPass) {
							 | 
						||
| 
								 | 
							
								            $analyzedContainer = new ContainerBuilder();
							 | 
						||
| 
								 | 
							
								            $analyzedContainer->setAliases($container->getAliases());
							 | 
						||
| 
								 | 
							
								            $analyzedContainer->setDefinitions($container->getDefinitions());
							 | 
						||
| 
								 | 
							
								            foreach ($container->getExpressionLanguageProviders() as $provider) {
							 | 
						||
| 
								 | 
							
								                $analyzedContainer->addExpressionLanguageProvider($provider);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        } else {
							 | 
						||
| 
								 | 
							
								            $analyzedContainer = $container;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        try {
							 | 
						||
| 
								 | 
							
								            $notInlinableIds = [];
							 | 
						||
| 
								 | 
							
								            $remainingInlinedIds = [];
							 | 
						||
| 
								 | 
							
								            $this->connectedIds = $this->notInlinedIds = $container->getDefinitions();
							 | 
						||
| 
								 | 
							
								            do {
							 | 
						||
| 
								 | 
							
								                if ($this->analyzingPass) {
							 | 
						||
| 
								 | 
							
								                    $analyzedContainer->setDefinitions(array_intersect_key($analyzedContainer->getDefinitions(), $this->connectedIds));
							 | 
						||
| 
								 | 
							
								                    $this->analyzingPass->process($analyzedContainer);
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                $this->graph = $analyzedContainer->getCompiler()->getServiceReferenceGraph();
							 | 
						||
| 
								 | 
							
								                $notInlinedIds = $this->notInlinedIds;
							 | 
						||
| 
								 | 
							
								                $notInlinableIds += $this->notInlinableIds;
							 | 
						||
| 
								 | 
							
								                $this->connectedIds = $this->notInlinedIds = $this->inlinedIds = $this->notInlinableIds = [];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                foreach ($analyzedContainer->getDefinitions() as $id => $definition) {
							 | 
						||
| 
								 | 
							
								                    if (!$this->graph->hasNode($id)) {
							 | 
						||
| 
								 | 
							
								                        continue;
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								                    if ($definition->isPublic()) {
							 | 
						||
| 
								 | 
							
								                        $this->connectedIds[$id] = true;
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								                    foreach ($this->graph->getNode($id)->getOutEdges() as $edge) {
							 | 
						||
| 
								 | 
							
								                        if (isset($notInlinedIds[$edge->getSourceNode()->getId()])) {
							 | 
						||
| 
								 | 
							
								                            $this->currentId = $id;
							 | 
						||
| 
								 | 
							
								                            $this->processValue($definition, true);
							 | 
						||
| 
								 | 
							
								                            break;
							 | 
						||
| 
								 | 
							
								                        }
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                foreach ($this->inlinedIds as $id => $isPublicOrNotShared) {
							 | 
						||
| 
								 | 
							
								                    if ($isPublicOrNotShared) {
							 | 
						||
| 
								 | 
							
								                        $remainingInlinedIds[$id] = $id;
							 | 
						||
| 
								 | 
							
								                    } else {
							 | 
						||
| 
								 | 
							
								                        $container->removeDefinition($id);
							 | 
						||
| 
								 | 
							
								                        $analyzedContainer->removeDefinition($id);
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            } while ($this->inlinedIds && $this->analyzingPass);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            foreach ($remainingInlinedIds as $id) {
							 | 
						||
| 
								 | 
							
								                if (isset($notInlinableIds[$id])) {
							 | 
						||
| 
								 | 
							
								                    continue;
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                $definition = $container->getDefinition($id);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                if (!$definition->isShared() && !$definition->isPublic()) {
							 | 
						||
| 
								 | 
							
								                    $container->removeDefinition($id);
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        } finally {
							 | 
						||
| 
								 | 
							
								            $this->container = null;
							 | 
						||
| 
								 | 
							
								            $this->connectedIds = $this->notInlinedIds = $this->inlinedIds = [];
							 | 
						||
| 
								 | 
							
								            $this->notInlinableIds = [];
							 | 
						||
| 
								 | 
							
								            $this->graph = null;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    protected function processValue(mixed $value, bool $isRoot = false): mixed
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if ($value instanceof ArgumentInterface) {
							 | 
						||
| 
								 | 
							
								            // References found in ArgumentInterface::getValues() are not inlineable
							 | 
						||
| 
								 | 
							
								            return $value;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if ($value instanceof Definition && $this->cloningIds) {
							 | 
						||
| 
								 | 
							
								            if ($value->isShared()) {
							 | 
						||
| 
								 | 
							
								                return $value;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            $value = clone $value;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!$value instanceof Reference) {
							 | 
						||
| 
								 | 
							
								            return parent::processValue($value, $isRoot);
							 | 
						||
| 
								 | 
							
								        } elseif (!$this->container->hasDefinition($id = (string) $value)) {
							 | 
						||
| 
								 | 
							
								            return $value;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $definition = $this->container->getDefinition($id);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (isset($this->notInlinableIds[$id]) || !$this->isInlineableDefinition($id, $definition)) {
							 | 
						||
| 
								 | 
							
								            if ($this->currentId !== $id) {
							 | 
						||
| 
								 | 
							
								                $this->notInlinableIds[$id] = true;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            return $value;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $this->container->log($this, \sprintf('Inlined service "%s" to "%s".', $id, $this->currentId));
							 | 
						||
| 
								 | 
							
								        $this->inlinedIds[$id] = $definition->isPublic() || !$definition->isShared();
							 | 
						||
| 
								 | 
							
								        $this->notInlinedIds[$this->currentId ?? ''] = true;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if ($definition->isShared()) {
							 | 
						||
| 
								 | 
							
								            return $definition;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (isset($this->cloningIds[$id])) {
							 | 
						||
| 
								 | 
							
								            $ids = array_keys($this->cloningIds);
							 | 
						||
| 
								 | 
							
								            $ids[] = $id;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            throw new ServiceCircularReferenceException($id, \array_slice($ids, array_search($id, $ids)));
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $this->cloningIds[$id] = true;
							 | 
						||
| 
								 | 
							
								        try {
							 | 
						||
| 
								 | 
							
								            return $this->processValue($definition);
							 | 
						||
| 
								 | 
							
								        } finally {
							 | 
						||
| 
								 | 
							
								            unset($this->cloningIds[$id]);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Checks if the definition is inlineable.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    private function isInlineableDefinition(string $id, Definition $definition): bool
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if (str_starts_with($id, '.autowire_inline.')) {
							 | 
						||
| 
								 | 
							
								            return true;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        if ($definition->hasErrors() || $definition->isDeprecated() || $definition->isLazy() || $definition->isSynthetic() || $definition->hasTag('container.do_not_inline')) {
							 | 
						||
| 
								 | 
							
								            return false;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!$definition->isShared()) {
							 | 
						||
| 
								 | 
							
								            if (!$this->graph->hasNode($id)) {
							 | 
						||
| 
								 | 
							
								                return true;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            foreach ($this->graph->getNode($id)->getInEdges() as $edge) {
							 | 
						||
| 
								 | 
							
								                $srcId = $edge->getSourceNode()->getId();
							 | 
						||
| 
								 | 
							
								                $this->connectedIds[$srcId] = true;
							 | 
						||
| 
								 | 
							
								                if ($edge->isWeak() || $edge->isLazy()) {
							 | 
						||
| 
								 | 
							
								                    return !$this->connectedIds[$id] = true;
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            return true;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if ($definition->isPublic()
							 | 
						||
| 
								 | 
							
								            || $this->currentId === $id
							 | 
						||
| 
								 | 
							
								            || !$this->graph->hasNode($id)
							 | 
						||
| 
								 | 
							
								        ) {
							 | 
						||
| 
								 | 
							
								            return false;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $this->connectedIds[$id] = true;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $srcIds = [];
							 | 
						||
| 
								 | 
							
								        $srcCount = 0;
							 | 
						||
| 
								 | 
							
								        foreach ($this->graph->getNode($id)->getInEdges() as $edge) {
							 | 
						||
| 
								 | 
							
								            $srcId = $edge->getSourceNode()->getId();
							 | 
						||
| 
								 | 
							
								            $this->connectedIds[$srcId] = true;
							 | 
						||
| 
								 | 
							
								            if ($edge->isWeak() || $edge->isLazy()) {
							 | 
						||
| 
								 | 
							
								                return false;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            $srcIds[$srcId] = true;
							 | 
						||
| 
								 | 
							
								            ++$srcCount;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (1 !== \count($srcIds)) {
							 | 
						||
| 
								 | 
							
								            $this->notInlinedIds[$id] = true;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            return false;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if ($srcCount > 1 && \is_array($factory = $definition->getFactory()) && ($factory[0] instanceof Reference || $factory[0] instanceof Definition)) {
							 | 
						||
| 
								 | 
							
								            return false;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $srcDefinition = $this->container->getDefinition($srcId);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return $srcDefinition->isShared() && !$srcDefinition->isLazy();
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								}
							 |