• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

api-platform / core / 19799301771

30 Nov 2025 01:04PM UTC coverage: 25.229% (-0.03%) from 25.257%
19799301771

push

github

web-flow
fix(metadata): repeatable attribute mutators (#7542)

14557 of 57700 relevant lines covered (25.23%)

28.11 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

82.05
/src/GraphQl/Serializer/SerializerContextBuilder.php
1
<?php
2

3
/*
4
 * This file is part of the API Platform project.
5
 *
6
 * (c) Kévin Dunglas <dunglas@gmail.com>
7
 *
8
 * For the full copyright and license information, please view the LICENSE
9
 * file that was distributed with this source code.
10
 */
11

12
declare(strict_types=1);
13

14
namespace ApiPlatform\GraphQl\Serializer;
15

16
use ApiPlatform\Metadata\GraphQl\Mutation;
17
use ApiPlatform\Metadata\GraphQl\Operation;
18
use ApiPlatform\Metadata\GraphQl\Subscription;
19
use GraphQL\Type\Definition\ResolveInfo;
20
use Symfony\Component\Serializer\NameConverter\NameConverterInterface;
21
use Symfony\Component\Serializer\Normalizer\AbstractObjectNormalizer;
22

23
/**
24
 * Builds the context used by the Symfony Serializer.
25
 *
26
 * @author Alan Poulain <contact@alanpoulain.eu>
27
 */
28
final class SerializerContextBuilder implements SerializerContextBuilderInterface
29
{
30
    public function __construct(private readonly ?NameConverterInterface $nameConverter)
31
    {
32
    }
352✔
33

34
    public function create(?string $resourceClass, Operation $operation, array $resolverContext, bool $normalization): array
35
    {
36
        $context = ['resource_class' => $resourceClass, 'operation_name' => $operation->getName(), 'graphql_operation_name' => $operation->getName()];
26✔
37

38
        if (isset($resolverContext['fields'])) {
26✔
39
            $context['no_resolver_data'] = true;
×
40
        }
41

42
        $context['operation'] = $operation;
26✔
43
        if ($operation->getInput()) {
26✔
44
            $context['input'] = $operation->getInput();
×
45
        }
46
        if ($operation->getOutput()) {
26✔
47
            $context['output'] = $operation->getOutput();
×
48
        }
49
        $context = $normalization ? array_merge($operation->getNormalizationContext() ?? [], $context) : array_merge($operation->getDenormalizationContext() ?? [], $context);
26✔
50

51
        if ($normalization) {
26✔
52
            $context['attributes'] = $this->fieldsToAttributes($resourceClass, $operation, $resolverContext, $context);
26✔
53
        }
54

55
        // to keep the cache computation smaller, we have "operation_name" and "iri" anyways
56
        $context[AbstractObjectNormalizer::EXCLUDE_FROM_CACHE_KEY][] = 'root_operation';
26✔
57
        $context[AbstractObjectNormalizer::EXCLUDE_FROM_CACHE_KEY][] = 'operation';
26✔
58
        $context[AbstractObjectNormalizer::EXCLUDE_FROM_CACHE_KEY][] = 'object';
26✔
59
        $context[AbstractObjectNormalizer::EXCLUDE_FROM_CACHE_KEY][] = 'data';
26✔
60
        $context[AbstractObjectNormalizer::EXCLUDE_FROM_CACHE_KEY][] = 'property_metadata';
26✔
61
        $context[AbstractObjectNormalizer::EXCLUDE_FROM_CACHE_KEY][] = 'circular_reference_limit_counters';
26✔
62
        $context[AbstractObjectNormalizer::EXCLUDE_FROM_CACHE_KEY][] = 'debug_trace_id';
26✔
63

64
        return $context;
26✔
65
    }
66

67
    /**
68
     * Retrieves fields, recursively replaces the "_id" key (the raw id) by "id" (the name of the property expected by the Serializer) and flattens edge and node structures (pagination).
69
     */
70
    private function fieldsToAttributes(?string $resourceClass, Operation $operation, array $resolverContext, array $context): array
71
    {
72
        if (isset($resolverContext['fields'])) {
26✔
73
            $fields = $resolverContext['fields'];
×
74
        } else {
75
            /** @var ResolveInfo $info */
76
            $info = $resolverContext['info'];
26✔
77
            $fields = $info->getFieldSelection(\PHP_INT_MAX);
26✔
78
        }
79

80
        $attributes = $this->replaceIdKeys($fields['edges']['node'] ?? $fields['collection'] ?? $fields, $resourceClass, $context);
26✔
81

82
        if ($operation instanceof Subscription || $operation instanceof Mutation) {
26✔
83
            $wrapFieldName = lcfirst($operation->getShortName());
2✔
84

85
            return $attributes[$wrapFieldName] ?? [];
2✔
86
        }
87

88
        return $attributes;
24✔
89
    }
90

91
    private function replaceIdKeys(array $fields, ?string $resourceClass, array $context): array
92
    {
93
        $denormalizedFields = [];
26✔
94

95
        foreach ($fields as $key => $value) {
26✔
96
            if ('_id' === $key) {
26✔
97
                $denormalizedFields['id'] = $fields['_id'];
×
98

99
                continue;
×
100
            }
101

102
            $denormalizedFields[$this->denormalizePropertyName((string) $key, $resourceClass, $context)] = \is_array($value) ? $this->replaceIdKeys($value, $resourceClass, $context) : $value;
26✔
103
        }
104

105
        return $denormalizedFields;
26✔
106
    }
107

108
    private function denormalizePropertyName(string $property, ?string $resourceClass, array $context): string
109
    {
110
        if (null === $this->nameConverter) {
26✔
111
            return $property;
×
112
        }
113

114
        return $this->nameConverter->denormalize($property, $resourceClass, null, $context);
26✔
115
    }
116
}
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc