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

api-platform / core / 17792263445

17 Sep 2025 08:54AM UTC coverage: 21.693% (-0.2%) from 21.902%
17792263445

push

github

soyuka
feat(state): remove @internal from the CreateProvider

fixes https://github.com/api-platform/api-platform/issues/571

11753 of 54178 relevant lines covered (21.69%)

12.64 hits per line

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

74.67
/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 Doctrine\Persistence\ManagerRegistry;
51
use PHPStan\PhpDocParser\Parser\PhpDocParser;
52
use Ramsey\Uuid\Uuid;
53
use Symfony\Bundle\FrameworkBundle\Controller\ControllerHelper;
54
use Symfony\Component\Config\FileLocator;
55
use Symfony\Component\Config\Resource\DirectoryResource;
56
use Symfony\Component\DependencyInjection\ChildDefinition;
57
use Symfony\Component\DependencyInjection\ContainerBuilder;
58
use Symfony\Component\DependencyInjection\Definition;
59
use Symfony\Component\DependencyInjection\Exception\RuntimeException;
60
use Symfony\Component\DependencyInjection\Extension\Extension;
61
use Symfony\Component\DependencyInjection\Extension\PrependExtensionInterface;
62
use Symfony\Component\DependencyInjection\Loader\XmlFileLoader;
63
use Symfony\Component\DependencyInjection\Reference;
64
use Symfony\Component\Finder\Finder;
65
use Symfony\Component\HttpClient\ScopingHttpClient;
66
use Symfony\Component\JsonStreamer\JsonStreamWriter;
67
use Symfony\Component\ObjectMapper\ObjectMapper;
68
use Symfony\Component\Serializer\NameConverter\CamelCaseToSnakeCaseNameConverter;
69
use Symfony\Component\Uid\AbstractUid;
70
use Symfony\Component\Validator\Validator\ValidatorInterface;
71
use Symfony\Component\Yaml\Yaml;
72
use Twig\Environment;
73

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

107
    /**
108
     * {@inheritdoc}
109
     */
110
    public function load(array $configs, ContainerBuilder $container): void
111
    {
112
        $loader = new XmlFileLoader($container, new FileLocator(__DIR__.'/../Resources/config'));
2✔
113

114
        $configuration = new Configuration();
2✔
115
        $config = $this->processConfiguration($configuration, $configs);
2✔
116
        $container->setParameter('api_platform.use_symfony_listeners', $config['use_symfony_listeners']);
2✔
117

118
        $formats = $this->getFormats($config['formats']);
2✔
119
        $patchFormats = $this->getFormats($config['patch_formats']);
2✔
120
        $errorFormats = $this->getFormats($config['error_formats']);
2✔
121
        $docsFormats = $this->getFormats($config['docs_formats']);
2✔
122

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

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

142
        if (!isset($errorFormats['json'])) {
2✔
143
            $errorFormats['json'] = ['application/problem+json', 'application/json'];
2✔
144
        }
145

146
        if (!isset($errorFormats['jsonproblem'])) {
2✔
147
            $errorFormats['jsonproblem'] = ['application/problem+json'];
×
148
        }
149

150
        if (isset($formats['jsonapi']) && !isset($patchFormats['jsonapi'])) {
2✔
151
            $patchFormats['jsonapi'] = ['application/vnd.api+json'];
2✔
152
        }
153

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

179
        if (class_exists(ObjectMapper::class)) {
2✔
180
            $loader->load('state/object_mapper.xml');
2✔
181
        }
182
        $container->registerForAutoconfiguration(FilterInterface::class)
2✔
183
            ->addTag('api_platform.filter');
2✔
184
        $container->registerForAutoconfiguration(ProviderInterface::class)
2✔
185
            ->addTag('api_platform.state_provider');
2✔
186
        $container->registerForAutoconfiguration(ProcessorInterface::class)
2✔
187
            ->addTag('api_platform.state_processor');
2✔
188
        $container->registerForAutoconfiguration(UriVariableTransformerInterface::class)
2✔
189
            ->addTag('api_platform.uri_variables.transformer');
2✔
190
        $container->registerForAutoconfiguration(ParameterProviderInterface::class)
2✔
191
            ->addTag('api_platform.parameter_provider');
2✔
192
        $container->registerAttributeForAutoconfiguration(ApiResource::class, static function (ChildDefinition $definition): void {
2✔
193
            $definition->setAbstract(true)
×
194
                ->addTag('api_platform.resource')
×
195
                ->addTag('container.excluded', ['source' => 'by #[ApiResource] attribute']);
×
196
        });
2✔
197
        $container->registerAttributeForAutoconfiguration(
2✔
198
            AsResourceMutator::class,
2✔
199
            static function (ChildDefinition $definition, AsResourceMutator $attribute, \Reflector $reflector): void {
2✔
200
                if (!$reflector instanceof \ReflectionClass) {
×
201
                    return;
×
202
                }
203

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
                ]);
×
211
            },
2✔
212
        );
2✔
213

