Files
2025-10-08 11:39:17 -04:00

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;
}
}