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

api-platform / core / 21006071786

14 Jan 2026 06:53PM UTC coverage: 21.366% (-7.7%) from 29.097%
21006071786

Pull #7675

github

web-flow
Merge 5c98a7330 into 73402fc61
Pull Request #7675: feat(doctrine): Add caseInsensitive option to PartialSearchFilter

0 of 11 new or added lines in 2 files covered. (0.0%)

15054 existing lines in 491 files now uncovered.

12306 of 57596 relevant lines covered (21.37%)

49.15 hits per line

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

0.0
/src/Symfony/Bundle/DependencyInjection/ApiPlatformExtension.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\Symfony\Bundle\DependencyInjection;
15

16
use ApiPlatform\Doctrine\Odm\Extension\AggregationCollectionExtensionInterface;
17
use ApiPlatform\Doctrine\Odm\Extension\AggregationItemExtensionInterface;
18
use ApiPlatform\Doctrine\Odm\Filter\AbstractFilter as DoctrineMongoDbOdmAbstractFilter;
19
use ApiPlatform\Doctrine\Odm\State\LinksHandlerInterface as OdmLinksHandlerInterface;
20
use ApiPlatform\Doctrine\Orm\Extension\EagerLoadingExtension;
21
use ApiPlatform\Doctrine\Orm\Extension\FilterEagerLoadingExtension;
22
use ApiPlatform\Doctrine\Orm\Extension\QueryCollectionExtensionInterface as DoctrineQueryCollectionExtensionInterface;
23
use ApiPlatform\Doctrine\Orm\Extension\QueryItemExtensionInterface;
24
use ApiPlatform\Doctrine\Orm\Filter\AbstractFilter as DoctrineOrmAbstractFilter;
25
use ApiPlatform\Doctrine\Orm\State\LinksHandlerInterface as OrmLinksHandlerInterface;
26
use ApiPlatform\Elasticsearch\Extension\RequestBodySearchCollectionExtensionInterface;
27
use ApiPlatform\GraphQl\Error\ErrorHandlerInterface;
28
use ApiPlatform\GraphQl\Executor;
29
use ApiPlatform\GraphQl\Resolver\MutationResolverInterface;
30
use ApiPlatform\GraphQl\Resolver\QueryCollectionResolverInterface;
31
use ApiPlatform\GraphQl\Resolver\QueryItemResolverInterface;
32
use ApiPlatform\GraphQl\Type\Definition\TypeInterface as GraphQlTypeInterface;
33
use ApiPlatform\Metadata\ApiResource;
34
use ApiPlatform\Metadata\AsOperationMutator;
35
use ApiPlatform\Metadata\AsResourceMutator;
36
use ApiPlatform\Metadata\FilterInterface;
37
use ApiPlatform\Metadata\OperationMutatorInterface;
38
use ApiPlatform\Metadata\ResourceMutatorInterface;
39
use ApiPlatform\Metadata\UriVariableTransformerInterface;
40
use ApiPlatform\Metadata\UrlGeneratorInterface;
41
use ApiPlatform\OpenApi\Model\Tag;
42
use ApiPlatform\RamseyUuid\Serializer\UuidDenormalizer;
43
use ApiPlatform\State\ApiResource\Error;
44
use ApiPlatform\State\ParameterProviderInterface;
45
use ApiPlatform\State\ProcessorInterface;
46
use ApiPlatform\State\ProviderInterface;
47
use ApiPlatform\Symfony\Validator\Metadata\Property\Restriction\PropertySchemaRestrictionMetadataInterface;
48
use ApiPlatform\Symfony\Validator\ValidationGroupsGeneratorInterface;
49
use ApiPlatform\Validator\Exception\ValidationException;
50
use Composer\InstalledVersions;
51
use Doctrine\Persistence\ManagerRegistry;
52
use PHPStan\PhpDocParser\Parser\PhpDocParser;
53
use Ramsey\Uuid\Uuid;
54
use Symfony\Bundle\FrameworkBundle\Command\TranslationExtractCommand;
55
use Symfony\Bundle\FrameworkBundle\Controller\ControllerHelper;
56
use Symfony\Component\Config\FileLocator;
57
use Symfony\Component\Config\Resource\DirectoryResource;
58
use Symfony\Component\DependencyInjection\ChildDefinition;
59
use Symfony\Component\DependencyInjection\ContainerBuilder;
60
use Symfony\Component\DependencyInjection\Definition;
61
use Symfony\Component\DependencyInjection\Exception\RuntimeException;
62
use Symfony\Component\DependencyInjection\Extension\Extension;
63
use Symfony\Component\DependencyInjection\Extension\PrependExtensionInterface;
64
use Symfony\Component\DependencyInjection\Loader\PhpFileLoader;
65
use Symfony\Component\DependencyInjection\Reference;
66
use Symfony\Component\Finder\Finder;
67
use Symfony\Component\HttpClient\ScopingHttpClient;
68
use Symfony\Component\JsonStreamer\JsonStreamWriter;
69
use Symfony\Component\ObjectMapper\ObjectMapper;
70
use Symfony\Component\Serializer\NameConverter\CamelCaseToSnakeCaseNameConverter;
71
use Symfony\Component\Serializer\Normalizer\NumberNormalizer;
72
use Symfony\Component\Uid\AbstractUid;
73
use Symfony\Component\Validator\Validator\ValidatorInterface;
74
use Symfony\Component\Yaml\Yaml;
75
use Twig\Environment;
76

77
/**
78
 * The extension of this bundle.
79
 *
80
 * @author Kévin Dunglas <dunglas@gmail.com>
81
 */