214
        $container->registerAttributeForAutoconfiguration(
2✔
215
            AsOperationMutator::class,
2✔
216
            static function (ChildDefinition $definition, AsOperationMutator $attribute, \Reflector $reflector): void {
2✔
217
                if (!$reflector instanceof \ReflectionClass) {
×
218
                    return;
×
219
                }
220

221
                if (!is_a($reflector->name, OperationMutatorInterface::class, true)) {
×
222
                    throw new RuntimeException(\sprintf('Operation mutator "%s" should implement %s', $reflector->name, OperationMutatorInterface::class));
×
223
                }
224

225
                $definition->addTag('api_platform.operation_mutator', [
×
226
                    'operationName' => $attribute->operationName,
×
227
                ]);
×
228
            },
2✔
229
        );
2✔
230

231
        if (!$container->has('api_platform.state.item_provider')) {
2✔
232
            $container->setAlias('api_platform.state.item_provider', 'api_platform.state_provider.object');
×
233
        }
234

235
        if ($container->getParameter('kernel.debug')) {
2✔
236
            $this->injectStopwatch($container);
2✔
237
        }
238
    }
239

240
    private function injectStopwatch(ContainerBuilder $container): void
241
    {
242
        $services = [
2✔
243
            'api_platform.state_processor.add_link_header',
2✔
244
            'api_platform.state_processor.respond',
2✔
245
            'api_platform.state_processor.serialize',
2✔
246
            'api_platform.state_processor.write',
2✔
247
            'api_platform.state_provider.content_negotiation',
2✔
248
            'api_platform.state_provider.deserialize',
2✔
249
            'api_platform.state_provider.parameter',
2✔
250
            'api_platform.state_provider.read',
2✔
251
        ];
2✔
252

253
        foreach ($services as $id) {
2✔
254
            if (!$container->hasDefinition($id)) {
2✔
255
                continue;
×
256
            }
257

258
            $definition = $container->getDefinition($id);
2✔
259
            $definition->addMethodCall('setStopwatch', [new Reference('debug.stopwatch', ContainerBuilder::IGNORE_ON_INVALID_REFERENCE)]);
2✔
260
        }
261
    }
262

263
    private function registerCommonConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader, array $formats, array $patchFormats, array $errorFormats, array $docsFormats): void
264
    {
265
        $loader->load('state/state.xml');
2✔
266
        $loader->load('symfony/symfony.xml');
2✔
267
        $loader->load('api.xml');
2✔
268
        $loader->load('filter.xml');
2✔
269

270
        if (class_exists(UuidDenormalizer::class) && class_exists(Uuid::class)) {
2✔
271
            $loader->load('ramsey_uuid.xml');
2✔
272
        }
273

274
        if (class_exists(AbstractUid::class)) {
2✔
275
            $loader->load('symfony/uid.xml');
2✔
276
        }
277

278
        $defaultContext = ['hydra_prefix' => $config['serializer']['hydra_prefix']] + ($container->hasParameter('serializer.default_context') ? $container->getParameter('serializer.default_context') : []);
2✔
279

280
        $container->setParameter('api_platform.serializer.default_context', $defaultContext);
2✔
281
        if (!$container->hasParameter('serializer.default_context')) {
2✔
282
            $container->setParameter('serializer.default_context', $container->getParameter('api_platform.serializer.default_context'));
2✔
283
        }
284
        if ($config['use_symfony_listeners']) {
2✔
285
            $loader->load('symfony/events.xml');
×
286
        } else {
287
            $loader->load('symfony/controller.xml');
2✔
288
            $loader->load('state/provider.xml');
2✔
289
            $loader->load('state/processor.xml');
2✔
290
        }
291
        $loader->load('state/parameter_provider.xml');
2✔
292

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

335
        $container->setAlias('api_platform.path_segment_name_generator', $config['path_segment_name_generator']);
2✔
336
        $container->setAlias('api_platform.inflector', $config['inflector']);
2✔
337

338
        if ($config['name_converter']) {
2✔
339
            $container->setAlias('api_platform.name_converter', $config['name_converter']);
2✔
340
        }
341
        $container->setParameter('api_platform.asset_package', $config['asset_package']);
2✔
342
        $container->setParameter('api_platform.defaults', $this->normalizeDefaults($config['defaults'] ?? []));
2✔
343

344
        if ($container->getParameter('kernel.debug')) {
2✔
345
            $container->removeDefinition('api_platform.serializer.mapping.cache_class_metadata_factory');
2✔
346
        }
347
    }
348

349
    /**
350
     * This method will be removed in 3.0 when "defaults" will be the regular configuration path for the pagination.
351
     */
352
    private function getPaginationDefaults(array $defaults, array $collectionPaginationConfiguration): array
353
    {
354
        $paginationOptions = [];
2✔
355

356
        foreach ($defaults as $key => $value) {
2✔
357
            if (!str_starts_with($key, 'pagination_')) {
2✔
358
                continue;
2✔
359
            }
360

361
            $paginationOptions[str_replace('pagination_', '', $key)] = $value;
2✔
362
        }
363

364
        return array_merge($collectionPaginationConfiguration, $paginationOptions);
2✔
365
    }
366

367
    private function normalizeDefaults(array $defaults): array
