172 lines
		
	
	
	
		
			5.8 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
		
		
			
		
	
	
			172 lines
		
	
	
	
		
			5.8 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\BoundArgument; | ||
|  | use Symfony\Component\DependencyInjection\ContainerBuilder; | ||
|  | use Symfony\Component\DependencyInjection\Definition; | ||
|  | use Symfony\Component\DependencyInjection\Exception\InvalidArgumentException; | ||
|  | use Symfony\Component\DependencyInjection\Exception\RuntimeException; | ||
|  | use Symfony\Component\DependencyInjection\LazyProxy\ProxyHelper; | ||
|  | use Symfony\Component\DependencyInjection\Reference; | ||
|  | use Symfony\Component\DependencyInjection\TypedReference; | ||
|  | 
 | ||
|  | /** | ||
|  |  * @author Guilhem Niot <guilhem.niot@gmail.com> | ||
|  |  */ | ||
|  | class ResolveBindingsPass extends AbstractRecursivePass | ||
|  | { | ||
|  |     private $usedBindings = array(); | ||
|  |     private $unusedBindings = array(); | ||
|  |     private $errorMessages = array(); | ||
|  | 
 | ||
|  |     /** | ||
|  |      * {@inheritdoc} | ||
|  |      */ | ||
|  |     public function process(ContainerBuilder $container) | ||
|  |     { | ||
|  |         try { | ||
|  |             parent::process($container); | ||
|  | 
 | ||
|  |             foreach ($this->unusedBindings as list($key, $serviceId)) { | ||
|  |                 $message = sprintf('Unused binding "%s" in service "%s".', $key, $serviceId); | ||
|  |                 if ($this->errorMessages) { | ||
|  |                     $message .= sprintf("\nCould be related to%s:", 1 < \count($this->errorMessages) ? ' one of' : ''); | ||
|  |                 } | ||
|  |                 foreach ($this->errorMessages as $m) { | ||
|  |                     $message .= "\n - ".$m; | ||
|  |                 } | ||
|  |                 throw new InvalidArgumentException($message); | ||
|  |             } | ||
|  |         } finally { | ||
|  |             $this->usedBindings = array(); | ||
|  |             $this->unusedBindings = array(); | ||
|  |             $this->errorMessages = array(); | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * {@inheritdoc} | ||
|  |      */ | ||
|  |     protected function processValue($value, $isRoot = false) | ||
|  |     { | ||
|  |         if ($value instanceof TypedReference && $value->getType() === $this->container->normalizeId($value)) { | ||
|  |             // Already checked
 | ||
|  |             $bindings = $this->container->getDefinition($this->currentId)->getBindings(); | ||
|  | 
 | ||
|  |             if (isset($bindings[$value->getType()])) { | ||
|  |                 return $this->getBindingValue($bindings[$value->getType()]); | ||
|  |             } | ||
|  | 
 | ||
|  |             return parent::processValue($value, $isRoot); | ||
|  |         } | ||
|  | 
 | ||
|  |         if (!$value instanceof Definition || !$bindings = $value->getBindings()) { | ||
|  |             return parent::processValue($value, $isRoot); | ||
|  |         } | ||
|  | 
 | ||
|  |         foreach ($bindings as $key => $binding) { | ||
|  |             list($bindingValue, $bindingId, $used) = $binding->getValues(); | ||
|  |             if ($used) { | ||
|  |                 $this->usedBindings[$bindingId] = true; | ||
|  |                 unset($this->unusedBindings[$bindingId]); | ||
|  |             } elseif (!isset($this->usedBindings[$bindingId])) { | ||
|  |                 $this->unusedBindings[$bindingId] = array($key, $this->currentId); | ||
|  |             } | ||
|  | 
 | ||
|  |             if (isset($key[0]) && '$' === $key[0]) { | ||
|  |                 continue; | ||
|  |             } | ||
|  | 
 | ||
|  |             if (null !== $bindingValue && !$bindingValue instanceof Reference && !$bindingValue instanceof Definition) { | ||
|  |                 throw new InvalidArgumentException(sprintf('Invalid value for binding key "%s" for service "%s": expected null, an instance of %s or an instance of %s, %s given.', $key, $this->currentId, Reference::class, Definition::class, \gettype($bindingValue))); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         if ($value->isAbstract()) { | ||
|  |             return parent::processValue($value, $isRoot); | ||
|  |         } | ||
|  | 
 | ||
|  |         $calls = $value->getMethodCalls(); | ||
|  | 
 | ||
|  |         try { | ||
|  |             if ($constructor = $this->getConstructor($value, false)) { | ||
|  |                 $calls[] = array($constructor, $value->getArguments()); | ||
|  |             } | ||
|  |         } catch (RuntimeException $e) { | ||
|  |             $this->errorMessages[] = $e->getMessage(); | ||
|  |             $this->container->getDefinition($this->currentId)->addError($e->getMessage()); | ||
|  | 
 | ||
|  |             return parent::processValue($value, $isRoot); | ||
|  |         } | ||
|  | 
 | ||
|  |         foreach ($calls as $i => $call) { | ||
|  |             list($method, $arguments) = $call; | ||
|  | 
 | ||
|  |             if ($method instanceof \ReflectionFunctionAbstract) { | ||
|  |                 $reflectionMethod = $method; | ||
|  |             } else { | ||
|  |                 $reflectionMethod = $this->getReflectionMethod($value, $method); | ||
|  |             } | ||
|  | 
 | ||
|  |             foreach ($reflectionMethod->getParameters() as $key => $parameter) { | ||
|  |                 if (array_key_exists($key, $arguments) && '' !== $arguments[$key]) { | ||
|  |                     continue; | ||
|  |                 } | ||
|  | 
 | ||
|  |                 if (array_key_exists('$'.$parameter->name, $bindings)) { | ||
|  |                     $arguments[$key] = $this->getBindingValue($bindings['$'.$parameter->name]); | ||
|  | 
 | ||
|  |                     continue; | ||
|  |                 } | ||
|  | 
 | ||
|  |                 $typeHint = ProxyHelper::getTypeHint($reflectionMethod, $parameter, true); | ||
|  | 
 | ||
|  |                 if (!isset($bindings[$typeHint])) { | ||
|  |                     continue; | ||
|  |                 } | ||
|  | 
 | ||
|  |                 $arguments[$key] = $this->getBindingValue($bindings[$typeHint]); | ||
|  |             } | ||
|  | 
 | ||
|  |             if ($arguments !== $call[1]) { | ||
|  |                 ksort($arguments); | ||
|  |                 $calls[$i][1] = $arguments; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         if ($constructor) { | ||
|  |             list(, $arguments) = array_pop($calls); | ||
|  | 
 | ||
|  |             if ($arguments !== $value->getArguments()) { | ||
|  |                 $value->setArguments($arguments); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         if ($calls !== $value->getMethodCalls()) { | ||
|  |             $value->setMethodCalls($calls); | ||
|  |         } | ||
|  | 
 | ||
|  |         return parent::processValue($value, $isRoot); | ||
|  |     } | ||
|  | 
 | ||
|  |     private function getBindingValue(BoundArgument $binding) | ||
|  |     { | ||
|  |         list($bindingValue, $bindingId) = $binding->getValues(); | ||
|  | 
 | ||
|  |         $this->usedBindings[$bindingId] = true; | ||
|  |         unset($this->unusedBindings[$bindingId]); | ||
|  | 
 | ||
|  |         return $bindingValue; | ||
|  |     } | ||
|  | } |