ControllerArgumentValueResolverPass.php 3.0 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970
  1. <?php
  2. /*
  3. * This file is part of the Symfony package.
  4. *
  5. * (c) Fabien Potencier <fabien@symfony.com>
  6. *
  7. * For the full copyright and license information, please view the LICENSE
  8. * file that was distributed with this source code.
  9. */
  10. namespace Symfony\Component\HttpKernel\DependencyInjection;
  11. use Symfony\Component\DependencyInjection\Argument\IteratorArgument;
  12. use Symfony\Component\DependencyInjection\Argument\ServiceLocatorArgument;
  13. use Symfony\Component\DependencyInjection\Argument\TaggedIteratorArgument;
  14. use Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface;
  15. use Symfony\Component\DependencyInjection\Compiler\PriorityTaggedServiceTrait;
  16. use Symfony\Component\DependencyInjection\ContainerBuilder;
  17. use Symfony\Component\DependencyInjection\Reference;
  18. use Symfony\Component\HttpKernel\Controller\ArgumentResolver\TraceableValueResolver;
  19. use Symfony\Component\Stopwatch\Stopwatch;
  20. /**
  21. * Gathers and configures the argument value resolvers.
  22. *
  23. * @author Iltar van der Berg <kjarli@gmail.com>
  24. */
  25. class ControllerArgumentValueResolverPass implements CompilerPassInterface
  26. {
  27. use PriorityTaggedServiceTrait;
  28. public function process(ContainerBuilder $container): void
  29. {
  30. if (!$container->hasDefinition('argument_resolver')) {
  31. return;
  32. }
  33. $definitions = $container->getDefinitions();
  34. $namedResolvers = $this->findAndSortTaggedServices(new TaggedIteratorArgument('controller.targeted_value_resolver', 'name', needsIndexes: true), $container);
  35. $resolvers = $this->findAndSortTaggedServices(new TaggedIteratorArgument('controller.argument_value_resolver', 'name', needsIndexes: true), $container);
  36. foreach ($resolvers as $name => $resolver) {
  37. if ($definitions[(string) $resolver]->hasTag('controller.targeted_value_resolver')) {
  38. unset($resolvers[$name]);
  39. } else {
  40. $namedResolvers[$name] ??= clone $resolver;
  41. }
  42. }
  43. if ($container->getParameter('kernel.debug') && class_exists(Stopwatch::class) && $container->has('debug.stopwatch')) {
  44. foreach ($resolvers as $name => $resolver) {
  45. $resolvers[$name] = new Reference('.debug.value_resolver.'.$resolver);
  46. $container->register('.debug.value_resolver.'.$resolver, TraceableValueResolver::class)
  47. ->setArguments([$resolver, new Reference('debug.stopwatch')]);
  48. }
  49. foreach ($namedResolvers as $name => $resolver) {
  50. $namedResolvers[$name] = new Reference('.debug.value_resolver.'.$resolver);
  51. $container->register('.debug.value_resolver.'.$resolver, TraceableValueResolver::class)
  52. ->setArguments([$resolver, new Reference('debug.stopwatch')]);
  53. }
  54. }
  55. $container
  56. ->getDefinition('argument_resolver')
  57. ->replaceArgument(1, new IteratorArgument(array_values($resolvers)))
  58. ->setArgument(2, new ServiceLocatorArgument($namedResolvers))
  59. ;
  60. }
  61. }