82
final class ApiPlatformExtension extends Extension implements PrependExtensionInterface
83
{
84
    /**
85
     * {@inheritdoc}
86
     */
87
    public function prepend(ContainerBuilder $container): void
88
    {
UNCOV
89
        if (isset($container->getExtensions()['framework'])) {
×
UNCOV
90
            $container->prependExtensionConfig('framework', [
×
UNCOV
91
                'serializer' => [
×
UNCOV
92
                    'enabled' => true,
×
UNCOV
93
                ],
×
UNCOV
94
            ]);
×
UNCOV
95
            $container->prependExtensionConfig('framework', [
×
UNCOV
96
                'property_info' => [
×
UNCOV
97
                    'enabled' => true,
×
UNCOV
98
                ],
×
UNCOV
99
            ]);
×
100
        }
UNCOV
101
        if (isset($container->getExtensions()['lexik_jwt_authentication'])) {
×
102
            $container->prependExtensionConfig('lexik_jwt_authentication', [
×
103
                'api_platform' => [
×
104
                    'enabled' => true,
×
105
                ],
×
106
            ]);
×
107
        }
108
    }
109

110
    /**
111
     * {@inheritdoc}
112
     */
113
    public function load(array $configs, ContainerBuilder $container): void
114
    {
UNCOV
115
        $loader = new PhpFileLoader($container, new FileLocator(__DIR__.'/../Resources/config'));
×
116

UNCOV
117
        $configuration = new Configuration();
×
UNCOV
118
        $config = $this->processConfiguration($configuration, $configs);
×
UNCOV
119
        $container->setParameter('api_platform.use_symfony_listeners', $config['use_symfony_listeners']);
×
120

UNCOV
121
        $formats = $this->getFormats($config['formats']);
×
UNCOV
122
        $patchFormats = $this->getFormats($config['patch_formats']);
×
UNCOV
123
        $errorFormats = $this->getFormats($config['error_formats']);
×
UNCOV
124
        $docsFormats = $this->getFormats($config['docs_formats']);
×
125

UNCOV
126
        if (!$config['enable_docs']) {
×
127
            // JSON-LD documentation format is mandatory, even if documentation is disabled.
128
            $docsFormats = isset($formats['jsonld']) ? ['jsonld' => ['application/ld+json']] : [];
×
129
            // If documentation is disabled, the Hydra documentation for all the resources is hidden by default.
130
            if (!isset($config['defaults']['hideHydraOperation']) && !isset($config['defaults']['hide_hydra_operation'])) {
×
131
                $config['defaults']['hideHydraOperation'] = true;
×
132
            }
133
        }
UNCOV
134
        $jsonSchemaFormats = $config['jsonschema_formats'];
×
135

UNCOV
136
        if (!$jsonSchemaFormats) {
×
UNCOV
137
            foreach (array_merge(array_keys($formats), array_keys($errorFormats)) as $f) {
×
138
                // Distinct JSON-based formats must have names that start with 'json'
UNCOV
139
                if (str_starts_with($f, 'json')) {
×
UNCOV
140
                    $jsonSchemaFormats[$f] = true;
×
141
                }
142
            }
143
        }
144

UNCOV
145
        if (!isset($errorFormats['json'])) {
×
UNCOV
146
            $errorFormats['json'] = ['application/problem+json', 'application/json'];
×
147
        }
148

UNCOV
149
        if (!isset($errorFormats['jsonproblem'])) {
×
150
            $errorFormats['jsonproblem'] = ['application/problem+json'];
×
151
        }
152

UNCOV
153
        if (isset($formats['jsonapi']) && !isset($patchFormats['jsonapi'])) {
×
UNCOV
154
            $patchFormats['jsonapi'] = ['application/vnd.api+json'];
×
155
        }
156

UNCOV
157
        $this->registerCommonConfiguration($container, $config, $loader, $formats, $patchFormats, $errorFormats, $docsFormats);
×
UNCOV
158
        $this->registerMetadataConfiguration($container, $config, $loader);
×
UNCOV
159
        $this->registerOAuthConfiguration($container, $config);
×
UNCOV
160
        $this->registerOpenApiConfiguration($container, $config, $loader);
×
UNCOV
161
        $this->registerSwaggerConfiguration($container, $config, $loader);
×
UNCOV
162
        $this->registerJsonApiConfiguration($formats, $loader, $config);
×
UNCOV
163
        $this->registerJsonLdHydraConfiguration($container, $formats, $loader, $config);
×
UNCOV
164
        $this->registerJsonHalConfiguration($formats, $loader);
×
UNCOV
165
        $this->registerJsonProblemConfiguration($errorFormats, $loader);
×
UNCOV
166
        $this->registerGraphQlConfiguration($container, $config, $loader);
×
UNCOV
167
        $this->registerCacheConfiguration($container);
×
UNCOV
168
        $this->registerDoctrineOrmConfiguration($container, $config, $loader);
×
UNCOV
169
        $this->registerDoctrineMongoDbOdmConfiguration($container, $config, $loader);
×
UNCOV
170
        $this->registerHttpCacheConfiguration($container, $config, $loader);
×
UNCOV
171
        $this->registerValidatorConfiguration($container, $config, $loader);
×
UNCOV
172
        $this->registerDataCollectorConfiguration($container, $config, $loader);
×
UNCOV
173
        $this->registerMercureConfiguration($container, $config, $loader);
×
UNCOV
174
        $this->registerMessengerConfiguration($container, $config, $loader);
×
UNCOV
175
        $this->registerElasticsearchConfiguration($container, $config, $loader);
×
UNCOV
176
        $this->registerSecurityConfiguration($container, $config, $loader);
×
UNCOV
177
        $this->registerMakerConfiguration($container, $config, $loader);
×
UNCOV
178
        $this->registerArgumentResolverConfiguration($loader);
×
UNCOV
179
        $this->registerLinkSecurityConfiguration($loader, $config);
×
UNCOV
180
        $this->registerJsonStreamerConfiguration($container, $loader, $formats, $config);
×
181

182
        // TranslationExtractCommand was introduced in framework-bundle/7.3 with the object mapper service
UNCOV
183
        if (class_exists(ObjectMapper::class) && class_exists(TranslationExtractCommand::class)) {
×
UNCOV
184
            $loader->load('state/object_mapper.php');
×
185
        }
UNCOV
186
        $container->registerForAutoconfiguration(FilterInterface::class)
×
UNCOV
187
            ->addTag('api_platform.filter');
×
UNCOV
188
        $container->registerForAutoconfiguration(ProviderInterface::class)
×
UNCOV
189
            ->addTag('api_platform.state_provider');
×
UNCOV
190
        $container->registerForAutoconfiguration(ProcessorInterface::class)
×
UNCOV
191
            ->addTag('api_platform.state_processor');
×
UNCOV
192
        $container->registerForAutoconfiguration(UriVariableTransformerInterface::class)
×
UNCOV
193
            ->addTag('api_platform.uri_variables.transformer');
×
UNCOV
194
        $container->registerForAutoconfiguration(ParameterProviderInterface::class)
×
UNCOV
195
            ->addTag('api_platform.parameter_provider');
×
UNCOV
196
        $container->registerAttributeForAutoconfiguration(ApiResource::class, static function (ChildDefinition $definition): void {
×
197
            $definition->setAbstract(true)
×
198
                ->addTag('api_platform.resource')
×
199
                ->addTag('container.excluded', ['source' => 'by #[ApiResource] attribute']);
×
UNCOV
200
        });
×
UNCOV
201
        $container->registerAttributeForAutoconfiguration(
×
UNCOV
202
            AsResourceMutator::class,
×
UNCOV
203
            static function (ChildDefinition $definition, AsResourceMutator $attribute, \ReflectionClass $reflector): void { // @phpstan-ignore-line
×
204
                if (!is_a($reflector->name, ResourceMutatorInterface::class, true)) {
×
205
                    throw new RuntimeException(\sprintf('Resource mutator "%s" should implement %s', $reflector->name, ResourceMutatorInterface::class));
×
206
                }
207

208
                $definition->addTag('api_platform.resource_mutator', [
×
209
                    'resourceClass' => $attribute->resourceClass,
×
210
                ]);
×
UNCOV
211
            },
×
UNCOV
212
        );
×
213

UNCOV
214
        $container->registerAttributeForAutoconfiguration(
×
UNCOV
215
            AsOperationMutator::class,
×
UNCOV
216
            static function (ChildDefinition $definition, AsOperationMutator $attribute, \ReflectionClass $reflector): void { // @phpstan-ignore-line
×
217
                if (!is_a($reflector->name, OperationMutatorInterface::class, true)) {
×
218
                    throw new RuntimeException(\sprintf('Operation mutator "%s" should implement %s', $reflector->name, OperationMutatorInterface::class));
×
219
                }
220

221
                $definition->addTag('api_platform.operation_mutator', [
×
222
                    'operationName' => $attribute->operationName,
×
223
                ]);
×
UNCOV
224
            },
×
UNCOV
225
        );
×
226

UNCOV
227
        if (!$container->has('api_platform.state.item_provider')) {
×
228
            $container->setAlias('api_platform.state.item_provider', 'api_platform.state_provider.object');
×
229
        }
230

UNCOV
231
        if ($container->getParameter('kernel.debug')) {
×
UNCOV
232
            $this->injectStopwatch($container);
×
233
        }
234
    }
235

236
    private function injectStopwatch(ContainerBuilder $container): void
237
    {
UNCOV
238
        $services = [
×
UNCOV
239
            'api_platform.state_processor.add_link_header',
×
UNCOV
240
            'api_platform.state_processor.respond',
×
UNCOV
241
            'api_platform.state_processor.serialize',
×
UNCOV
242
            'api_platform.state_processor.write',
×
UNCOV
243
            'api_platform.state_provider.content_negotiation',
×
UNCOV
244
            'api_platform.state_provider.deserialize',
×
UNCOV
245
            'api_platform.state_provider.parameter',
×
UNCOV
246
            'api_platform.state_provider.read',
×
UNCOV
247
        ];
×
248

UNCOV
249
        foreach ($services as $id) {
×
UNCOV
250
            if (!$container->hasDefinition($id)) {
×
251
                continue;
×
252
            }
253

UNCOV
254
            $definition = $container->getDefinition($id);
×
UNCOV
255
            $definition->addMethodCall('setStopwatch', [new Reference('debug.stopwatch', ContainerBuilder::IGNORE_ON_INVALID_REFERENCE)]);
×
256
        }
257
    }
258

259
    private function registerCommonConfiguration(ContainerBuilder $container, array $config, PhpFileLoader $loader, array $formats, array $patchFormats, array $errorFormats, array $docsFormats): void
260
    {
UNCOV
261
        $loader->load('state/state.php');
×
UNCOV
262
        $loader->load('symfony/symfony.php');
×
UNCOV
263
        $loader->load('api.php');
×
UNCOV
264
        $loader->load('filter.php');
×
265

UNCOV
266
        if (class_exists(UuidDenormalizer::class) && class_exists(Uuid::class)) {
×
UNCOV
267
            $loader->load('ramsey_uuid.php');
×
268
        }
269

UNCOV
270
        if (class_exists(AbstractUid::class)) {
×
UNCOV
271
            $loader->load('symfony/uid.php');
×
272
        }
273

274
        // symfony/serializer:7.3 added the NumberNormalizer
275
        // symfony/framework-bundle:7.3 added the serializer.normalizer.number` service
276
        // if symfony/serializer >= 7.3 and symfony/framework-bundle < 7.3, the service is registred
UNCOV
277
        if (class_exists(NumberNormalizer::class) && !$container->has('serializer.normalizer.number')) {
×
UNCOV
278
            $numberNormalizerDefinition = new Definition(NumberNormalizer::class);
×
UNCOV
279
            $numberNormalizerDefinition->addTag('serializer.normalizer', ['built_in' => true, 'priority' => -915]);
×
UNCOV
280
            $container->setDefinition('serializer.normalizer.number', $numberNormalizerDefinition);
×
281
        }
282

UNCOV
283
        $defaultContext = ['hydra_prefix' => $config['serializer']['hydra_prefix']] + ($container->hasParameter('serializer.default_context') ? $container->getParameter('serializer.default_context') : []);
×
284

UNCOV
285
        $container->setParameter('api_platform.serializer.default_context', $defaultContext);
×
UNCOV
286
        if (!$container->hasParameter('serializer.default_context')) {
×
UNCOV
287
            $container->setParameter('serializer.default_context', $container->getParameter('api_platform.serializer.default_context'));
×
288
        }
UNCOV
289
        if ($config['use_symfony_listeners']) {
×
UNCOV
290
            $loader->load('symfony/events.php');
×
291
        } else {
UNCOV
292
            $loader->load('symfony/controller.php');
×
UNCOV
293
            $loader->load('state/provider.php');
×
UNCOV
294
            $loader->load('state/processor.php');
×
295
        }
UNCOV
296
        $loader->load('state/parameter_provider.php');
×
297

UNCOV
298
        $container->setParameter('api_platform.enable_entrypoint', $config['enable_entrypoint']);
×
UNCOV
299
        $container->setParameter('api_platform.enable_docs', $config['enable_docs']);
×
UNCOV
300
        $container->setParameter('api_platform.title', $config['title']);
×
UNCOV
301
        $container->setParameter('api_platform.description', $config['description']);
×
UNCOV
302
        $container->setParameter('api_platform.version', $config['version']);
×
UNCOV
303
        $container->setParameter('api_platform.show_webby', $config['show_webby']);
×
UNCOV
304
        $container->setParameter('api_platform.url_generation_strategy', $config['defaults']['url_generation_strategy'] ?? UrlGeneratorInterface::ABS_PATH);
×
UNCOV
305
        $container->setParameter('api_platform.exception_to_status', $config['exception_to_status']);
×
UNCOV
306
        $container->setParameter('api_platform.formats', $formats);
×
UNCOV
307
        $container->setParameter('api_platform.patch_formats', $patchFormats);
×
UNCOV
308
        $container->setParameter('api_platform.error_formats', $errorFormats);
×
UNCOV
309
        $container->setParameter('api_platform.docs_formats', $docsFormats);
×
UNCOV
310
        $container->setParameter('api_platform.jsonschema_formats', []);
×
UNCOV
311
        $container->setParameter('api_platform.eager_loading.enabled', $this->isConfigEnabled($container, $config['eager_loading']));
×
UNCOV
312
        $container->setParameter('api_platform.eager_loading.max_joins', $config['eager_loading']['max_joins']);
×
UNCOV
313
        $container->setParameter('api_platform.eager_loading.fetch_partial', $config['eager_loading']['fetch_partial']);
×
UNCOV
314
        $container->setParameter('api_platform.eager_loading.force_eager', $config['eager_loading']['force_eager']);
×
UNCOV
315
        $container->setParameter('api_platform.collection.exists_parameter_name', $config['collection']['exists_parameter_name']);
×
UNCOV
316
        $container->setParameter('api_platform.collection.order', $config['collection']['order']);
×
UNCOV
317
        $container->setParameter('api_platform.collection.order_parameter_name', $config['collection']['order_parameter_name']);
×
UNCOV
318
        $container->setParameter('api_platform.collection.order_nulls_comparison', $config['collection']['order_nulls_comparison']);
×
UNCOV
319
        $container->setParameter('api_platform.collection.pagination.enabled', $config['defaults']['pagination_enabled'] ?? true);
×
UNCOV
320
        $container->setParameter('api_platform.collection.pagination.partial', $config['defaults']['pagination_partial'] ?? false);
×
UNCOV
321
        $container->setParameter('api_platform.collection.pagination.client_enabled', $config['defaults']['pagination_client_enabled'] ?? false);
×
UNCOV
322
        $container->setParameter('api_platform.collection.pagination.client_items_per_page', $config['defaults']['pagination_client_items_per_page'] ?? false);
×
UNCOV
323
        $container->setParameter('api_platform.collection.pagination.client_partial', $config['defaults']['pagination_client_partial'] ?? false);
×
UNCOV
324
        $container->setParameter('api_platform.collection.pagination.items_per_page', $config['defaults']['pagination_items_per_page'] ?? 30);
×
UNCOV
325
        $container->setParameter('api_platform.collection.pagination.maximum_items_per_page', \array_key_exists('pagination_maximum_items_per_page', $config['defaults'] ?? []) ? $config['defaults']['pagination_maximum_items_per_page'] : 30);
×
UNCOV
326
        $container->setParameter('api_platform.collection.pagination.page_parameter_name', $config['defaults']['pagination_page_parameter_name'] ?? $config['collection']['pagination']['page_parameter_name']);
×
UNCOV
327
        $container->setParameter('api_platform.collection.pagination.enabled_parameter_name', $config['defaults']['pagination_enabled_parameter_name'] ?? $config['collection']['pagination']['enabled_parameter_name']);
×
UNCOV
328
        $container->setParameter('api_platform.collection.pagination.items_per_page_parameter_name', $config['defaults']['pagination_items_per_page_parameter_name'] ?? $config['collection']['pagination']['items_per_page_parameter_name']);
×
UNCOV
329
        $container->setParameter('api_platform.collection.pagination.partial_parameter_name', $config['defaults']['pagination_partial_parameter_name'] ?? $config['collection']['pagination']['partial_parameter_name']);
×
UNCOV
330
        $container->setParameter('api_platform.collection.pagination', $this->getPaginationDefaults($config['defaults'] ?? [], $config['collection']['pagination']));
×
UNCOV
331
        $container->setParameter('api_platform.handle_symfony_errors', $config['handle_symfony_errors'] ?? false);
×
UNCOV
332
        $container->setParameter('api_platform.http_cache.etag', $config['defaults']['cache_headers']['etag'] ?? true);
×
UNCOV
333
        $container->setParameter('api_platform.http_cache.max_age', $config['defaults']['cache_headers']['max_age'] ?? null);
×
UNCOV
334
        $container->setParameter('api_platform.http_cache.shared_max_age', $config['defaults']['cache_headers']['shared_max_age'] ?? null);
×
UNCOV
335
        $container->setParameter('api_platform.http_cache.vary', $config['defaults']['cache_headers']['vary'] ?? ['Accept']);
×
UNCOV
336
        $container->setParameter('api_platform.http_cache.public', $config['defaults']['cache_headers']['public'] ?? $config['http_cache']['public']);
×
UNCOV
337
        $container->setParameter('api_platform.http_cache.stale_while_revalidate', $config['defaults']['cache_headers']['stale_while_revalidate'] ?? null);
×
UNCOV
338
        $container->setParameter('api_platform.http_cache.stale_if_error', $config['defaults']['cache_headers']['stale_if_error'] ?? null);
×
UNCOV
339
        $container->setParameter('api_platform.http_cache.invalidation.max_header_length', $config['defaults']['cache_headers']['invalidation']['max_header_length'] ?? $config['http_cache']['invalidation']['max_header_length']);
×
UNCOV
340
        $container->setParameter('api_platform.http_cache.invalidation.xkey.glue', $config['defaults']['cache_headers']['invalidation']['xkey']['glue'] ?? $config['http_cache']['invalidation']['xkey']['glue']);
×
341

UNCOV
342
        $container->setAlias('api_platform.path_segment_name_generator', $config['path_segment_name_generator']);
×
UNCOV
343
        $container->setAlias('api_platform.inflector', $config['inflector']);
×
344

UNCOV
345
        if ($config['name_converter']) {
×
UNCOV
346
            $container->setAlias('api_platform.name_converter', $config['name_converter']);
×
347
        }
UNCOV
348
        $container->setParameter('api_platform.asset_package', $config['asset_package']);
×
UNCOV
349
        $container->setParameter('api_platform.defaults', $this->normalizeDefaults($config['defaults'] ?? []));
×
350

UNCOV
351
        if ($container->getParameter('kernel.debug')) {
×
UNCOV
352
            $container->removeDefinition('api_platform.serializer.mapping.cache_class_metadata_factory');
×
353
        }
354
    }
355

356
    /**
357
     * This method will be removed in 3.0 when "defaults" will be the regular configuration path for the pagination.
358
     */
359
    private function getPaginationDefaults(array $defaults, array $collectionPaginationConfiguration): array
360
    {
UNCOV
361
        $paginationOptions = [];
×
362

UNCOV
363
        foreach ($defaults as $key => $value) {
×
UNCOV
364
            if (!str_starts_with($key, 'pagination_')) {
×
UNCOV
365
                continue;
×
366
            }
367

UNCOV
368
            $paginationOptions[str_replace('pagination_', '', $key)] = $value;
×
369
        }
370

UNCOV
371
        return array_merge($collectionPaginationConfiguration, $paginationOptions);
×
372
    }
373

374
    private function normalizeDefaults(array $defaults): array
375
    {
UNCOV
376
        $normalizedDefaults = ['extra_properties' => $defaults['extra_properties'] ?? []];
×
UNCOV
377
        unset($defaults['extra_properties']);
×
378

UNCOV
379
        $rc = new \ReflectionClass(ApiResource::class);
×
UNCOV
380
        $publicProperties = [];
×
UNCOV
381
        foreach ($rc->getConstructor()->getParameters() as $param) {
×
UNCOV
382
            $publicProperties[$param->getName()] = true;
×
383
        }
384

UNCOV
385
        $nameConverter = new CamelCaseToSnakeCaseNameConverter();
×
UNCOV
386
        foreach ($defaults as $option => $value) {
×
UNCOV
387
            if (isset($publicProperties[$nameConverter->denormalize($option)])) {
×
UNCOV
388
                $normalizedDefaults[$option] = $value;
×
389

UNCOV
390
                continue;
×
391
            }
392

393
            $normalizedDefaults['extra_properties'][$option] = $value;
×
394
        }
395

UNCOV
396
        return $normalizedDefaults;
×
397
    }
398

399
    private function registerMetadataConfiguration(ContainerBuilder $container, array $config, PhpFileLoader $loader): void
400
    {
UNCOV
401
        [$xmlResources, $yamlResources, $phpResources] = $this->getResourcesToWatch($container, $config);
×
402

UNCOV
403
        $container->setParameter('api_platform.class_name_resources', $this->getClassNameResources());
×
404

UNCOV
405
        $loader->load('metadata/resource_name.php');
×
UNCOV
406
        $loader->load('metadata/property_name.php');
×
407

UNCOV
408
        if (!empty($config['resource_class_directories'])) {
×
409
            $container->setParameter('api_platform.resource_class_directories', array_merge(
×
410
                $config['resource_class_directories'],
×
411
                $container->getParameter('api_platform.resource_class_directories')
×
412
            ));
×
413
        }
414

415
        // V3 metadata
UNCOV
416
        $loader->load('metadata/php.php');
×
UNCOV
417
        $loader->load('metadata/xml.php');
×
UNCOV
418
        $loader->load('metadata/links.php');
×
UNCOV
419
        $loader->load('metadata/property.php');
×
UNCOV
420
        $loader->load('metadata/resource.php');
×
UNCOV
421
        $loader->load('metadata/operation.php');
×
UNCOV
422
        $loader->load('metadata/mutator.php');
×
423

UNCOV
424
        $container->getDefinition('api_platform.metadata.resource_extractor.xml')->replaceArgument(0, $xmlResources);
×
UNCOV
425
        $container->getDefinition('api_platform.metadata.property_extractor.xml')->replaceArgument(0, $xmlResources);
×
426

UNCOV
427
        if ($config['enable_phpdoc_parser'] && class_exists(PhpDocParser::class)) {
×
UNCOV
428
            $loader->load('metadata/php_doc.php');
×
429
        }
430

UNCOV
431
        if (class_exists(Yaml::class)) {
×
UNCOV
432
            $loader->load('metadata/yaml.php');
×
UNCOV
433
            $container->getDefinition('api_platform.metadata.resource_extractor.yaml')->replaceArgument(0, $yamlResources);
×
UNCOV
434
            $container->getDefinition('api_platform.metadata.property_extractor.yaml')->replaceArgument(0, $yamlResources);
×
435
        }
436

UNCOV
437
        $container->getDefinition('api_platform.metadata.resource_extractor.php_file')->replaceArgument(0, $phpResources);
×
438
    }
439

440
    private function getClassNameResources(): array
441
    {
UNCOV
442
        return [
×
UNCOV
443
            Error::class,
×
UNCOV
444
            ValidationException::class,
×
UNCOV
445
        ];
×
446
    }
447

448
    private function getBundlesResourcesPaths(ContainerBuilder $container, array $config): array
449
    {
UNCOV
450
        $bundlesResourcesPaths = [];
×
451

UNCOV
452
        foreach ($container->getParameter('kernel.bundles_metadata') as $bundle) {
×
UNCOV
453
            $dirname = $bundle['path'];
×
UNCOV
454
            $paths = [
×
UNCOV
455
                "$dirname/ApiResource",
×
UNCOV
456
                "$dirname/src/ApiResource",
×
UNCOV
457
            ];
×
UNCOV
458
            foreach (['.yaml', '.yml', '.xml', ''] as $extension) {
×
UNCOV
459
                $paths[] = "$dirname/Resources/config/api_resources$extension";
×
UNCOV
460
                $paths[] = "$dirname/config/api_resources$extension";
×
461
            }
UNCOV
462
            if ($this->isConfigEnabled($container, $config['doctrine'])) {
×
UNCOV
463
                $paths[] = "$dirname/Entity";
×
UNCOV
464
                $paths[] = "$dirname/src/Entity";
×
465
            }
UNCOV
466
            if ($this->isConfigEnabled($container, $config['doctrine_mongodb_odm'])) {
×
467
                $paths[] = "$dirname/Document";
×
468
                $paths[] = "$dirname/src/Document";
×
469
            }
470

UNCOV
471
            foreach ($paths as $path) {
×
UNCOV
472
                if ($container->fileExists($path, false)) {
×
UNCOV
473
                    $bundlesResourcesPaths[] = $path;
×
474
                }
475
            }
476
        }
477

UNCOV
478
        return $bundlesResourcesPaths;
×
479
    }
480

481
    private function getResourcesToWatch(ContainerBuilder $container, array $config): array
482
    {
UNCOV
483
        $paths = array_unique(array_merge($this->getBundlesResourcesPaths($container, $config), $config['mapping']['paths']));
×
484

UNCOV
485
        if (!$config['mapping']['paths']) {
×
486
            $projectDir = $container->getParameter('kernel.project_dir');
×
487
            foreach (["$projectDir/config/api_platform", "$projectDir/src/ApiResource"] as $dir) {
×
488
                if (is_dir($dir)) {
×
489
                    $paths[] = $dir;
×
490
                }
491
            }
492

493
            if ($this->isConfigEnabled($container, $config['doctrine']) && is_dir($doctrinePath = "$projectDir/src/Entity")) {
×
494
                $paths[] = $doctrinePath;
×
495
            }
496

497
            if ($this->isConfigEnabled($container, $config['doctrine_mongodb_odm']) && is_dir($documentPath = "$projectDir/src/Document")) {
×
498
                $paths[] = $documentPath;
×
499
            }
500
        }
501

UNCOV
502
        $resources = ['yml' => [], 'xml' => [], 'php' => [], 'dir' => []];
×
503

UNCOV
504
        foreach ($config['mapping']['imports'] ?? [] as $path) {
×
505
            if (is_dir($path)) {
×
506
                foreach (Finder::create()->followLinks()->files()->in($path)->name('/\.php$/')->sortByName() as $file) {
×
507
                    $resources[$file->getExtension()][] = $file->getRealPath();
×
508
                }
509

510
                $resources['dir'][] = $path;
×
511
                $container->addResource(new DirectoryResource($path, '/\.php$/'));
×
512

513
                continue;
×
514
            }
515

516
            if ($container->fileExists($path, false)) {
×
517
                if (!str_ends_with($path, '.php')) {
×
518
                    throw new RuntimeException(\sprintf('Unsupported mapping type in "%s", supported type is PHP.', $path));
×
519
                }
520

521
                $resources['php'][] = $path;
×
522

523
                continue;
×
524
            }
525

526
            throw new RuntimeException(\sprintf('Could not open file or directory "%s".', $path));
×
527
        }
528

UNCOV
529
        foreach ($paths as $path) {
×
UNCOV
530
            if (is_dir($path)) {
×
UNCOV
531
                foreach (Finder::create()->followLinks()->files()->in($path)->name('/\.(xml|ya?ml)$/')->sortByName() as $file) {
×
UNCOV
532
                    $resources['yaml' === ($extension = $file->getExtension()) ? 'yml' : $extension][] = $file->getRealPath();
×
533
                }
534

UNCOV
535
                $resources['dir'][] = $path;
×
UNCOV
536
                $container->addResource(new DirectoryResource($path, '/\.(xml|ya?ml|php)$/'));
×
537

UNCOV
538
                continue;
×
539
            }
540

541
            if ($container->fileExists($path, false)) {
×
542
                if (!preg_match('/\.(xml|ya?ml)$/', (string) $path, $matches)) {
×
543
                    throw new RuntimeException(\sprintf('Unsupported mapping type in "%s", supported types are XML & YAML.', $path));
×
544
                }
545

546
                $resources['yaml' === $matches[1] ? 'yml' : $matches[1]][] = $path;
×
547

548
                continue;
×
549
            }
550

551
            throw new RuntimeException(\sprintf('Could not open file or directory "%s".', $path));
×
552
        }
553

UNCOV
554
        $container->setParameter('api_platform.resource_class_directories', $resources['dir']);
×
555

UNCOV
556
        return [$resources['xml'], $resources['yml'], $resources['php']];
×
557
    }
558

559
    private function registerOAuthConfiguration(ContainerBuilder $container, array $config): void
560
    {
UNCOV
561
        if (!$config['oauth']) {
×
562
            return;
×
563
        }
564

UNCOV
565
        $container->setParameter('api_platform.oauth.enabled', $this->isConfigEnabled($container, $config['oauth']));
×
UNCOV
566
        $container->setParameter('api_platform.oauth.clientId', $config['oauth']['clientId']);
×
UNCOV
567
        $container->setParameter('api_platform.oauth.clientSecret', $config['oauth']['clientSecret']);
×
UNCOV
568
        $container->setParameter('api_platform.oauth.type', $config['oauth']['type']);
×
UNCOV
569
        $container->setParameter('api_platform.oauth.flow', $config['oauth']['flow']);
×
UNCOV
570
        $container->setParameter('api_platform.oauth.tokenUrl', $config['oauth']['tokenUrl']);
×
UNCOV
571
        $container->setParameter('api_platform.oauth.authorizationUrl', $config['oauth']['authorizationUrl']);
×
UNCOV
572
        $container->setParameter('api_platform.oauth.refreshUrl', $config['oauth']['refreshUrl']);
×
UNCOV
573
        $container->setParameter('api_platform.oauth.scopes', $config['oauth']['scopes']);
×
UNCOV
574
        $container->setParameter('api_platform.oauth.pkce', $config['oauth']['pkce']);
×
575
    }
576

577
    /**
578
     * Registers the Swagger, ReDoc and Swagger UI configuration.
579
     */
580
    private function registerSwaggerConfiguration(ContainerBuilder $container, array $config, PhpFileLoader $loader): void
581
    {
UNCOV
582
        foreach (array_keys($config['swagger']['api_keys']) as $keyName) {
×
UNCOV
583
            if (!preg_match('/^[a-zA-Z0-9._-]+$/', $keyName)) {
×
584
                throw new RuntimeException(\sprintf('The swagger api_keys key "%s" is not valid, it should match "^[a-zA-Z0-9._-]+$"', $keyName));
×
585
            }
586
        }
587

UNCOV
588
        $container->setParameter('api_platform.swagger.versions', $config['swagger']['versions']);
×
589

UNCOV
590
        if (!$config['enable_swagger'] && $config['enable_swagger_ui']) {
×
591
            throw new RuntimeException('You can not enable the Swagger UI without enabling Swagger, fix this by enabling swagger via the configuration "enable_swagger: true".');
×
592
        }
593

UNCOV
594
        if (!$config['enable_swagger']) {
×
595
            $container->setParameter('api_platform.enable_swagger_ui', false);
×
596
            $container->setParameter('api_platform.enable_re_doc', false);
×
597

598
            return;
×
599
        }
600

UNCOV
601
        $loader->load('openapi.php');
×
602

UNCOV
603
        if (class_exists(Yaml::class)) {
×
UNCOV
604
            $loader->load('openapi/yaml.php');
×
605
        }
606

UNCOV
607
        if ($config['enable_swagger_ui'] || $config['enable_re_doc']) {
×
UNCOV
608
            $loader->load('swagger_ui.php');
×
609

UNCOV
610
            if ($config['use_symfony_listeners']) {
×
UNCOV
611
                $loader->load('symfony/swagger_ui.php');
×
612
            }
613

UNCOV
614
            $loader->load('state/swagger_ui.php');
×
615
        }
616

UNCOV
617
        if (!$config['enable_swagger_ui'] && !$config['enable_re_doc']) {
×
618
            // Remove the listener but keep the controller to allow customizing the path of the UI
UNCOV
619
            $container->removeDefinition('api_platform.swagger.listener.ui');
×
620
        }
621

UNCOV
622
        $container->setParameter('api_platform.enable_swagger_ui', $config['enable_swagger_ui']);
×
UNCOV
623
        $container->setParameter('api_platform.enable_re_doc', $config['enable_re_doc']);
×
UNCOV
624
        $container->setParameter('api_platform.swagger.api_keys', $config['swagger']['api_keys']);
×
UNCOV
625
        $container->setParameter('api_platform.swagger.persist_authorization', $config['swagger']['persist_authorization']);
×
UNCOV
626
        $container->setParameter('api_platform.swagger.http_auth', $config['swagger']['http_auth']);
×
UNCOV
627
        if ($config['openapi']['swagger_ui_extra_configuration'] && $config['swagger']['swagger_ui_extra_configuration']) {
×
628
            throw new RuntimeException('You can not set "swagger_ui_extra_configuration" twice - in "openapi" and "swagger" section.');
×
629
        }
UNCOV
630
        $container->setParameter('api_platform.swagger_ui.extra_configuration', $config['openapi']['swagger_ui_extra_configuration'] ?: $config['swagger']['swagger_ui_extra_configuration']);
×
631
    }
632

633
    private function registerJsonApiConfiguration(array $formats, PhpFileLoader $loader, array $config): void
634
    {
UNCOV
635
        if (!isset($formats['jsonapi'])) {
×
636
            return;
×
637
        }
638

UNCOV
639
        if (!InstalledVersions::isInstalled('api-platform/json-api')) {
×
640
            throw new \LogicException('JSON-API support cannot be enabled as the JSON-API component is not installed. Try running "composer require api-platform/json-api".');
×
641
        }
642

UNCOV
643
        $loader->load('jsonapi.php');
×
UNCOV
644
        $loader->load('state/jsonapi.php');
×
645
    }
646

647
    private function registerJsonLdHydraConfiguration(ContainerBuilder $container, array $formats, PhpFileLoader $loader, array $config): void
648
    {
UNCOV
649
        if (!isset($formats['jsonld'])) {
×
650
            return;
×
651
        }
652

UNCOV
653
        if ($config['use_symfony_listeners']) {
×
UNCOV
654
            $loader->load('symfony/jsonld.php');
×
655
        } else {
UNCOV
656
            $loader->load('state/jsonld.php');
×
657
        }
658

UNCOV
659
        $loader->load('state/hydra.php');
×
UNCOV
660
        $loader->load('jsonld.php');
×
UNCOV
661
        $loader->load('hydra.php');
×
662

UNCOV
663
        if (!$container->has('api_platform.json_schema.schema_factory')) {
×
664
            $container->removeDefinition('api_platform.hydra.json_schema.schema_factory');
×
665
        }
666
    }
667

668
    private function registerJsonHalConfiguration(array $formats, PhpFileLoader $loader): void
669
    {
UNCOV
670
        if (!isset($formats['jsonhal'])) {
×
671
            return;
×
672
        }
673

UNCOV
674
        if (!InstalledVersions::isInstalled('api-platform/hal')) {
×
675
            throw new \LogicException('HAL support cannot be enabled as the HAL component is not installed. Try running "composer require api-platform/hal".');
×
676
        }
677

UNCOV
678
        $loader->load('hal.php');
×
679
    }
680

681
    private function registerJsonProblemConfiguration(array $errorFormats, PhpFileLoader $loader): void
682
    {
UNCOV
683
        if (!isset($errorFormats['jsonproblem'])) {
×
684
            return;
×
685
        }
686

UNCOV
687
        $loader->load('problem.php');
×
688
    }
689

690
    private function registerGraphQlConfiguration(ContainerBuilder $container, array $config, PhpFileLoader $loader): void
691
    {
UNCOV
692
        $enabled = $this->isConfigEnabled($container, $config['graphql']);
×
UNCOV
693
        $graphqlIntrospectionEnabled = $enabled && $this->isConfigEnabled($container, $config['graphql']['introspection']);
×
UNCOV
694
        $graphiqlEnabled = $enabled && $this->isConfigEnabled($container, $config['graphql']['graphiql']);
×
UNCOV
695
        $maxQueryDepth = (int) $config['graphql']['max_query_depth'];
×
UNCOV
696
        $maxQueryComplexity = (int) $config['graphql']['max_query_complexity'];
×
697

UNCOV
698
        $container->setParameter('api_platform.graphql.enabled', $enabled);
×
UNCOV
699
        $container->setParameter('api_platform.graphql.max_query_depth', $maxQueryDepth);
×
UNCOV
700
        $container->setParameter('api_platform.graphql.max_query_complexity', $maxQueryComplexity);
×
UNCOV
701
        $container->setParameter('api_platform.graphql.introspection.enabled', $graphqlIntrospectionEnabled);
×
UNCOV
702
        $container->setParameter('api_platform.graphql.graphiql.enabled', $graphiqlEnabled);
×
UNCOV
703
        $container->setParameter('api_platform.graphql.collection.pagination', $config['graphql']['collection']['pagination']);
×
704

UNCOV
705
        if (!$enabled) {
×
706
            return;
×
707
        }
708

UNCOV
709
        if (!class_exists(Executor::class)) {
×
710
            throw new \RuntimeException('Graphql is enabled but not installed, run: composer require "api-platform/graphql".');
×
711
        }
712

UNCOV
713
        $container->setParameter('api_platform.graphql.default_ide', $config['graphql']['default_ide']);
×
UNCOV
714
        $container->setParameter('api_platform.graphql.nesting_separator', $config['graphql']['nesting_separator']);
×
715

UNCOV
716
        $loader->load('graphql.php');
×
717

UNCOV
718
        if (!class_exists(Environment::class) || !isset($container->getParameter('kernel.bundles')['TwigBundle'])) {
×
719
            if ($graphiqlEnabled) {
×
720
                throw new RuntimeException(\sprintf('GraphiQL interfaces depend on Twig. Please activate TwigBundle for the %s environnement or disable GraphiQL.', $container->getParameter('kernel.environment')));
×
721
            }
722
            $container->removeDefinition('api_platform.graphql.action.graphiql');
×
723
        }
724

UNCOV
725
        $container->registerForAutoconfiguration(QueryItemResolverInterface::class)
×
UNCOV
726
            ->addTag('api_platform.graphql.resolver');
×
UNCOV
727
        $container->registerForAutoconfiguration(QueryCollectionResolverInterface::class)
×
UNCOV
728
            ->addTag('api_platform.graphql.resolver');
×
UNCOV
729
        $container->registerForAutoconfiguration(MutationResolverInterface::class)
×
UNCOV
730
            ->addTag('api_platform.graphql.resolver');
×
UNCOV
731
        $container->registerForAutoconfiguration(GraphQlTypeInterface::class)
×
UNCOV
732
            ->addTag('api_platform.graphql.type');
×
UNCOV
733
        $container->registerForAutoconfiguration(ErrorHandlerInterface::class)
×
UNCOV
734
            ->addTag('api_platform.graphql.error_handler');
×
735
    }
736

737
    private function registerCacheConfiguration(ContainerBuilder $container): void
738
    {
UNCOV
739
        if (!$container->hasParameter('kernel.debug') || !$container->getParameter('kernel.debug')) {
×
740
            $container->removeDefinition('api_platform.cache_warmer.cache_pool_clearer');
×
741
        }
742
    }
743

744
    private function registerDoctrineOrmConfiguration(ContainerBuilder $container, array $config, PhpFileLoader $loader): void
745
    {
UNCOV
746
        if (!$this->isConfigEnabled($container, $config['doctrine'])) {
×
747
            return;
×
748
        }
749

UNCOV
750
        if (!InstalledVersions::isInstalled('api-platform/doctrine-orm')) {
×
751
            throw new \LogicException('Doctrine support cannot be enabled as the doctrine ORM component is not installed. Try running "composer require api-platform/doctrine-orm".');
×
752
        }
753

754
        // For older versions of doctrine bridge this allows autoconfiguration for filters
UNCOV
755
        if (!$container->has(ManagerRegistry::class)) {
×
UNCOV
756
            $container->setAlias(ManagerRegistry::class, 'doctrine');
×
757
        }
758

UNCOV
759
        $container->registerForAutoconfiguration(QueryItemExtensionInterface::class)
×
UNCOV
760
            ->addTag('api_platform.doctrine.orm.query_extension.item');
×
UNCOV
761
        $container->registerForAutoconfiguration(DoctrineQueryCollectionExtensionInterface::class)
×
UNCOV
762
            ->addTag('api_platform.doctrine.orm.query_extension.collection');
×
UNCOV
763
        $container->registerForAutoconfiguration(DoctrineOrmAbstractFilter::class);
×
764

UNCOV
765
        $container->registerForAutoconfiguration(OrmLinksHandlerInterface::class)
×
UNCOV
766
            ->addTag('api_platform.doctrine.orm.links_handler');
×
767

UNCOV
768
        $loader->load('doctrine_orm.php');
×
769

UNCOV
770
        if ($this->isConfigEnabled($container, $config['eager_loading'])) {
×
UNCOV
771
            return;
×
772
        }
773

774
        $container->removeAlias(EagerLoadingExtension::class);
×
775
        $container->removeDefinition('api_platform.doctrine.orm.query_extension.eager_loading');
×
776
        $container->removeAlias(FilterEagerLoadingExtension::class);
×
777
        $container->removeDefinition('api_platform.doctrine.orm.query_extension.filter_eager_loading');
×
778
    }
779

780
    private function registerDoctrineMongoDbOdmConfiguration(ContainerBuilder $container, array $config, PhpFileLoader $loader): void
781
    {
UNCOV
782
        if (!$this->isConfigEnabled($container, $config['doctrine_mongodb_odm'])) {
×
UNCOV
783
            return;
×
784
        }
785

786
        if (!InstalledVersions::isInstalled('api-platform/doctrine-odm')) {
×
787
            throw new \LogicException('Doctrine MongoDB ODM support cannot be enabled as the doctrine ODM component is not installed. Try running "composer require api-platform/doctrine-odm".');
×
788
        }
789

790
        $container->registerForAutoconfiguration(AggregationItemExtensionInterface::class)
×
791
            ->addTag('api_platform.doctrine_mongodb.odm.aggregation_extension.item');
×
792
        $container->registerForAutoconfiguration(AggregationCollectionExtensionInterface::class)
×
793
            ->addTag('api_platform.doctrine_mongodb.odm.aggregation_extension.collection');
×
794
        $container->registerForAutoconfiguration(DoctrineMongoDbOdmAbstractFilter::class)
×
795
            ->setBindings(['$managerRegistry' => new Reference('doctrine_mongodb')]);
×
796
        $container->registerForAutoconfiguration(OdmLinksHandlerInterface::class)
×
797
            ->addTag('api_platform.doctrine.odm.links_handler');
×
798

799
        $loader->load('doctrine_mongodb_odm.php');
×
800
    }
801

802
    private function registerHttpCacheConfiguration(ContainerBuilder $container, array $config, PhpFileLoader $loader): void
803
    {
UNCOV
804
        $loader->load('http_cache.php');
×
805

UNCOV
806
        if (!$this->isConfigEnabled($container, $config['http_cache']['invalidation'])) {
×
807
            return;
×
808
        }
809

UNCOV
810
        if ($this->isConfigEnabled($container, $config['doctrine'])) {
×
UNCOV
811
            $loader->load('doctrine_orm_http_cache_purger.php');
×
812
        }
813

UNCOV
814
        $loader->load('state/http_cache_purger.php');
×
UNCOV
815
        $loader->load('http_cache_purger.php');
×
816

UNCOV
817
        foreach ($config['http_cache']['invalidation']['scoped_clients'] as $client) {
×
818
            $definition = $container->getDefinition($client);
×
819
            $definition->addTag('api_platform.http_cache.http_client');
×
820
        }
821

UNCOV
822
        if (!($urls = $config['http_cache']['invalidation']['urls'])) {
×
UNCOV
823
            $urls = $config['http_cache']['invalidation']['varnish_urls'];
×
824
        }
825

UNCOV
826
        foreach ($urls as $key => $url) {
×
827
            $definition = new Definition(ScopingHttpClient::class, [new Reference('http_client'), $url, ['base_uri' => $url] + $config['http_cache']['invalidation']['request_options']]);
×
828
            $definition->setFactory([ScopingHttpClient::class, 'forBaseUri']);
×
829
            $definition->addTag('api_platform.http_cache.http_client');
×
830
            $container->setDefinition('api_platform.invalidation_http_client.'.$key, $definition);
×
831
        }
832

UNCOV
833
        $serviceName = $config['http_cache']['invalidation']['purger'];
×
834

UNCOV
835
        if (!$container->hasDefinition('api_platform.http_cache.purger')) {
×
UNCOV
836
            $container->setAlias('api_platform.http_cache.purger', $serviceName);
×
837
        }
838
    }
839

840
    /**
841
     * Normalizes the format from config to the one accepted by Symfony HttpFoundation.
842
     */
843
    private function getFormats(array $configFormats): array
844
    {
UNCOV
845
        $formats = [];
×
UNCOV
846
        foreach ($configFormats as $format => $value) {
×
UNCOV
847
            foreach ($value['mime_types'] as $mimeType) {
×
UNCOV
848
                $formats[$format][] = $mimeType;
×
849
            }
850
        }
851

UNCOV
852
        return $formats;
×
853
    }
854

855
    private function registerValidatorConfiguration(ContainerBuilder $container, array $config, PhpFileLoader $loader): void
856
    {
UNCOV
857
        if (interface_exists(ValidatorInterface::class)) {
×
UNCOV
858
            $loader->load('metadata/validator.php');
×
UNCOV
859
            $loader->load('validator/validator.php');
×
860

UNCOV
861
            if ($this->isConfigEnabled($container, $config['graphql'])) {
×
UNCOV
862
                $loader->load('graphql/validator.php');
×
863
            }
864

UNCOV
865
            $loader->load($config['use_symfony_listeners'] ? 'validator/events.php' : 'validator/state.php');
×
866

UNCOV
867
            $container->registerForAutoconfiguration(ValidationGroupsGeneratorInterface::class)
×
UNCOV
868
                ->addTag('api_platform.validation_groups_generator');
×
UNCOV
869
            $container->registerForAutoconfiguration(PropertySchemaRestrictionMetadataInterface::class)
×
UNCOV
870
                ->addTag('api_platform.metadata.property_schema_restriction');
×
871
        }
872

UNCOV
873
        if (!$config['validator']) {
×
874
            return;
×
875
        }
876

UNCOV
877
        $container->setParameter('api_platform.validator.serialize_payload_fields', $config['validator']['serialize_payload_fields']);
×
878
    }
879

880
    private function registerDataCollectorConfiguration(ContainerBuilder $container, array $config, PhpFileLoader $loader): void
881
    {
UNCOV
882
        if (!$config['enable_profiler']) {
×
883
            return;
×
884
        }
885

UNCOV
886
        $loader->load('data_collector.php');
×
887

UNCOV
888
        if ($container->hasParameter('kernel.debug') && $container->getParameter('kernel.debug')) {
×
UNCOV
889
            $loader->load('debug.php');
×
890
        }
891
    }
892

893
    private function registerMercureConfiguration(ContainerBuilder $container, array $config, PhpFileLoader $loader): void
894
    {
UNCOV
895
        if (!$this->isConfigEnabled($container, $config['mercure'])) {
×
896
            return;
×
897
        }
898

UNCOV
899
        if (!InstalledVersions::isInstalled('symfony/mercure-bundle')) {
×
900
            throw new \LogicException('Mercure support cannot be enabled as the Symfony Mercure Bundle is not installed. Try running "composer require symfony/mercure-bundle".');
×
901
        }
902

UNCOV
903
        $container->setParameter('api_platform.mercure.include_type', $config['mercure']['include_type']);
×
UNCOV
904
        $loader->load('state/mercure.php');
×
905

UNCOV
906
        if ($this->isConfigEnabled($container, $config['doctrine'])) {
×
UNCOV
907
            $loader->load('doctrine_orm_mercure_publisher.php');
×
908
        }
UNCOV
909
        if ($this->isConfigEnabled($container, $config['doctrine_mongodb_odm'])) {
×
910
            $loader->load('doctrine_odm_mercure_publisher.php');
×
911
        }
912

UNCOV
913
        if ($this->isConfigEnabled($container, $config['graphql'])) {
×
UNCOV
914
            $loader->load('graphql_mercure.php');
×
915
        }
916
    }
917

918
    private function registerMessengerConfiguration(ContainerBuilder $container, array $config, PhpFileLoader $loader): void
919
    {
UNCOV
920
        if (!$this->isConfigEnabled($container, $config['messenger'])) {
×
921
            return;
×
922
        }
923

UNCOV
924
        if (!InstalledVersions::isInstalled('symfony/messenger')) {
×
925
            throw new \LogicException('Messenger support cannot be enabled as the Symfony Messenger component is not installed. Try running "composer require symfony/messenger".');
×
926
        }
927

UNCOV
928
        $loader->load('messenger.php');
×
929
    }
930

931
    private function registerElasticsearchConfiguration(ContainerBuilder $container, array $config, PhpFileLoader $loader): void
932
    {
UNCOV
933
        $enabled = $this->isConfigEnabled($container, $config['elasticsearch']);
×
934

UNCOV
935
        $container->setParameter('api_platform.elasticsearch.enabled', $enabled);
×
936

UNCOV
937
        if (!$enabled) {
×
UNCOV
938
            return;
×
939
        }
940

941
        if (!InstalledVersions::isInstalled('api-platform/elasticsearch')) {
×
942
            throw new \LogicException('Elasticsearch support cannot be enabled as the Elasticsearch component is not installed. Try running "composer require api-platform/elasticsearch".');
×
943
        }
944

945
        $clientClass = !class_exists(\Elasticsearch\Client::class)
×
946
            // ES v7
×
947
            ? \Elastic\Elasticsearch\Client::class
×
948
            // ES v8 and up
×
949
            : \Elasticsearch\Client::class;
×
950

951
        $clientDefinition = new Definition($clientClass);
×
952
        $container->setDefinition('api_platform.elasticsearch.client', $clientDefinition);
×
953
        $container->registerForAutoconfiguration(RequestBodySearchCollectionExtensionInterface::class)
×
954
            ->addTag('api_platform.elasticsearch.request_body_search_extension.collection');
×
955
        $container->setParameter('api_platform.elasticsearch.hosts', $config['elasticsearch']['hosts']);
×
956
        $container->setParameter('api_platform.elasticsearch.ssl_ca_bundle', $config['elasticsearch']['ssl_ca_bundle']);
×
957
        $container->setParameter('api_platform.elasticsearch.ssl_verification', $config['elasticsearch']['ssl_verification']);
×
958
        $loader->load('elasticsearch.php');
×
959
    }
960

961
    private function registerSecurityConfiguration(ContainerBuilder $container, array $config, PhpFileLoader $loader): void
962
    {
963
        /** @var string[] $bundles */
UNCOV
964
        $bundles = $container->getParameter('kernel.bundles');
×
965

UNCOV
966
        if (!isset($bundles['SecurityBundle'])) {
×
967
            return;
×
968
        }
969

UNCOV
970
        $loader->load('security.php');
×
971

UNCOV
972
        $loader->load('state/security.php');
×
973

UNCOV
974
        if (interface_exists(ValidatorInterface::class)) {
×
UNCOV
975
            $loader->load('state/security_validator.php');
×
976
        }
977

UNCOV
978
        if ($this->isConfigEnabled($container, $config['graphql'])) {
×
UNCOV
979
            $loader->load('graphql/security.php');
×
980
        }
981
    }
982

983
    private function registerOpenApiConfiguration(ContainerBuilder $container, array $config, PhpFileLoader $loader): void
984
    {
UNCOV
985
        $container->setParameter('api_platform.openapi.termsOfService', $config['openapi']['termsOfService']);
×
UNCOV
986
        $container->setParameter('api_platform.openapi.contact.name', $config['openapi']['contact']['name']);
×
UNCOV
987
        $container->setParameter('api_platform.openapi.contact.url', $config['openapi']['contact']['url']);
×
UNCOV
988
        $container->setParameter('api_platform.openapi.contact.email', $config['openapi']['contact']['email']);
×
UNCOV
989
        $container->setParameter('api_platform.openapi.license.name', $config['openapi']['license']['name']);
×
UNCOV
990
        $container->setParameter('api_platform.openapi.license.url', $config['openapi']['license']['url']);
×
UNCOV
991
        $container->setParameter('api_platform.openapi.license.identifier', $config['openapi']['license']['identifier']);
×
UNCOV
992
        $container->setParameter('api_platform.openapi.overrideResponses', $config['openapi']['overrideResponses']);
×
993

UNCOV
994
        $tags = [];
×
UNCOV
995
        foreach ($config['openapi']['tags'] as $tag) {
×
996
            $tags[] = new Tag($tag['name'], $tag['description'] ?? null);
×
997
        }
998

UNCOV
999
        $container->setParameter('api_platform.openapi.tags', $tags);
×
1000

UNCOV
1001
        $container->setParameter('api_platform.openapi.errorResourceClass', $config['openapi']['error_resource_class'] ?? null);
×
UNCOV
1002
        $container->setParameter('api_platform.openapi.validationErrorResourceClass', $config['openapi']['validation_error_resource_class'] ?? null);
×
1003

UNCOV
1004
        $loader->load('json_schema.php');
×
1005
    }
1006

1007
    private function registerMakerConfiguration(ContainerBuilder $container, array $config, PhpFileLoader $loader): void
1008
    {
UNCOV
1009
        if (!$this->isConfigEnabled($container, $config['maker'])) {
×
1010
            return;
×
1011
        }
1012

UNCOV
1013
        $namespaceprefix = $config['maker']['namespace_prefix'] ?? '';
×
UNCOV
1014
        if ('' !== $namespaceprefix) {
×
1015
            $namespaceprefix = trim($namespaceprefix, '\\').'\\';
×
1016
        }
UNCOV
1017
        $container->setParameter('api_platform.maker.namespace_prefix', $namespaceprefix);
×
1018

UNCOV
1019
        $loader->load('maker.php');
×
1020
    }
1021

1022
    private function registerArgumentResolverConfiguration(PhpFileLoader $loader): void
1023
    {
UNCOV
1024
        $loader->load('argument_resolver.php');
×
1025
    }
1026

1027
    private function registerLinkSecurityConfiguration(PhpFileLoader $loader, array $config): void
1028
    {
UNCOV
1029
        $loader->load('link_security.php');
×
1030
    }
1031

1032
    private function registerJsonStreamerConfiguration(ContainerBuilder $container, PhpFileLoader $loader, array $formats, array $config): void
1033
    {
UNCOV
1034
        if (!$config['enable_json_streamer']) {
×
1035
            return;
×
1036
        }
1037

UNCOV
1038
        if (!class_exists(JsonStreamWriter::class)) {
×
1039
            throw new RuntimeException('symfony/json-streamer is not installed.');
×
1040
        }
1041

1042
        // @TODO symfony/json-streamer:>=7.4.1 add composer conflict
UNCOV
1043
        if (!class_exists(ControllerHelper::class)) {
×
1044
            throw new RuntimeException('Symfony symfony/json-stream:^7.4 is needed.');
×
1045
        }
1046

UNCOV
1047
        if (isset($formats['jsonld'])) {
×
UNCOV
1048
            $container->setParameter('.json_streamer.stream_writers_dir.jsonld', '%kernel.cache_dir%/json_streamer/stream_writer/jsonld');
×
UNCOV
1049
            $container->setParameter('.json_streamer.stream_readers_dir.jsonld', '%kernel.cache_dir%/json_streamer/stream_reader/jsonld');
×
UNCOV
1050
            $container->setParameter('.json_streamer.lazy_ghosts_dir.jsonld', '%kernel.cache_dir%/json_streamer/lazy_ghost/jsonld');
×
1051
        }
1052

UNCOV
1053
        $loader->load('json_streamer/common.php');
×
1054

UNCOV
1055
        if ($config['use_symfony_listeners']) {
×
UNCOV
1056
            $loader->load('json_streamer/events.php');
×
1057
        } else {
UNCOV
1058
            if (isset($formats['jsonld'])) {
×
UNCOV
1059
                $loader->load('json_streamer/hydra.php');
×
1060
            }
1061

UNCOV
1062
            if (isset($formats['json'])) {
×
UNCOV
1063
                $loader->load('json_streamer/json.php');
×
1064
            }
1065
        }
1066
    }
1067
}
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