242 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
		
		
			
		
	
	
			242 lines
		
	
	
		
			12 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\Serializer\DependencyInjection;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\Argument\BoundArgument;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\Compiler\PriorityTaggedServiceTrait;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\ContainerBuilder;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\Exception\RuntimeException;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\DependencyInjection\Reference;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Debug\TraceableEncoder;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Debug\TraceableNormalizer;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Normalizer\DenormalizerInterface;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Normalizer\NormalizerInterface;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Normalizer\ObjectNormalizer;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\SerializerInterface;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Adds all services with the tags "serializer.encoder" and "serializer.normalizer" as
							 | 
						||
| 
								 | 
							
								 * encoders and normalizers to the "serializer" service.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * @author Javier Lopez <f12loalf@gmail.com>
							 | 
						||
| 
								 | 
							
								 * @author Robin Chalas <robin.chalas@gmail.com>
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								class SerializerPass implements CompilerPassInterface
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    use PriorityTaggedServiceTrait;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private const NAME_CONVERTER_METADATA_AWARE_ID = 'serializer.name_converter.metadata_aware';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    public function process(ContainerBuilder $container): void
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if (!$container->hasDefinition('serializer')) {
							 | 
						||
| 
								 | 
							
								            return;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $namedSerializers = $container->hasParameter('.serializer.named_serializers')
							 | 
						||
| 
								 | 
							
								            ? $container->getParameter('.serializer.named_serializers') : [];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $this->createNamedSerializerTags($container, 'serializer.normalizer', 'include_built_in_normalizers', $namedSerializers);
							 | 
						||
| 
								 | 
							
								        $this->createNamedSerializerTags($container, 'serializer.encoder', 'include_built_in_encoders', $namedSerializers);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!$normalizers = $this->findAndSortTaggedServices('serializer.normalizer.default', $container)) {
							 | 
						||
| 
								 | 
							
								            throw new RuntimeException('You must tag at least one service as "serializer.normalizer" to use the "serializer" service.');
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!$encoders = $this->findAndSortTaggedServices('serializer.encoder.default', $container)) {
							 | 
						||
| 
								 | 
							
								            throw new RuntimeException('You must tag at least one service as "serializer.encoder" to use the "serializer" service.');
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $defaultContext = [];
							 | 
						||
| 
								 | 
							
								        if ($container->hasParameter('serializer.default_context')) {
							 | 
						||
| 
								 | 
							
								            $defaultContext = $container->getParameter('serializer.default_context');
							 | 
						||
| 
								 | 
							
								            $container->getParameterBag()->remove('serializer.default_context');
							 | 
						||
| 
								 | 
							
								            $container->getDefinition('serializer')->setArgument('$defaultContext', $defaultContext);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        /** @var ?string $circularReferenceHandler */
							 | 
						||
| 
								 | 
							
								        $circularReferenceHandler = $container->hasParameter('.serializer.circular_reference_handler')
							 | 
						||
| 
								 | 
							
								            ? $container->getParameter('.serializer.circular_reference_handler') : null;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        /** @var ?string $maxDepthHandler */
							 | 
						||
| 
								 | 
							
								        $maxDepthHandler = $container->hasParameter('.serializer.max_depth_handler')
							 | 
						||
| 
								 | 
							
								            ? $container->getParameter('.serializer.max_depth_handler') : null;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $this->bindDefaultContext($container, array_merge($normalizers, $encoders), $defaultContext, $circularReferenceHandler, $maxDepthHandler);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $this->configureSerializer($container, 'serializer', $normalizers, $encoders, 'default');
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if ($namedSerializers) {
							 | 
						||
| 
								 | 
							
								            $this->configureNamedSerializers($container, $circularReferenceHandler, $maxDepthHandler);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private function createNamedSerializerTags(ContainerBuilder $container, string $tagName, string $configName, array $namedSerializers): void
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $serializerNames = array_keys($namedSerializers);
							 | 
						||
| 
								 | 
							
								        $withBuiltIn = array_filter($serializerNames, fn (string $name) => $namedSerializers[$name][$configName] ?? false);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        foreach ($container->findTaggedServiceIds($tagName) as $serviceId => $tags) {
							 | 
						||
| 
								 | 
							
								            $definition = $container->getDefinition($serviceId);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if (array_any($tags, $closure = fn (array $tag) => (bool) $tag)) {
							 | 
						||
| 
								 | 
							
								                $tags = array_filter($tags, $closure);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            foreach ($tags as $tag) {
							 | 
						||
| 
								 | 
							
								                $names = (array) ($tag['serializer'] ?? []);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                if (!$names) {
							 | 
						||
| 
								 | 
							
								                    $names = ['default'];
							 | 
						||
| 
								 | 
							
								                } elseif (\in_array('*', $names, true)) {
							 | 
						||
| 
								 | 
							
								                    $names = array_unique(['default', ...$serializerNames]);
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                if ($tag['built_in'] ?? false) {
							 | 
						||
| 
								 | 
							
								                    $names = array_unique(['default', ...$names, ...$withBuiltIn]);
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                unset($tag['serializer'], $tag['built_in']);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                foreach ($names as $name) {
							 | 
						||
| 
								 | 
							
								                    $definition->addTag($tagName.'.'.$name, $tag);
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private function bindDefaultContext(ContainerBuilder $container, array $services, array $defaultContext, ?string $circularReferenceHandler, ?string $maxDepthHandler): void
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        foreach ($services as $id) {
							 | 
						||
| 
								 | 
							
								            $definition = $container->getDefinition((string) $id);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            $context = $defaultContext;
							 | 
						||
| 
								 | 
							
								            if (is_a($definition->getClass(), ObjectNormalizer::class, true)) {
							 | 
						||
| 
								 | 
							
								                if (null !== $circularReferenceHandler) {
							 | 
						||
| 
								 | 
							
								                    $context += ['circular_reference_handler' => new Reference($circularReferenceHandler)];
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								                if (null !== $maxDepthHandler) {
							 | 
						||
| 
								 | 
							
								                    $context += ['max_depth_handler' => new Reference($maxDepthHandler)];
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            $definition->setBindings(['array $defaultContext' => new BoundArgument($context, false)] + $definition->getBindings());
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private function configureSerializer(ContainerBuilder $container, string $id, array $normalizers, array $encoders, string $serializerName): void
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if ($container->getParameter('kernel.debug') && $container->hasDefinition('serializer.data_collector')) {
							 | 
						||
| 
								 | 
							
								            foreach ($normalizers as $i => $normalizer) {
							 | 
						||
| 
								 | 
							
								                $normalizers[$i] = $container->register('.debug.serializer.normalizer.'.$normalizer, TraceableNormalizer::class)
							 | 
						||
| 
								 | 
							
								                    ->setArguments([$normalizer, new Reference('serializer.data_collector'), $serializerName]);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            foreach ($encoders as $i => $encoder) {
							 | 
						||
| 
								 | 
							
								                $encoders[$i] = $container->register('.debug.serializer.encoder.'.$encoder, TraceableEncoder::class)
							 | 
						||
| 
								 | 
							
								                    ->setArguments([$encoder, new Reference('serializer.data_collector'), $serializerName]);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $serializerDefinition = $container->getDefinition($id);
							 | 
						||
| 
								 | 
							
								        $serializerDefinition->replaceArgument(0, $normalizers);
							 | 
						||
| 
								 | 
							
								        $serializerDefinition->replaceArgument(1, $encoders);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private function configureNamedSerializers(ContainerBuilder $container, ?string $circularReferenceHandler, ?string $maxDepthHandler): void
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $defaultSerializerNameConverter = $container->hasParameter('.serializer.name_converter')
							 | 
						||
| 
								 | 
							
								            ? $container->getParameter('.serializer.name_converter') : null;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        foreach ($container->getParameter('.serializer.named_serializers') as $serializerName => $config) {
							 | 
						||
| 
								 | 
							
								            $config += ['default_context' => [], 'name_converter' => null];
							 | 
						||
| 
								 | 
							
								            $serializerId = 'serializer.'.$serializerName;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if (!$normalizers = $this->findAndSortTaggedServices('serializer.normalizer.'.$serializerName, $container)) {
							 | 
						||
| 
								 | 
							
								                throw new RuntimeException(\sprintf('The named serializer "%1$s" requires at least one registered normalizer. Tag the normalizers as "serializer.normalizer" with the "serializer" attribute set to "%1$s".', $serializerName));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if (!$encoders = $this->findAndSortTaggedServices('serializer.encoder.'.$serializerName, $container)) {
							 | 
						||
| 
								 | 
							
								                throw new RuntimeException(\sprintf('The named serializer "%1$s" requires at least one registered encoder. Tag the encoders as "serializer.encoder" with the "serializer" attribute set to "%1$s".', $serializerName));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            $config['name_converter'] = $defaultSerializerNameConverter !== $config['name_converter']
							 | 
						||
| 
								 | 
							
								                ? $this->buildChildNameConverterDefinition($container, $config['name_converter'])
							 | 
						||
| 
								 | 
							
								                : self::NAME_CONVERTER_METADATA_AWARE_ID;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            $normalizers = $this->buildChildDefinitions($container, $serializerName, $normalizers, $config);
							 | 
						||
| 
								 | 
							
								            $encoders = $this->buildChildDefinitions($container, $serializerName, $encoders, $config);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            $this->bindDefaultContext($container, array_merge($normalizers, $encoders), $config['default_context'], $circularReferenceHandler, $maxDepthHandler);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            $container->registerChild($serializerId, 'serializer')->setArgument('$defaultContext', $config['default_context']);
							 | 
						||
| 
								 | 
							
								            $container->registerAliasForArgument($serializerId, SerializerInterface::class, $serializerName.'.serializer');
							 | 
						||
| 
								 | 
							
								            $container->registerAliasForArgument($serializerId, NormalizerInterface::class, $serializerName.'.normalizer');
							 | 
						||
| 
								 | 
							
								            $container->registerAliasForArgument($serializerId, DenormalizerInterface::class, $serializerName.'.denormalizer');
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            $this->configureSerializer($container, $serializerId, $normalizers, $encoders, $serializerName);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if ($container->getParameter('kernel.debug') && $container->hasDefinition('debug.serializer')) {
							 | 
						||
| 
								 | 
							
								                $container->registerChild($debugId = 'debug.'.$serializerId, 'debug.serializer')
							 | 
						||
| 
								 | 
							
								                    ->setDecoratedService($serializerId)
							 | 
						||
| 
								 | 
							
								                    ->replaceArgument(0, new Reference($debugId.'.inner'))
							 | 
						||
| 
								 | 
							
								                    ->replaceArgument(2, $serializerName);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private function buildChildNameConverterDefinition(ContainerBuilder $container, ?string $nameConverter): ?string
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $childId = self::NAME_CONVERTER_METADATA_AWARE_ID.'.'.ContainerBuilder::hash($nameConverter);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!$container->hasDefinition($childId)) {
							 | 
						||
| 
								 | 
							
								            $childDefinition = $container->registerChild($childId, self::NAME_CONVERTER_METADATA_AWARE_ID.'.abstract');
							 | 
						||
| 
								 | 
							
								            if (null !== $nameConverter) {
							 | 
						||
| 
								 | 
							
								                $childDefinition->addArgument(new Reference($nameConverter));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return $childId;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private function buildChildDefinitions(ContainerBuilder $container, string $serializerName, array $services, array $config): array
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        foreach ($services as &$id) {
							 | 
						||
| 
								 | 
							
								            $childId = $id.'.'.$serializerName;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            $definition = $container->registerChild($childId, (string) $id)
							 | 
						||
| 
								 | 
							
								                ->setClass($container->getDefinition((string) $id)->getClass())
							 | 
						||
| 
								 | 
							
								            ;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if (null !== $nameConverterIndex = $this->findNameConverterIndex($container, (string) $id)) {
							 | 
						||
| 
								 | 
							
								                $definition->replaceArgument($nameConverterIndex, new Reference($config['name_converter']));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            $id = new Reference($childId);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return $services;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private function findNameConverterIndex(ContainerBuilder $container, string $id): int|string|null
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        foreach ($container->getDefinition($id)->getArguments() as $index => $argument) {
							 | 
						||
| 
								 | 
							
								            if ($argument instanceof Reference && self::NAME_CONVERTER_METADATA_AWARE_ID === (string) $argument) {
							 | 
						||
| 
								 | 
							
								                return $index;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return null;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								}
							 |