368
    {
369
        $normalizedDefaults = ['extra_properties' => $defaults['extra_properties'] ?? []];
2✔
370
        unset($defaults['extra_properties']);
2✔
371

372
        $rc = new \ReflectionClass(ApiResource::class);
2✔
373
        $publicProperties = [];
2✔
374
        foreach ($rc->getConstructor()->getParameters() as $param) {
2✔
375
            $publicProperties[$param->getName()] = true;
2✔
376
        }
377

378
        $nameConverter = new CamelCaseToSnakeCaseNameConverter();
2✔
379
        foreach ($defaults as $option => $value) {
2✔
380
            if (isset($publicProperties[$nameConverter->denormalize($option)])) {
2✔
381
                $normalizedDefaults[$option] = $value;
2✔
382

383
                continue;
2✔
384
            }
385

386
            $normalizedDefaults['extra_properties'][$option] = $value;
×
387
        }
388

389
        return $normalizedDefaults;
2✔
390
    }
391

392
    private function registerMetadataConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
393
    {
394
        [$xmlResources, $yamlResources, $phpResources] = $this->getResourcesToWatch($container, $config);
2✔
395

396
        $container->setParameter('api_platform.class_name_resources', $this->getClassNameResources());
2✔
397

398
        $loader->load('metadata/resource_name.xml');
2✔
399
        $loader->load('metadata/property_name.xml');
2✔
400

401
        if (!empty($config['resource_class_directories'])) {
2✔
402
            $container->setParameter('api_platform.resource_class_directories', array_merge(
×
403
                $config['resource_class_directories'],
×
404
                $container->getParameter('api_platform.resource_class_directories')
×
405
            ));
×
406
        }
407

408
        // V3 metadata
409
        $loader->load('metadata/php.xml');
2✔
410
        $loader->load('metadata/xml.xml');
2✔
411
        $loader->load('metadata/links.xml');
2✔
412
        $loader->load('metadata/property.xml');
2✔
413
        $loader->load('metadata/resource.xml');
2✔
414
        $loader->load('metadata/operation.xml');
2✔
415
        $loader->load('metadata/mutator.xml');
2✔
416

417
        $container->getDefinition('api_platform.metadata.resource_extractor.xml')->replaceArgument(0, $xmlResources);
2✔
418
        $container->getDefinition('api_platform.metadata.property_extractor.xml')->replaceArgument(0, $xmlResources);
2✔
419

420
        if (class_exists(PhpDocParser::class)) {
2✔
421
            $loader->load('metadata/php_doc.xml');
2✔
422
        }
423

424
        if (class_exists(Yaml::class)) {
2✔
425
            $loader->load('metadata/yaml.xml');
2✔
426
            $container->getDefinition('api_platform.metadata.resource_extractor.yaml')->replaceArgument(0, $yamlResources);
2✔
427
            $container->getDefinition('api_platform.metadata.property_extractor.yaml')->replaceArgument(0, $yamlResources);
2✔
428
        }
429

430
        $container->getDefinition('api_platform.metadata.resource_extractor.php_file')->replaceArgument(0, $phpResources);
2✔
431
    }
432

433
    private function getClassNameResources(): array
434
    {
435
        return [
2✔
436
            Error::class,
2✔
437
            ValidationException::class,
2✔
438
        ];
2✔
439
    }
440

441
    private function getBundlesResourcesPaths(ContainerBuilder $container, array $config): array
442
    {
443
        $bundlesResourcesPaths = [];
2✔
444

445
        foreach ($container->getParameter('kernel.bundles_metadata') as $bundle) {
2✔
446
            $dirname = $bundle['path'];
2✔
447
            $paths = [
2✔
448
                "$dirname/ApiResource",
2✔
449
                "$dirname/src/ApiResource",
2✔
450
            ];
2✔
451
            foreach (['.yaml', '.yml', '.xml', ''] as $extension) {
2✔
452
                $paths[] = "$dirname/Resources/config/api_resources$extension";
2✔
453
                $paths[] = "$dirname/config/api_resources$extension";
2✔
454
            }
455
            if ($this->isConfigEnabled($container, $config['doctrine'])) {
2✔
456
                $paths[] = "$dirname/Entity";
2✔
457
                $paths[] = "$dirname/src/Entity";
2✔
458
            }
459
            if ($this->isConfigEnabled($container, $config['doctrine_mongodb_odm'])) {
2✔
460
                $paths[] = "$dirname/Document";
×
461
                $paths[] = "$dirname/src/Document";
×
462
            }
463

464
            foreach ($paths as $path) {
2✔
465
                if ($container->fileExists($path, false)) {
2✔
466
                    $bundlesResourcesPaths[] = $path;
2✔
467
                }
468
            }
469
        }
470

471
        return $bundlesResourcesPaths;
2✔
472
    }
473

474
    private function getResourcesToWatch(ContainerBuilder $container, array $config): array
