| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174 |
- <?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\Definition;
- use Symfony\Component\DependencyInjection\ExpressionLanguage;
- use Symfony\Component\DependencyInjection\Reference;
- use Symfony\Component\DependencyInjection\ContainerBuilder;
- use Symfony\Component\ExpressionLanguage\Expression;
- /**
- * Run this pass before passes that need to know more about the relation of
- * your services.
- *
- * This class will populate the ServiceReferenceGraph with information. You can
- * retrieve the graph in other passes from the compiler.
- *
- * @author Johannes M. Schmitt <schmittjoh@gmail.com>
- */
- class AnalyzeServiceReferencesPass extends AbstractRecursivePass implements RepeatablePassInterface
- {
- private $graph;
- private $currentDefinition;
- private $onlyConstructorArguments;
- private $lazy;
- private $expressionLanguage;
- /**
- * @param bool $onlyConstructorArguments Sets this Service Reference pass to ignore method calls
- */
- public function __construct($onlyConstructorArguments = false)
- {
- $this->onlyConstructorArguments = (bool) $onlyConstructorArguments;
- }
- /**
- * {@inheritdoc}
- */
- public function setRepeatedPass(RepeatedPass $repeatedPass)
- {
- // no-op for BC
- }
- /**
- * Processes a ContainerBuilder object to populate the service reference graph.
- */
- public function process(ContainerBuilder $container)
- {
- $this->container = $container;
- $this->graph = $container->getCompiler()->getServiceReferenceGraph();
- $this->graph->clear();
- $this->lazy = false;
- foreach ($container->getAliases() as $id => $alias) {
- $this->graph->connect($id, $alias, (string) $alias, $this->getDefinition((string) $alias), null);
- }
- parent::process($container);
- }
- protected function processValue($value, $isRoot = false)
- {
- $lazy = $this->lazy;
- if ($value instanceof ArgumentInterface) {
- $this->lazy = true;
- parent::processValue($value->getValues());
- $this->lazy = $lazy;
- return $value;
- }
- if ($value instanceof Expression) {
- $this->getExpressionLanguage()->compile((string) $value, array('this' => 'container'));
- return $value;
- }
- if ($value instanceof Reference) {
- $targetDefinition = $this->getDefinition((string) $value);
- $this->graph->connect(
- $this->currentId,
- $this->currentDefinition,
- $this->getDefinitionId((string) $value),
- $targetDefinition,
- $value,
- $this->lazy || ($targetDefinition && $targetDefinition->isLazy())
- );
- return $value;
- }
- if (!$value instanceof Definition) {
- return parent::processValue($value, $isRoot);
- }
- if ($isRoot) {
- if ($value->isSynthetic() || $value->isAbstract()) {
- return $value;
- }
- $this->currentDefinition = $value;
- }
- $this->lazy = false;
- $this->processValue($value->getFactory());
- $this->processValue($value->getArguments());
- if (!$this->onlyConstructorArguments) {
- $this->processValue($value->getProperties());
- $this->processValue($value->getMethodCalls());
- $this->processValue($value->getConfigurator());
- }
- $this->lazy = $lazy;
- return $value;
- }
- /**
- * Returns a service definition given the full name or an alias.
- *
- * @param string $id A full id or alias for a service definition
- *
- * @return Definition|null The definition related to the supplied id
- */
- private function getDefinition($id)
- {
- $id = $this->getDefinitionId($id);
- return null === $id ? null : $this->container->getDefinition($id);
- }
- private function getDefinitionId($id)
- {
- while ($this->container->hasAlias($id)) {
- $id = (string) $this->container->getAlias($id);
- }
- if (!$this->container->hasDefinition($id)) {
- return;
- }
- return $id;
- }
- private function getExpressionLanguage()
- {
- if (null === $this->expressionLanguage) {
- $providers = $this->container->getExpressionLanguageProviders();
- $this->expressionLanguage = new ExpressionLanguage(null, $providers, function ($arg) {
- if ('""' === substr_replace($arg, '', 1, -1)) {
- $id = stripcslashes(substr($arg, 1, -1));
- $this->graph->connect(
- $this->currentId,
- $this->currentDefinition,
- $this->getDefinitionId($id),
- $this->getDefinition($id)
- );
- }
- return sprintf('$this->get(%s)', $arg);
- });
- }
- return $this->expressionLanguage;
- }
- }
|