4 * This file is part of the Symfony package.
6 * (c) Fabien Potencier <fabien@symfony.com>
8 * For the full copyright and license information, please view the LICENSE
9 * file that was distributed with this source code.
12 namespace Symfony\Component\DependencyInjection\Compiler;
14 use Symfony\Component\DependencyInjection\Definition;
15 use Symfony\Component\DependencyInjection\Reference;
16 use Symfony\Component\DependencyInjection\ContainerBuilder;
19 * Inline service definitions where this is possible.
21 * @author Johannes M. Schmitt <schmittjoh@gmail.com>
23 class InlineServiceDefinitionsPass implements RepeatablePassInterface
25 private $repeatedPass;
34 public function setRepeatedPass(RepeatedPass $repeatedPass)
36 $this->repeatedPass = $repeatedPass;
40 * Processes the ContainerBuilder for inline service definitions.
42 * @param ContainerBuilder $container
44 public function process(ContainerBuilder $container)
46 $this->compiler = $container->getCompiler();
47 $this->formatter = $this->compiler->getLoggingFormatter();
48 $this->graph = $this->compiler->getServiceReferenceGraph();
50 $container->setDefinitions($this->inlineArguments($container, $container->getDefinitions(), true));
54 * Processes inline arguments.
56 * @param ContainerBuilder $container The ContainerBuilder
57 * @param array $arguments An array of arguments
58 * @param bool $isRoot If we are processing the root definitions or not
62 private function inlineArguments(ContainerBuilder $container, array $arguments, $isRoot = false)
64 foreach ($arguments as $k => $argument) {
66 $this->currentId = $k;
68 if (is_array($argument)) {
69 $arguments[$k] = $this->inlineArguments($container, $argument);
70 } elseif ($argument instanceof Reference) {
71 if (!$container->hasDefinition($id = (string) $argument)) {
75 if ($this->isInlineableDefinition($id, $definition = $container->getDefinition($id))) {
76 $this->compiler->addLogMessage($this->formatter->formatInlineService($this, $id, $this->currentId));
78 if ($definition->isShared()) {
79 $arguments[$k] = $definition;
81 $arguments[$k] = clone $definition;
84 } elseif ($argument instanceof Definition) {
85 $argument->setArguments($this->inlineArguments($container, $argument->getArguments()));
86 $argument->setMethodCalls($this->inlineArguments($container, $argument->getMethodCalls()));
87 $argument->setProperties($this->inlineArguments($container, $argument->getProperties()));
89 $configurator = $this->inlineArguments($container, array($argument->getConfigurator()));
90 $argument->setConfigurator($configurator[0]);
92 $factory = $this->inlineArguments($container, array($argument->getFactory()));
93 $argument->setFactory($factory[0]);
101 * Checks if the definition is inlineable.
104 * @param Definition $definition
106 * @return bool If the definition is inlineable
108 private function isInlineableDefinition($id, Definition $definition)
110 if (!$definition->isShared()) {
114 if ($definition->isDeprecated() || $definition->isPublic() || $definition->isLazy()) {
118 if (!$this->graph->hasNode($id)) {
122 if ($this->currentId == $id) {
127 foreach ($this->graph->getNode($id)->getInEdges() as $edge) {
128 $ids[] = $edge->getSourceNode()->getId();
131 if (count(array_unique($ids)) > 1) {
135 if (count($ids) > 1 && is_array($factory = $definition->getFactory()) && ($factory[0] instanceof Reference || $factory[0] instanceof Definition)) {