mirror of
https://github.com/rectorphp/rector.git
synced 2024-06-13 14:42:23 +00:00
9ccb6367c0
9b03d8367c
[TypeDeclaration] Add ReturnUnionTypeRector (#4655)
107 lines
5.2 KiB
PHP
107 lines
5.2 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 RectorPrefix202308\Symfony\Component\DependencyInjection\Compiler;
|
|
|
|
use RectorPrefix202308\Symfony\Component\DependencyInjection\Alias;
|
|
use RectorPrefix202308\Symfony\Component\DependencyInjection\ContainerBuilder;
|
|
use RectorPrefix202308\Symfony\Component\DependencyInjection\ContainerInterface;
|
|
use RectorPrefix202308\Symfony\Component\DependencyInjection\Exception\InvalidArgumentException;
|
|
use RectorPrefix202308\Symfony\Component\DependencyInjection\Exception\ServiceNotFoundException;
|
|
use RectorPrefix202308\Symfony\Component\DependencyInjection\Reference;
|
|
/**
|
|
* Overwrites a service but keeps the overridden one.
|
|
*
|
|
* @author Christophe Coevoet <stof@notk.org>
|
|
* @author Fabien Potencier <fabien@symfony.com>
|
|
* @author Diego Saint Esteben <diego@saintesteben.me>
|
|
*/
|
|
class DecoratorServicePass extends AbstractRecursivePass
|
|
{
|
|
public function process(ContainerBuilder $container)
|
|
{
|
|
$definitions = new \SplPriorityQueue();
|
|
$order = \PHP_INT_MAX;
|
|
foreach ($container->getDefinitions() as $id => $definition) {
|
|
if (!($decorated = $definition->getDecoratedService())) {
|
|
continue;
|
|
}
|
|
$definitions->insert([$id, $definition], [$decorated[2], --$order]);
|
|
}
|
|
$decoratingDefinitions = [];
|
|
$tagsToKeep = $container->hasParameter('container.behavior_describing_tags') ? $container->getParameter('container.behavior_describing_tags') : ['container.do_not_inline', 'container.service_locator', 'container.service_subscriber', 'container.service_subscriber.locator'];
|
|
foreach ($definitions as [$id, $definition]) {
|
|
$decoratedService = $definition->getDecoratedService();
|
|
[$inner, $renamedId] = $decoratedService;
|
|
$invalidBehavior = $decoratedService[3] ?? ContainerInterface::EXCEPTION_ON_INVALID_REFERENCE;
|
|
$definition->setDecoratedService(null);
|
|
if (!$renamedId) {
|
|
$renamedId = $id . '.inner';
|
|
}
|
|
$this->currentId = $renamedId;
|
|
$this->processValue($definition);
|
|
$definition->innerServiceId = $renamedId;
|
|
$definition->decorationOnInvalid = $invalidBehavior;
|
|
// we create a new alias/service for the service we are replacing
|
|
// to be able to reference it in the new one
|
|
if ($container->hasAlias($inner)) {
|
|
$alias = $container->getAlias($inner);
|
|
$public = $alias->isPublic();
|
|
$container->setAlias($renamedId, new Alias((string) $alias, \false));
|
|
$decoratedDefinition = $container->findDefinition($alias);
|
|
} elseif ($container->hasDefinition($inner)) {
|
|
$decoratedDefinition = $container->getDefinition($inner);
|
|
$public = $decoratedDefinition->isPublic();
|
|
$decoratedDefinition->setPublic(\false);
|
|
$container->setDefinition($renamedId, $decoratedDefinition);
|
|
$decoratingDefinitions[$inner] = $decoratedDefinition;
|
|
} elseif (ContainerInterface::IGNORE_ON_INVALID_REFERENCE === $invalidBehavior) {
|
|
$container->removeDefinition($id);
|
|
continue;
|
|
} elseif (ContainerInterface::NULL_ON_INVALID_REFERENCE === $invalidBehavior) {
|
|
$public = $definition->isPublic();
|
|
$decoratedDefinition = null;
|
|
} else {
|
|
throw new ServiceNotFoundException($inner, $id);
|
|
}
|
|
if (($nullsafeVariable9 = $decoratedDefinition) ? $nullsafeVariable9->isSynthetic() : null) {
|
|
throw new InvalidArgumentException(\sprintf('A synthetic service cannot be decorated: service "%s" cannot decorate "%s".', $id, $inner));
|
|
}
|
|
if (isset($decoratingDefinitions[$inner])) {
|
|
$decoratingDefinition = $decoratingDefinitions[$inner];
|
|
$decoratingTags = $decoratingDefinition->getTags();
|
|
$resetTags = [];
|
|
// Behavior-describing tags must not be transferred out to decorators
|
|
foreach ($tagsToKeep as $containerTag) {
|
|
if (isset($decoratingTags[$containerTag])) {
|
|
$resetTags[$containerTag] = $decoratingTags[$containerTag];
|
|
unset($decoratingTags[$containerTag]);
|
|
}
|
|
}
|
|
$definition->setTags(\array_merge($decoratingTags, $definition->getTags()));
|
|
$decoratingDefinition->setTags($resetTags);
|
|
$decoratingDefinitions[$inner] = $definition;
|
|
}
|
|
$container->setAlias($inner, $id)->setPublic($public);
|
|
}
|
|
}
|
|
/**
|
|
* @param mixed $value
|
|
* @return mixed
|
|
*/
|
|
protected function processValue($value, bool $isRoot = \false)
|
|
{
|
|
if ($value instanceof Reference && '.inner' === (string) $value) {
|
|
return new Reference($this->currentId, $value->getInvalidBehavior());
|
|
}
|
|
return parent::processValue($value, $isRoot);
|
|
}
|
|
}
|