475
    {
476
        $paths = array_unique(array_merge($this->getBundlesResourcesPaths($container, $config), $config['mapping']['paths']));
2✔
477

478
        if (!$config['mapping']['paths']) {
2✔
479
            $projectDir = $container->getParameter('kernel.project_dir');
×
480
            foreach (["$projectDir/config/api_platform", "$projectDir/src/ApiResource"] as $dir) {
×
481
                if (is_dir($dir)) {
×
482
                    $paths[] = $dir;
×
483
                }
484
            }
485

486
            if ($this->isConfigEnabled($container, $config['doctrine']) && is_dir($doctrinePath = "$projectDir/src/Entity")) {
×
487
                $paths[] = $doctrinePath;
×
488
            }
489

490
            if ($this->isConfigEnabled($container, $config['doctrine_mongodb_odm']) && is_dir($documentPath = "$projectDir/src/Document")) {
×
491
                $paths[] = $documentPath;
×
492
            }
493
        }
494

495
        $resources = ['yml' => [], 'xml' => [], 'php' => [], 'dir' => []];
2✔
496

497
        foreach ($config['mapping']['imports'] ?? [] as $path) {
2✔
498
            if (is_dir($path)) {
×
499
                foreach (Finder::create()->followLinks()->files()->in($path)->name('/\.php$/')->sortByName() as $file) {
×
500
                    $resources[$file->getExtension()][] = $file->getRealPath();
×
501
                }
502

503
                $resources['dir'][] = $path;
×
504
                $container->addResource(new DirectoryResource($path, '/\.php$/'));
×
505

506
                continue;
×
507
            }
508

509
            if ($container->fileExists($path, false)) {
×
510
                if (!str_ends_with($path, '.php')) {
×
511
                    throw new RuntimeException(\sprintf('Unsupported mapping type in "%s", supported type is PHP.', $path));
×
512
                }
513

514
                $resources['php'][] = $path;
×
515

516
                continue;
×
517
            }
518

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

522
        foreach ($paths as $path) {
2✔
523
            if (is_dir($path)) {
2✔
524
                foreach (Finder::create()->followLinks()->files()->in($path)->name('/\.(xml|ya?ml)$/')->sortByName() as $file) {
2✔
525
                    $resources['yaml' === ($extension = $file->getExtension()) ? 'yml' : $extension][] = $file->getRealPath();
2✔
526
                }
527

528
                $resources['dir'][] = $path;
2✔
529
                $container->addResource(new DirectoryResource($path, '/\.(xml|ya?ml|php)$/'));
2✔
530

531
                continue;
2✔
532
            }
533

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

539
                $resources['yaml' === $matches[1] ? 'yml' : $matches[1]][] = $path;
×
540

541
                continue;
×
542
            }
543

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

547
        $container->setParameter('api_platform.resource_class_directories', $resources['dir']);
2✔
548

549
        return [$resources['xml'], $resources['yml'], $resources['php']];
2✔
550
    }
551

552
    private function registerOAuthConfiguration(ContainerBuilder $container, array $config): void
553
    {
554
        if (!$config['oauth']) {
2✔
555
            return;
×
556
        }
557

558
        $container->setParameter('api_platform.oauth.enabled', $this->isConfigEnabled($container, $config['oauth']));
2✔
559
        $container->setParameter('api_platform.oauth.clientId', $config['oauth']['clientId']);
2✔
560
        $container->setParameter('api_platform.oauth.clientSecret', $config['oauth']['clientSecret']);
2✔
561
        $container->setParameter('api_platform.oauth.type', $config['oauth']['type']);
2✔
562
        $container->setParameter('api_platform.oauth.flow', $config['oauth']['flow']);
2✔
563
        $container->setParameter('api_platform.oauth.tokenUrl', $config['oauth']['tokenUrl']);
2✔
564
        $container->setParameter('api_platform.oauth.authorizationUrl', $config['oauth']['authorizationUrl']);
2✔
565
        $container->setParameter('api_platform.oauth.refreshUrl', $config['oauth']['refreshUrl']);
2✔
566
        $container->setParameter('api_platform.oauth.scopes', $config['oauth']['scopes']);
2✔
567
        $container->setParameter('api_platform.oauth.pkce', $config['oauth']['pkce']);
2✔
568
    }
569

570
    /**
571
     * Registers the Swagger, ReDoc and Swagger UI configuration.
572
     */
573
    private function registerSwaggerConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
574
    {
575
        foreach (array_keys($config['swagger']['api_keys']) as $keyName) {
2✔
576
            if (!preg_match('/^[a-zA-Z0-9._-]+$/', $keyName)) {
2✔
577
                throw new RuntimeException(\sprintf('The swagger api_keys key "%s" is not valid, it should match "^[a-zA-Z0-9._-]+$"', $keyName));
×
578
            }
579
        }
580

581
        $container->setParameter('api_platform.swagger.versions', $config['swagger']['versions']);
2✔
582

583
        if (!$config['enable_swagger'] && $config['enable_swagger_ui']) {
2✔
584
            throw new RuntimeException('You can not enable the Swagger UI without enabling Swagger, fix this by enabling swagger via the configuration "enable_swagger: true".');
×
585
        }
586

587
        if (!$config['enable_swagger']) {
2✔
588
            return;
×
589
        }
590

591
        $loader->load('openapi.xml');
2✔
592

593
        if (class_exists(Yaml::class)) {
2✔
594
            $loader->load('openapi/yaml.xml');
2✔
595
        }
596

597
        $loader->load('swagger_ui.xml');
2✔
598

599
        if ($config['use_symfony_listeners']) {
2✔
600
            $loader->load('symfony/swagger_ui.xml');
×
601
        }
602

603
        if ($config['enable_swagger_ui']) {
2✔
604
            $loader->load('state/swagger_ui.xml');
2✔
605
        }
606

607
        if (!$config['enable_swagger_ui'] && !$config['enable_re_doc']) {
2✔
608
            // Remove the listener but keep the controller to allow customizing the path of the UI
609
            $container->removeDefinition('api_platform.swagger.listener.ui');
×
610
        }
611

612
        $container->setParameter('api_platform.enable_swagger_ui', $config['enable_swagger_ui']);
2✔
613
        $container->setParameter('api_platform.enable_re_doc', $config['enable_re_doc']);
2✔
614
        $container->setParameter('api_platform.swagger.api_keys', $config['swagger']['api_keys']);
2✔
615
        $container->setParameter('api_platform.swagger.persist_authorization', $config['swagger']['persist_authorization']);
2✔
616
        $container->setParameter('api_platform.swagger.http_auth', $config['swagger']['http_auth']);
2✔
617
        if ($config['openapi']['swagger_ui_extra_configuration'] && $config['swagger']['swagger_ui_extra_configuration']) {
2✔
618
            throw new RuntimeException('You can not set "swagger_ui_extra_configuration" twice - in "openapi" and "swagger" section.');
×
619
        }
620
        $container->setParameter('api_platform.swagger_ui.extra_configuration', $config['openapi']['swagger_ui_extra_configuration'] ?: $config['swagger']['swagger_ui_extra_configuration']);
2✔
621
    }
622

623
    private function registerJsonApiConfiguration(array $formats, XmlFileLoader $loader, array $config): void
624
    {
625
        if (!isset($formats['jsonapi'])) {
2✔
626
            return;
×
627
        }
628

629
        $loader->load('jsonapi.xml');
2✔
630
        $loader->load('state/jsonapi.xml');
2✔
631
    }
632

633
    private function registerJsonLdHydraConfiguration(ContainerBuilder $container, array $formats, XmlFileLoader $loader, array $config): void
634
    {
635
        if (!isset($formats['jsonld'])) {
2✔
636
            return;
×
637
        }
638

639
        if ($config['use_symfony_listeners']) {
2✔
640
            $loader->load('symfony/jsonld.xml');
×
641
        } else {
642
            $loader->load('state/jsonld.xml');
2✔
643
        }
644

645
        $loader->load('state/hydra.xml');
2✔
646
        $loader->load('jsonld.xml');
2✔
647
        $loader->load('hydra.xml');
2✔
648

649
        if (!$container->has('api_platform.json_schema.schema_factory')) {
2✔
650
            $container->removeDefinition('api_platform.hydra.json_schema.schema_factory');
×
651
        }
652
    }
653

654
    private function registerJsonHalConfiguration(array $formats, XmlFileLoader $loader): void
655
    {
656
        if (!isset($formats['jsonhal'])) {
2✔
657
            return;
×
658
        }
659

660
        $loader->load('hal.xml');
2✔
661
    }
662

663
    private function registerJsonProblemConfiguration(array $errorFormats, XmlFileLoader $loader): void
664
    {
665
        if (!isset($errorFormats['jsonproblem'])) {
2✔
666
            return;
×
667
        }
668

669
        $loader->load('problem.xml');
2✔
670
    }
671

672
    private function registerGraphQlConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
673
    {
674
        $enabled = $this->isConfigEnabled($container, $config['graphql']);
2✔
675
        $graphqlIntrospectionEnabled = $enabled && $this->isConfigEnabled($container, $config['graphql']['introspection']);
2✔
676
        $graphiqlEnabled = $enabled && $this->isConfigEnabled($container, $config['graphql']['graphiql']);
2✔
677
        $maxQueryDepth = (int) $config['graphql']['max_query_depth'];
2✔
678
        $maxQueryComplexity = (int) $config['graphql']['max_query_complexity'];
2✔
679

680
        $container->setParameter('api_platform.graphql.enabled', $enabled);
2✔
681
        $container->setParameter('api_platform.graphql.max_query_depth', $maxQueryDepth);
2✔
682
        $container->setParameter('api_platform.graphql.max_query_complexity', $maxQueryComplexity);
2✔
683
        $container->setParameter('api_platform.graphql.introspection.enabled', $graphqlIntrospectionEnabled);
2✔
684
        $container->setParameter('api_platform.graphql.graphiql.enabled', $graphiqlEnabled);
2✔
685
        $container->setParameter('api_platform.graphql.collection.pagination', $config['graphql']['collection']['pagination']);
2✔
686

687
        if (!$enabled) {
2✔
688
            return;
×
689
        }
690

691
        if (!class_exists(Executor::class)) {
2✔
692
            throw new \RuntimeException('Graphql is enabled but not installed, run: composer require "api-platform/graphql".');
×
693
        }
694

695
        $container->setParameter('api_platform.graphql.default_ide', $config['graphql']['default_ide']);
2✔
696
        $container->setParameter('api_platform.graphql.nesting_separator', $config['graphql']['nesting_separator']);
2✔
697

698
        $loader->load('graphql.xml');
2✔
699

700
        if (!class_exists(Environment::class) || !isset($container->getParameter('kernel.bundles')['TwigBundle'])) {
2✔
701
            if ($graphiqlEnabled) {
×
702
                throw new RuntimeException(\sprintf('GraphiQL interfaces depend on Twig. Please activate TwigBundle for the %s environnement or disable GraphiQL.', $container->getParameter('kernel.environment')));
×
703
            }
704
            $container->removeDefinition('api_platform.graphql.action.graphiql');
×
705
        }
706

707
        $container->registerForAutoconfiguration(QueryItemResolverInterface::class)
2✔
708
            ->addTag('api_platform.graphql.resolver');
2✔
709
        $container->registerForAutoconfiguration(QueryCollectionResolverInterface::class)
2✔
710
            ->addTag('api_platform.graphql.resolver');
2✔
711
        $container->registerForAutoconfiguration(MutationResolverInterface::class)
2✔
712
            ->addTag('api_platform.graphql.resolver');
2✔
713
        $container->registerForAutoconfiguration(GraphQlTypeInterface::class)
2✔
714
            ->addTag('api_platform.graphql.type');
2✔
715
        $container->registerForAutoconfiguration(ErrorHandlerInterface::class)
2✔
716
            ->addTag('api_platform.graphql.error_handler');
2✔
717
    }
718

719
    private function registerCacheConfiguration(ContainerBuilder $container): void
720
    {
721
        if (!$container->hasParameter('kernel.debug') || !$container->getParameter('kernel.debug')) {
2✔
722
            $container->removeDefinition('api_platform.cache_warmer.cache_pool_clearer');
×
723
        }
724
    }
725

726
    private function registerDoctrineOrmConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
727
    {
728
        if (!$this->isConfigEnabled($container, $config['doctrine'])) {
2✔
729
            return;
×
730
        }
731

732
        // For older versions of doctrine bridge this allows autoconfiguration for filters
733
        if (!$container->has(ManagerRegistry::class)) {
2✔
734
            $container->setAlias(ManagerRegistry::class, 'doctrine');
2✔
735
        }
736

737
        $container->registerForAutoconfiguration(QueryItemExtensionInterface::class)
2✔
738
            ->addTag('api_platform.doctrine.orm.query_extension.item');
2✔
739
        $container->registerForAutoconfiguration(DoctrineQueryCollectionExtensionInterface::class)
2✔
740
            ->addTag('api_platform.doctrine.orm.query_extension.collection');
2✔
741
        $container->registerForAutoconfiguration(DoctrineOrmAbstractFilter::class);
2✔
742

743
        $container->registerForAutoconfiguration(OrmLinksHandlerInterface::class)
2✔
744
            ->addTag('api_platform.doctrine.orm.links_handler');
2✔
745

746
        $loader->load('doctrine_orm.xml');
2✔
747

748
        if ($this->isConfigEnabled($container, $config['eager_loading'])) {
2✔
749
            return;
2✔
750
        }
751

752
        $container->removeAlias(EagerLoadingExtension::class);
×
753
        $container->removeDefinition('api_platform.doctrine.orm.query_extension.eager_loading');
×
754
        $container->removeAlias(FilterEagerLoadingExtension::class);
×
755
        $container->removeDefinition('api_platform.doctrine.orm.query_extension.filter_eager_loading');
×
756
    }
757

758
    private function registerDoctrineMongoDbOdmConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
759
    {
760
        if (!$this->isConfigEnabled($container, $config['doctrine_mongodb_odm'])) {
2✔
761
            return;
2✔
762
        }
763

764
        $container->registerForAutoconfiguration(AggregationItemExtensionInterface::class)
×
765
            ->addTag('api_platform.doctrine_mongodb.odm.aggregation_extension.item');
×
766
        $container->registerForAutoconfiguration(AggregationCollectionExtensionInterface::class)
×
767
            ->addTag('api_platform.doctrine_mongodb.odm.aggregation_extension.collection');
×
768
        $container->registerForAutoconfiguration(DoctrineMongoDbOdmAbstractFilter::class)
×
769
            ->setBindings(['$managerRegistry' => new Reference('doctrine_mongodb')]);
×
770
        $container->registerForAutoconfiguration(OdmLinksHandlerInterface::class)
×
771
            ->addTag('api_platform.doctrine.odm.links_handler');
×
772

773
        $loader->load('doctrine_mongodb_odm.xml');
×
774
    }
775

776
    private function registerHttpCacheConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
777
    {
778
        $loader->load('http_cache.xml');
2✔
779

780
        if (!$this->isConfigEnabled($container, $config['http_cache']['invalidation'])) {
2✔
781
            return;
×
782
        }
783

784
        if ($this->isConfigEnabled($container, $config['doctrine'])) {
2✔
785
            $loader->load('doctrine_orm_http_cache_purger.xml');
2✔
786
        }
787

788
        $loader->load('state/http_cache_purger.xml');
2✔
789
        $loader->load('http_cache_purger.xml');
2✔
790

791
        foreach ($config['http_cache']['invalidation']['scoped_clients'] as $client) {
2✔
792
            $definition = $container->getDefinition($client);
×
793
            $definition->addTag('api_platform.http_cache.http_client');
×
794
        }
795

796
        if (!($urls = $config['http_cache']['invalidation']['urls'])) {
2✔
797
            $urls = $config['http_cache']['invalidation']['varnish_urls'];
2✔
798
        }
799

800
        foreach ($urls as $key => $url) {
2✔
801
            $definition = new Definition(ScopingHttpClient::class, [new Reference('http_client'), $url, ['base_uri' => $url] + $config['http_cache']['invalidation']['request_options']]);
×
802
            $definition->setFactory([ScopingHttpClient::class, 'forBaseUri']);
×
803
            $definition->addTag('api_platform.http_cache.http_client');
×
804
            $container->setDefinition('api_platform.invalidation_http_client.'.$key, $definition);
×
805
        }
806

807
        $serviceName = $config['http_cache']['invalidation']['purger'];
2✔
808

809
        if (!$container->hasDefinition('api_platform.http_cache.purger')) {
2✔
810
            $container->setAlias('api_platform.http_cache.purger', $serviceName);
2✔
811
        }
812
    }
813

814
    /**
815
     * Normalizes the format from config to the one accepted by Symfony HttpFoundation.
816
     */
817
    private function getFormats(array $configFormats): array
818
    {
819
        $formats = [];
2✔
820
        foreach ($configFormats as $format => $value) {
2✔
821
            foreach ($value['mime_types'] as $mimeType) {
2✔
822
                $formats[$format][] = $mimeType;
2✔
823
            }
824
        }
825

826
        return $formats;
2✔
827
    }
828

829
    private function registerValidatorConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
830
    {
831
        if (interface_exists(ValidatorInterface::class)) {
2✔
832
            $loader->load('metadata/validator.xml');
2✔
833
            $loader->load('validator/validator.xml');
2✔
834

835
            if ($this->isConfigEnabled($container, $config['graphql'])) {
2✔
836
                $loader->load('graphql/validator.xml');
2✔
837
            }
838

839
            $loader->load($config['use_symfony_listeners'] ? 'validator/events.xml' : 'validator/state.xml');
2✔
840

841
            $container->registerForAutoconfiguration(ValidationGroupsGeneratorInterface::class)
2✔
842
                ->addTag('api_platform.validation_groups_generator');
2✔
843
            $container->registerForAutoconfiguration(PropertySchemaRestrictionMetadataInterface::class)
2✔
844
                ->addTag('api_platform.metadata.property_schema_restriction');
2✔
845
        }
846

847
        if (!$config['validator']) {
2✔
848
            return;
×
849
        }
850

851
        $container->setParameter('api_platform.validator.serialize_payload_fields', $config['validator']['serialize_payload_fields']);
2✔
852
    }
853

854
    private function registerDataCollectorConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
855
    {
856
        if (!$config['enable_profiler']) {
2✔
857
            return;
×
858
        }
859

860
        $loader->load('data_collector.xml');
2✔
861

862
        if ($container->hasParameter('kernel.debug') && $container->getParameter('kernel.debug')) {
2✔
863
            $loader->load('debug.xml');
2✔
864
        }
865
    }
866

867
    private function registerMercureConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
868
    {
869
        if (!$this->isConfigEnabled($container, $config['mercure'])) {
2✔
870
            return;
×
871
        }
872

873
        $container->setParameter('api_platform.mercure.include_type', $config['mercure']['include_type']);
2✔
874
        $loader->load('state/mercure.xml');
2✔
875

876
        if ($this->isConfigEnabled($container, $config['doctrine'])) {
2✔
877
            $loader->load('doctrine_orm_mercure_publisher.xml');
2✔
878
        }
879
        if ($this->isConfigEnabled($container, $config['doctrine_mongodb_odm'])) {
2✔
880
            $loader->load('doctrine_odm_mercure_publisher.xml');
×
881
        }
882

883
        if ($this->isConfigEnabled($container, $config['graphql'])) {
2✔
884
            $loader->load('graphql_mercure.xml');
2✔
885
        }
886
    }
887

888
    private function registerMessengerConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
889
    {
890
        if (!$this->isConfigEnabled($container, $config['messenger'])) {
2✔
891
            return;
×
892
        }
893

894
        $loader->load('messenger.xml');
2✔
895
    }
896

897
    private function registerElasticsearchConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
898
    {
899
        $enabled = $this->isConfigEnabled($container, $config['elasticsearch']);
2✔
900

901
        $container->setParameter('api_platform.elasticsearch.enabled', $enabled);
2✔
902

903
        if (!$enabled) {
2✔
904
            return;
2✔
905
        }
906

907
        $clientClass = !class_exists(\Elasticsearch\Client::class)
×
908
            // ES v7
×
909
            ? \Elastic\Elasticsearch\Client::class
×
910
            // ES v8 and up
×
911
            : \Elasticsearch\Client::class;
×
912

913
        $clientDefinition = new Definition($clientClass);
×
914
        $container->setDefinition('api_platform.elasticsearch.client', $clientDefinition);
×
915
        $container->registerForAutoconfiguration(RequestBodySearchCollectionExtensionInterface::class)
×
916
            ->addTag('api_platform.elasticsearch.request_body_search_extension.collection');
×
917
        $container->setParameter('api_platform.elasticsearch.hosts', $config['elasticsearch']['hosts']);
×
918
        $loader->load('elasticsearch.xml');
×
919
    }
920

921
    private function registerSecurityConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
922
    {
923
        /** @var string[] $bundles */
924
        $bundles = $container->getParameter('kernel.bundles');
2✔
925

926
        if (!isset($bundles['SecurityBundle'])) {
2✔
927
            return;
×
928
        }
929

930
        $loader->load('security.xml');
2✔
931

932
        $loader->load('state/security.xml');
2✔
933

934
        if (interface_exists(ValidatorInterface::class)) {
2✔
935
            $loader->load('state/security_validator.xml');
2✔
936
        }
937

938
        if ($this->isConfigEnabled($container, $config['graphql'])) {
2✔
939
            $loader->load('graphql/security.xml');
2✔
940
        }
941
    }
942

943
    private function registerOpenApiConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
944
    {
945
        $container->setParameter('api_platform.openapi.termsOfService', $config['openapi']['termsOfService']);
2✔
946
        $container->setParameter('api_platform.openapi.contact.name', $config['openapi']['contact']['name']);
2✔
947
        $container->setParameter('api_platform.openapi.contact.url', $config['openapi']['contact']['url']);
2✔
948
        $container->setParameter('api_platform.openapi.contact.email', $config['openapi']['contact']['email']);
2✔
949
        $container->setParameter('api_platform.openapi.license.name', $config['openapi']['license']['name']);
2✔
950
        $container->setParameter('api_platform.openapi.license.url', $config['openapi']['license']['url']);
2✔
951
        $container->setParameter('api_platform.openapi.license.identifier', $config['openapi']['license']['identifier']);
2✔
952
        $container->setParameter('api_platform.openapi.overrideResponses', $config['openapi']['overrideResponses']);
2✔
953

954
        $tags = [];
2✔
955
        foreach ($config['openapi']['tags'] as $tag) {
2✔
956
            $tags[] = new Tag($tag['name'], $tag['description'] ?? null);
×
957
        }
958

959
        $container->setParameter('api_platform.openapi.tags', $tags);
2✔
960

961
        $container->setParameter('api_platform.openapi.errorResourceClass', $config['openapi']['error_resource_class'] ?? null);
2✔
962
        $container->setParameter('api_platform.openapi.validationErrorResourceClass', $config['openapi']['validation_error_resource_class'] ?? null);
2✔
963

964
        $loader->load('json_schema.xml');
2✔
965
    }
966

967
    private function registerMakerConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
968
    {
969
        if (!$this->isConfigEnabled($container, $config['maker'])) {
2✔
970
            return;
×
971
        }
972

973
        $loader->load('maker.xml');
2✔
974
    }
975

976
    private function registerArgumentResolverConfiguration(XmlFileLoader $loader): void
977
    {
978
        $loader->load('argument_resolver.xml');
2✔
979
    }
980

981
    private function registerLinkSecurityConfiguration(XmlFileLoader $loader, array $config): void
982
    {
983
        if ($config['enable_link_security']) {
2✔
984
            $loader->load('link_security.xml');
2✔
985
        }
986
    }
987

988
    private function registerJsonStreamerConfiguration(ContainerBuilder $container, XmlFileLoader $loader, array $formats, array $config): void
989
    {
990
        if (!$config['enable_json_streamer']) {
2✔
991
            return;
2✔
992
        }
993

994
        if (!class_exists(JsonStreamWriter::class)) {
×
995
            throw new RuntimeException('symfony/json-streamer is not installed.');
×
996
        }
997

998
        // @TODO symfony/json-streamer:>=7.4.1 add composer conflict
999
        if (!class_exists(ControllerHelper::class)) {
×
1000
            throw new RuntimeException('Symfony symfony/json-stream:^7.4 is needed.');
×
1001
        }
1002

1003
        if (isset($formats['jsonld'])) {
×
1004
            $container->setParameter('.json_streamer.stream_writers_dir.jsonld', '%kernel.cache_dir%/json_streamer/stream_writer/jsonld');
×
1005
            $container->setParameter('.json_streamer.stream_readers_dir.jsonld', '%kernel.cache_dir%/json_streamer/stream_reader/jsonld');
×
1006
            $container->setParameter('.json_streamer.lazy_ghosts_dir.jsonld', '%kernel.cache_dir%/json_streamer/lazy_ghost/jsonld');
×
1007
        }
1008

1009
        $loader->load('json_streamer/common.xml');
×
1010

1011
        if ($config['use_symfony_listeners']) {
×
1012
            $loader->load('json_streamer/events.xml');
×
1013
        } else {
1014
            if (isset($formats['jsonld'])) {
×
1015
                $loader->load('json_streamer/hydra.xml');
×
1016
            }
1017

1018
            if (isset($formats['json'])) {
×
1019
                $loader->load('json_streamer/json.xml');
×
1020
            }
1021
        }
1022
    }
1023
}
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