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

api-platform / core / 15927963363

27 Jun 2025 01:52PM UTC coverage: 22.489% (-0.01%) from 22.5%
15927963363

Pull #7247

github

web-flow
Merge 66de6d385 into d3e73f09a
Pull Request #7247: fix(laravel): decorate error handler

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

8 existing lines in 4 files now uncovered.

11063 of 49194 relevant lines covered (22.49%)

11.03 hits per line

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

79.21
/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\FilterInterface;
35
use ApiPlatform\Metadata\UriVariableTransformerInterface;
36
use ApiPlatform\Metadata\UrlGeneratorInterface;
37
use ApiPlatform\OpenApi\Model\Tag;
38
use ApiPlatform\RamseyUuid\Serializer\UuidDenormalizer;
39
use ApiPlatform\State\ApiResource\Error;
40
use ApiPlatform\State\ParameterProviderInterface;
41
use ApiPlatform\State\ProcessorInterface;
42
use ApiPlatform\State\ProviderInterface;
43
use ApiPlatform\Symfony\Validator\Metadata\Property\Restriction\PropertySchemaRestrictionMetadataInterface;
44
use ApiPlatform\Symfony\Validator\ValidationGroupsGeneratorInterface;
45
use ApiPlatform\Validator\Exception\ValidationException;
46
use Doctrine\Persistence\ManagerRegistry;
47
use PHPStan\PhpDocParser\Parser\PhpDocParser;
48
use Ramsey\Uuid\Uuid;
49
use Symfony\Component\Config\FileLocator;
50
use Symfony\Component\Config\Resource\DirectoryResource;
51
use Symfony\Component\DependencyInjection\ContainerBuilder;
52
use Symfony\Component\DependencyInjection\Definition;
53
use Symfony\Component\DependencyInjection\Exception\RuntimeException;
54
use Symfony\Component\DependencyInjection\Extension\Extension;
55
use Symfony\Component\DependencyInjection\Extension\PrependExtensionInterface;
56
use Symfony\Component\DependencyInjection\Loader\XmlFileLoader;
57
use Symfony\Component\DependencyInjection\Reference;
58
use Symfony\Component\Finder\Finder;
59
use Symfony\Component\HttpClient\ScopingHttpClient;
60
use Symfony\Component\Serializer\NameConverter\CamelCaseToSnakeCaseNameConverter;
61
use Symfony\Component\Uid\AbstractUid;
62
use Symfony\Component\Validator\Validator\ValidatorInterface;
63
use Symfony\Component\Yaml\Yaml;
64
use Twig\Environment;
65

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

99
    /**
100
     * {@inheritdoc}
101
     */
102
    public function load(array $configs, ContainerBuilder $container): void
103
    {
104
        $loader = new XmlFileLoader($container, new FileLocator(__DIR__.'/../Resources/config'));
1✔
105

106
        $configuration = new Configuration();
1✔
107
        $config = $this->processConfiguration($configuration, $configs);
1✔
108
        $container->setParameter('api_platform.use_symfony_listeners', $config['use_symfony_listeners']);
1✔
109

110
        $formats = $this->getFormats($config['formats']);
1✔
111
        $patchFormats = $this->getFormats($config['patch_formats']);
1✔
112
        $errorFormats = $this->getFormats($config['error_formats']);
1✔
113
        $docsFormats = $this->getFormats($config['docs_formats']);
1✔
114
        if (!$config['enable_docs']) {
1✔
115
            // JSON-LD documentation format is mandatory, even if documentation is disabled.
116
            $docsFormats = isset($formats['jsonld']) ? ['jsonld' => ['application/ld+json']] : [];
×
117
            // If documentation is disabled, the Hydra documentation for all the resources is hidden by default.
118
            if (!isset($config['defaults']['hideHydraOperation']) && !isset($config['defaults']['hide_hydra_operation'])) {
×
119
                $config['defaults']['hideHydraOperation'] = true;
×
120
            }
121
        }
122
        $jsonSchemaFormats = $config['jsonschema_formats'];
1✔
123

124
        if (!$jsonSchemaFormats) {
1✔
125
            foreach (array_merge(array_keys($formats), array_keys($errorFormats)) as $f) {
1✔
126
                // Distinct JSON-based formats must have names that start with 'json'
127
                if (str_starts_with($f, 'json')) {
1✔
128
                    $jsonSchemaFormats[$f] = true;
1✔
129
                }
130
            }
131
        }
132

133
        if (!isset($errorFormats['json'])) {
1✔
134
            $errorFormats['json'] = ['application/problem+json', 'application/json'];
1✔
135
        }
136

137
        if (!isset($errorFormats['jsonproblem'])) {
1✔
138
            $errorFormats['jsonproblem'] = ['application/problem+json'];
×
139
        }
140

141
        if (isset($formats['jsonapi']) && !isset($patchFormats['jsonapi'])) {
1✔
142
            $patchFormats['jsonapi'] = ['application/vnd.api+json'];
1✔
143
        }
144

145
        $this->registerCommonConfiguration($container, $config, $loader, $formats, $patchFormats, $errorFormats, $docsFormats, $jsonSchemaFormats);
1✔
146
        $this->registerMetadataConfiguration($container, $config, $loader);
1✔
147
        $this->registerOAuthConfiguration($container, $config);
1✔
148
        $this->registerOpenApiConfiguration($container, $config, $loader);
1✔
149
        $this->registerSwaggerConfiguration($container, $config, $loader);
1✔
150
        $this->registerJsonApiConfiguration($formats, $loader, $config);
1✔
151
        $this->registerJsonLdHydraConfiguration($container, $formats, $loader, $config);
1✔
152
        $this->registerJsonHalConfiguration($formats, $loader);
1✔
153
        $this->registerJsonProblemConfiguration($errorFormats, $loader);
1✔
154
        $this->registerGraphQlConfiguration($container, $config, $loader);
1✔
155
        $this->registerCacheConfiguration($container);
1✔
156
        $this->registerDoctrineOrmConfiguration($container, $config, $loader);
1✔
157
        $this->registerDoctrineMongoDbOdmConfiguration($container, $config, $loader);
1✔
158
        $this->registerHttpCacheConfiguration($container, $config, $loader);
1✔
159
        $this->registerValidatorConfiguration($container, $config, $loader);
1✔
160
        $this->registerDataCollectorConfiguration($container, $config, $loader);
1✔
161
        $this->registerMercureConfiguration($container, $config, $loader);
1✔
162
        $this->registerMessengerConfiguration($container, $config, $loader);
1✔
163
        $this->registerElasticsearchConfiguration($container, $config, $loader);
1✔
164
        $this->registerSecurityConfiguration($container, $config, $loader);
1✔
165
        $this->registerMakerConfiguration($container, $config, $loader);
1✔
166
        $this->registerArgumentResolverConfiguration($loader);
1✔
167
        $this->registerLinkSecurityConfiguration($loader, $config);
1✔
168

169
        $container->registerForAutoconfiguration(FilterInterface::class)
1✔
170
            ->addTag('api_platform.filter');
1✔
171
        $container->registerForAutoconfiguration(ProviderInterface::class)
1✔
172
            ->addTag('api_platform.state_provider');
1✔
173
        $container->registerForAutoconfiguration(ProcessorInterface::class)
1✔
174
            ->addTag('api_platform.state_processor');
1✔
175
        $container->registerForAutoconfiguration(UriVariableTransformerInterface::class)
1✔
176
            ->addTag('api_platform.uri_variables.transformer');
1✔
177
        $container->registerForAutoconfiguration(ParameterProviderInterface::class)
1✔
178
            ->addTag('api_platform.parameter_provider');
1✔
179

180
        if (!$container->has('api_platform.state.item_provider')) {
1✔
181
            $container->setAlias('api_platform.state.item_provider', 'api_platform.state_provider.object');
×
182
        }
183
    }
184

185
    private function registerCommonConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader, array $formats, array $patchFormats, array $errorFormats, array $docsFormats, array $jsonSchemaFormats): void
186
    {
187
        $loader->load('state/state.xml');
1✔
188
        $loader->load('symfony/symfony.xml');
1✔
189
        $loader->load('api.xml');
1✔
190
        $loader->load('filter.xml');
1✔
191

192
        if (class_exists(UuidDenormalizer::class) && class_exists(Uuid::class)) {
1✔
193
            $loader->load('ramsey_uuid.xml');
1✔
194
        }
195

196
        if (class_exists(AbstractUid::class)) {
1✔
197
            $loader->load('symfony/uid.xml');
1✔
198
        }
199

200
        $defaultContext = ['hydra_prefix' => $config['serializer']['hydra_prefix']] + ($container->hasParameter('serializer.default_context') ? $container->getParameter('serializer.default_context') : []);
1✔
201

202
        $container->setParameter('api_platform.serializer.default_context', $defaultContext);
1✔
203
        if (!$container->hasParameter('serializer.default_context')) {
1✔
204
            $container->setParameter('serializer.default_context', $container->getParameter('api_platform.serializer.default_context'));
1✔
205
        }
206
        if ($config['use_symfony_listeners']) {
1✔
207
            $loader->load('symfony/events.xml');
1✔
208
        } else {
UNCOV
209
            $loader->load('symfony/controller.xml');
×
UNCOV
210
            $loader->load('state/provider.xml');
×
UNCOV
211
            $loader->load('state/processor.xml');
×
212
        }
213
        $loader->load('state/parameter_provider.xml');
1✔
214

215
        $container->setParameter('api_platform.enable_entrypoint', $config['enable_entrypoint']);
1✔
216
        $container->setParameter('api_platform.enable_docs', $config['enable_docs']);
1✔
217
        $container->setParameter('api_platform.title', $config['title']);
1✔
218
        $container->setParameter('api_platform.description', $config['description']);
1✔
219
        $container->setParameter('api_platform.version', $config['version']);
1✔
220
        $container->setParameter('api_platform.show_webby', $config['show_webby']);
1✔
221
        $container->setParameter('api_platform.url_generation_strategy', $config['defaults']['url_generation_strategy'] ?? UrlGeneratorInterface::ABS_PATH);
1✔
222
        $container->setParameter('api_platform.exception_to_status', $config['exception_to_status']);
1✔
223
        $container->setParameter('api_platform.formats', $formats);
1✔
224
        $container->setParameter('api_platform.patch_formats', $patchFormats);
1✔
225
        $container->setParameter('api_platform.error_formats', $errorFormats);
1✔
226
        $container->setParameter('api_platform.docs_formats', $docsFormats);
1✔
227
        $container->setParameter('api_platform.jsonschema_formats', $jsonSchemaFormats);
1✔
228
        $container->setParameter('api_platform.eager_loading.enabled', $this->isConfigEnabled($container, $config['eager_loading']));
1✔
229
        $container->setParameter('api_platform.eager_loading.max_joins', $config['eager_loading']['max_joins']);
1✔
230
        $container->setParameter('api_platform.eager_loading.fetch_partial', $config['eager_loading']['fetch_partial']);
1✔
231
        $container->setParameter('api_platform.eager_loading.force_eager', $config['eager_loading']['force_eager']);
1✔
232
        $container->setParameter('api_platform.collection.exists_parameter_name', $config['collection']['exists_parameter_name']);
1✔
233
        $container->setParameter('api_platform.collection.order', $config['collection']['order']);
1✔
234
        $container->setParameter('api_platform.collection.order_parameter_name', $config['collection']['order_parameter_name']);
1✔
235
        $container->setParameter('api_platform.collection.order_nulls_comparison', $config['collection']['order_nulls_comparison']);
1✔
236
        $container->setParameter('api_platform.collection.pagination.enabled', $config['defaults']['pagination_enabled'] ?? true);
1✔
237
        $container->setParameter('api_platform.collection.pagination.partial', $config['defaults']['pagination_partial'] ?? false);
1✔
238
        $container->setParameter('api_platform.collection.pagination.client_enabled', $config['defaults']['pagination_client_enabled'] ?? false);
1✔
239
        $container->setParameter('api_platform.collection.pagination.client_items_per_page', $config['defaults']['pagination_client_items_per_page'] ?? false);
1✔
240
        $container->setParameter('api_platform.collection.pagination.client_partial', $config['defaults']['pagination_client_partial'] ?? false);
1✔
241
        $container->setParameter('api_platform.collection.pagination.items_per_page', $config['defaults']['pagination_items_per_page'] ?? 30);
1✔
242
        $container->setParameter('api_platform.collection.pagination.maximum_items_per_page', $config['defaults']['pagination_maximum_items_per_page'] ?? null);
1✔
243
        $container->setParameter('api_platform.collection.pagination.page_parameter_name', $config['defaults']['pagination_page_parameter_name'] ?? $config['collection']['pagination']['page_parameter_name']);
1✔
244
        $container->setParameter('api_platform.collection.pagination.enabled_parameter_name', $config['defaults']['pagination_enabled_parameter_name'] ?? $config['collection']['pagination']['enabled_parameter_name']);
1✔
245
        $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']);
1✔
246
        $container->setParameter('api_platform.collection.pagination.partial_parameter_name', $config['defaults']['pagination_partial_parameter_name'] ?? $config['collection']['pagination']['partial_parameter_name']);
1✔
247
        $container->setParameter('api_platform.collection.pagination', $this->getPaginationDefaults($config['defaults'] ?? [], $config['collection']['pagination']));
1✔
248
        $container->setParameter('api_platform.handle_symfony_errors', $config['handle_symfony_errors'] ?? false);
1✔
249
        $container->setParameter('api_platform.http_cache.etag', $config['defaults']['cache_headers']['etag'] ?? true);
1✔
250
        $container->setParameter('api_platform.http_cache.max_age', $config['defaults']['cache_headers']['max_age'] ?? null);
1✔
251
        $container->setParameter('api_platform.http_cache.shared_max_age', $config['defaults']['cache_headers']['shared_max_age'] ?? null);
1✔
252
        $container->setParameter('api_platform.http_cache.vary', $config['defaults']['cache_headers']['vary'] ?? ['Accept']);
1✔
253
        $container->setParameter('api_platform.http_cache.public', $config['defaults']['cache_headers']['public'] ?? $config['http_cache']['public']);
1✔
254
        $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']);
1✔
255
        $container->setParameter('api_platform.http_cache.invalidation.xkey.glue', $config['defaults']['cache_headers']['invalidation']['xkey']['glue'] ?? $config['http_cache']['invalidation']['xkey']['glue']);
1✔
256

257
        $container->setAlias('api_platform.path_segment_name_generator', $config['path_segment_name_generator']);
1✔
258
        $container->setAlias('api_platform.inflector', $config['inflector']);
1✔
259

260
        if ($config['name_converter']) {
1✔
261
            $container->setAlias('api_platform.name_converter', $config['name_converter']);
1✔
262
        }
263
        $container->setParameter('api_platform.asset_package', $config['asset_package']);
1✔
264
        $container->setParameter('api_platform.defaults', $this->normalizeDefaults($config['defaults'] ?? []));
1✔
265

266
        if ($container->getParameter('kernel.debug')) {
1✔
267
            $container->removeDefinition('api_platform.serializer.mapping.cache_class_metadata_factory');
1✔
268
        }
269
    }
270

271
    /**
272
     * This method will be removed in 3.0 when "defaults" will be the regular configuration path for the pagination.
273
     */
274
    private function getPaginationDefaults(array $defaults, array $collectionPaginationConfiguration): array
275
    {
276
        $paginationOptions = [];
1✔
277

278
        foreach ($defaults as $key => $value) {
1✔
279
            if (!str_starts_with($key, 'pagination_')) {
1✔
280
                continue;
1✔
281
            }
282

283
            $paginationOptions[str_replace('pagination_', '', $key)] = $value;
1✔
284
        }
285

286
        return array_merge($collectionPaginationConfiguration, $paginationOptions);
1✔
287
    }
288

289
    private function normalizeDefaults(array $defaults): array
290
    {
291
        $normalizedDefaults = ['extra_properties' => $defaults['extra_properties'] ?? []];
1✔
292
        unset($defaults['extra_properties']);
1✔
293

294
        $rc = new \ReflectionClass(ApiResource::class);
1✔
295
        $publicProperties = [];
1✔
296
        foreach ($rc->getConstructor()->getParameters() as $param) {
1✔
297
            $publicProperties[$param->getName()] = true;
1✔
298
        }
299

300
        $nameConverter = new CamelCaseToSnakeCaseNameConverter();
1✔
301
        foreach ($defaults as $option => $value) {
1✔
302
            if (isset($publicProperties[$nameConverter->denormalize($option)])) {
1✔
303
                $normalizedDefaults[$option] = $value;
1✔
304

305
                continue;
1✔
306
            }
307

308
            $normalizedDefaults['extra_properties'][$option] = $value;
×
309
        }
310

311
        return $normalizedDefaults;
1✔
312
    }
313

314
    private function registerMetadataConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
315
    {
316
        [$xmlResources, $yamlResources] = $this->getResourcesToWatch($container, $config);
1✔
317

318
        $container->setParameter('api_platform.class_name_resources', $this->getClassNameResources());
1✔
319

320
        $loader->load('metadata/resource_name.xml');
1✔
321
        $loader->load('metadata/property_name.xml');
1✔
322

323
        if (!empty($config['resource_class_directories'])) {
1✔
324
            $container->setParameter('api_platform.resource_class_directories', array_merge(
×
325
                $config['resource_class_directories'],
×
326
                $container->getParameter('api_platform.resource_class_directories')
×
327
            ));
×
328
        }
329

330
        // V3 metadata
331
        $loader->load('metadata/xml.xml');
1✔
332
        $loader->load('metadata/links.xml');
1✔
333
        $loader->load('metadata/property.xml');
1✔
334
        $loader->load('metadata/resource.xml');
1✔
335
        $loader->load('metadata/operation.xml');
1✔
336

337
        $container->getDefinition('api_platform.metadata.resource_extractor.xml')->replaceArgument(0, $xmlResources);
1✔
338
        $container->getDefinition('api_platform.metadata.property_extractor.xml')->replaceArgument(0, $xmlResources);
1✔
339

340
        if (class_exists(PhpDocParser::class)) {
1✔
341
            $loader->load('metadata/php_doc.xml');
1✔
342
        }
343

344
        if (class_exists(Yaml::class)) {
1✔
345
            $loader->load('metadata/yaml.xml');
1✔
346
            $container->getDefinition('api_platform.metadata.resource_extractor.yaml')->replaceArgument(0, $yamlResources);
1✔
347
            $container->getDefinition('api_platform.metadata.property_extractor.yaml')->replaceArgument(0, $yamlResources);
1✔
348
        }
349
    }
350

351
    private function getClassNameResources(): array
352
    {
353
        return [
1✔
354
            Error::class,
1✔
355
            ValidationException::class,
1✔
356
        ];
1✔
357
    }
358

359
    private function getBundlesResourcesPaths(ContainerBuilder $container, array $config): array
360
    {
361
        $bundlesResourcesPaths = [];
1✔
362

363
        foreach ($container->getParameter('kernel.bundles_metadata') as $bundle) {
1✔
364
            $dirname = $bundle['path'];
1✔
365
            $paths = [
1✔
366
                "$dirname/ApiResource",
1✔
367
                "$dirname/src/ApiResource",
1✔
368
            ];
1✔
369
            foreach (['.yaml', '.yml', '.xml', ''] as $extension) {
1✔
370
                $paths[] = "$dirname/Resources/config/api_resources$extension";
1✔
371
                $paths[] = "$dirname/config/api_resources$extension";
1✔
372
            }
373
            if ($this->isConfigEnabled($container, $config['doctrine'])) {
1✔
374
                $paths[] = "$dirname/Entity";
1✔
375
                $paths[] = "$dirname/src/Entity";
1✔
376
            }
377
            if ($this->isConfigEnabled($container, $config['doctrine_mongodb_odm'])) {
1✔
378
                $paths[] = "$dirname/Document";
×
379
                $paths[] = "$dirname/src/Document";
×
380
            }
381

382
            foreach ($paths as $path) {
1✔
383
                if ($container->fileExists($path, false)) {
1✔
384
                    $bundlesResourcesPaths[] = $path;
1✔
385
                }
386
            }
387
        }
388

389
        return $bundlesResourcesPaths;
1✔
390
    }
391

392
    private function getResourcesToWatch(ContainerBuilder $container, array $config): array
393
    {
394
        $paths = array_unique(array_merge($this->getBundlesResourcesPaths($container, $config), $config['mapping']['paths']));
1✔
395

396
        if (!$config['mapping']['paths']) {
1✔
397
            $projectDir = $container->getParameter('kernel.project_dir');
×
398
            foreach (["$projectDir/config/api_platform", "$projectDir/src/ApiResource"] as $dir) {
×
399
                if (is_dir($dir)) {
×
400
                    $paths[] = $dir;
×
401
                }
402
            }
403

404
            if ($this->isConfigEnabled($container, $config['doctrine']) && is_dir($doctrinePath = "$projectDir/src/Entity")) {
×
405
                $paths[] = $doctrinePath;
×
406
            }
407

408
            if ($this->isConfigEnabled($container, $config['doctrine_mongodb_odm']) && is_dir($documentPath = "$projectDir/src/Document")) {
×
409
                $paths[] = $documentPath;
×
410
            }
411
        }
412

413
        $resources = ['yml' => [], 'xml' => [], 'dir' => []];
1✔
414

415
        foreach ($paths as $path) {
1✔
416
            if (is_dir($path)) {
1✔
417
                foreach (Finder::create()->followLinks()->files()->in($path)->name('/\.(xml|ya?ml)$/')->sortByName() as $file) {
1✔
418
                    $resources['yaml' === ($extension = $file->getExtension()) ? 'yml' : $extension][] = $file->getRealPath();
1✔
419
                }
420

421
                $resources['dir'][] = $path;
1✔
422
                $container->addResource(new DirectoryResource($path, '/\.(xml|ya?ml|php)$/'));
1✔
423

424
                continue;
1✔
425
            }
426

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

432
                $resources['yaml' === $matches[1] ? 'yml' : $matches[1]][] = $path;
×
433

434
                continue;
×
435
            }
436

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

440
        $container->setParameter('api_platform.resource_class_directories', $resources['dir']);
1✔
441

442
        return [$resources['xml'], $resources['yml']];
1✔
443
    }
444

445
    private function registerOAuthConfiguration(ContainerBuilder $container, array $config): void
446
    {
447
        if (!$config['oauth']) {
1✔
448
            return;
×
449
        }
450

451
        $container->setParameter('api_platform.oauth.enabled', $this->isConfigEnabled($container, $config['oauth']));
1✔
452
        $container->setParameter('api_platform.oauth.clientId', $config['oauth']['clientId']);
1✔
453
        $container->setParameter('api_platform.oauth.clientSecret', $config['oauth']['clientSecret']);
1✔
454
        $container->setParameter('api_platform.oauth.type', $config['oauth']['type']);
1✔
455
        $container->setParameter('api_platform.oauth.flow', $config['oauth']['flow']);
1✔
456
        $container->setParameter('api_platform.oauth.tokenUrl', $config['oauth']['tokenUrl']);
1✔
457
        $container->setParameter('api_platform.oauth.authorizationUrl', $config['oauth']['authorizationUrl']);
1✔
458
        $container->setParameter('api_platform.oauth.refreshUrl', $config['oauth']['refreshUrl']);
1✔
459
        $container->setParameter('api_platform.oauth.scopes', $config['oauth']['scopes']);
1✔
460
        $container->setParameter('api_platform.oauth.pkce', $config['oauth']['pkce']);
1✔
461
    }
462

463
    /**
464
     * Registers the Swagger, ReDoc and Swagger UI configuration.
465
     */
466
    private function registerSwaggerConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
467
    {
468
        foreach (array_keys($config['swagger']['api_keys']) as $keyName) {
1✔
469
            if (!preg_match('/^[a-zA-Z0-9._-]+$/', $keyName)) {
1✔
470
                throw new RuntimeException(\sprintf('The swagger api_keys key "%s" is not valid, it should match "^[a-zA-Z0-9._-]+$"', $keyName));
×
471
            }
472
        }
473

474
        $container->setParameter('api_platform.swagger.versions', $config['swagger']['versions']);
1✔
475

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

480
        if (!$config['enable_swagger']) {
1✔
481
            return;
×
482
        }
483

484
        $loader->load('openapi.xml');
1✔
485

486
        if (class_exists(Yaml::class)) {
1✔
487
            $loader->load('openapi/yaml.xml');
1✔
488
        }
489

490
        $loader->load('swagger_ui.xml');
1✔
491

492
        if ($config['use_symfony_listeners']) {
1✔
493
            $loader->load('symfony/swagger_ui.xml');
1✔
494
        }
495

496
        if ($config['enable_swagger_ui']) {
1✔
497
            $loader->load('state/swagger_ui.xml');
1✔
498
        }
499

500
        if (!$config['enable_swagger_ui'] && !$config['enable_re_doc']) {
1✔
501
            // Remove the listener but keep the controller to allow customizing the path of the UI
502
            $container->removeDefinition('api_platform.swagger.listener.ui');
×
503
        }
504

505
        $container->setParameter('api_platform.enable_swagger_ui', $config['enable_swagger_ui']);
1✔
506
        $container->setParameter('api_platform.enable_re_doc', $config['enable_re_doc']);
1✔
507
        $container->setParameter('api_platform.swagger.api_keys', $config['swagger']['api_keys']);
1✔
508
        $container->setParameter('api_platform.swagger.persist_authorization', $config['swagger']['persist_authorization']);
1✔
509
        $container->setParameter('api_platform.swagger.http_auth', $config['swagger']['http_auth']);
1✔
510
        if ($config['openapi']['swagger_ui_extra_configuration'] && $config['swagger']['swagger_ui_extra_configuration']) {
1✔
511
            throw new RuntimeException('You can not set "swagger_ui_extra_configuration" twice - in "openapi" and "swagger" section.');
×
512
        }
513
        $container->setParameter('api_platform.swagger_ui.extra_configuration', $config['openapi']['swagger_ui_extra_configuration'] ?: $config['swagger']['swagger_ui_extra_configuration']);
1✔
514
    }
515

516
    private function registerJsonApiConfiguration(array $formats, XmlFileLoader $loader, array $config): void
517
    {
518
        if (!isset($formats['jsonapi'])) {
1✔
519
            return;
×
520
        }
521

522
        $loader->load('jsonapi.xml');
1✔
523
        $loader->load('state/jsonapi.xml');
1✔
524
    }
525

526
    private function registerJsonLdHydraConfiguration(ContainerBuilder $container, array $formats, XmlFileLoader $loader, array $config): void
527
    {
528
        if (!isset($formats['jsonld'])) {
1✔
529
            return;
×
530
        }
531

532
        if ($config['use_symfony_listeners']) {
1✔
533
            $loader->load('symfony/jsonld.xml');
1✔
534
        } else {
UNCOV
535
            $loader->load('state/jsonld.xml');
×
536
        }
537

538
        $loader->load('state/hydra.xml');
1✔
539
        $loader->load('jsonld.xml');
1✔
540
        $loader->load('hydra.xml');
1✔
541

542
        if (!$container->has('api_platform.json_schema.schema_factory')) {
1✔
543
            $container->removeDefinition('api_platform.hydra.json_schema.schema_factory');
×
544
        }
545
    }
546

547
    private function registerJsonHalConfiguration(array $formats, XmlFileLoader $loader): void
548
    {
549
        if (!isset($formats['jsonhal'])) {
1✔
550
            return;
×
551
        }
552

553
        $loader->load('hal.xml');
1✔
554
    }
555

556
    private function registerJsonProblemConfiguration(array $errorFormats, XmlFileLoader $loader): void
557
    {
558
        if (!isset($errorFormats['jsonproblem'])) {
1✔
559
            return;
×
560
        }
561

562
        $loader->load('problem.xml');
1✔
563
    }
564

565
    private function registerGraphQlConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
566
    {
567
        $enabled = $this->isConfigEnabled($container, $config['graphql']);
1✔
568
        $graphqlIntrospectionEnabled = $enabled && $this->isConfigEnabled($container, $config['graphql']['introspection']);
1✔
569
        $graphiqlEnabled = $enabled && $this->isConfigEnabled($container, $config['graphql']['graphiql']);
1✔
570
        $maxQueryDepth = (int) $config['graphql']['max_query_depth'];
1✔
571
        $maxQueryComplexity = (int) $config['graphql']['max_query_complexity'];
1✔
572

573
        $container->setParameter('api_platform.graphql.enabled', $enabled);
1✔
574
        $container->setParameter('api_platform.graphql.max_query_depth', $maxQueryDepth);
1✔
575
        $container->setParameter('api_platform.graphql.max_query_complexity', $maxQueryComplexity);
1✔
576
        $container->setParameter('api_platform.graphql.introspection.enabled', $graphqlIntrospectionEnabled);
1✔
577
        $container->setParameter('api_platform.graphql.graphiql.enabled', $graphiqlEnabled);
1✔
578
        $container->setParameter('api_platform.graphql.collection.pagination', $config['graphql']['collection']['pagination']);
1✔
579

580
        if (!$enabled) {
1✔
581
            return;
×
582
        }
583

584
        if (!class_exists(Executor::class)) {
1✔
585
            throw new \RuntimeException('Graphql is enabled but not installed, run: composer require "api-platform/graphql".');
×
586
        }
587

588
        $container->setParameter('api_platform.graphql.default_ide', $config['graphql']['default_ide']);
1✔
589
        $container->setParameter('api_platform.graphql.nesting_separator', $config['graphql']['nesting_separator']);
1✔
590

591
        $loader->load('graphql.xml');
1✔
592

593
        // @phpstan-ignore-next-line because PHPStan uses the container of the test env cache and in test the parameter kernel.bundles always contains the key TwigBundle
594
        if (!class_exists(Environment::class) || !isset($container->getParameter('kernel.bundles')['TwigBundle'])) {
1✔
595
            if ($graphiqlEnabled) {
×
596
                throw new RuntimeException(\sprintf('GraphiQL interfaces depend on Twig. Please activate TwigBundle for the %s environnement or disable GraphiQL.', $container->getParameter('kernel.environment')));
×
597
            }
598
            $container->removeDefinition('api_platform.graphql.action.graphiql');
×
599
        }
600

601
        $container->registerForAutoconfiguration(QueryItemResolverInterface::class)
1✔
602
            ->addTag('api_platform.graphql.resolver');
1✔
603
        $container->registerForAutoconfiguration(QueryCollectionResolverInterface::class)
1✔
604
            ->addTag('api_platform.graphql.resolver');
1✔
605
        $container->registerForAutoconfiguration(MutationResolverInterface::class)
1✔
606
            ->addTag('api_platform.graphql.resolver');
1✔
607
        $container->registerForAutoconfiguration(GraphQlTypeInterface::class)
1✔
608
            ->addTag('api_platform.graphql.type');
1✔
609
        $container->registerForAutoconfiguration(ErrorHandlerInterface::class)
1✔
610
            ->addTag('api_platform.graphql.error_handler');
1✔
611
    }
612

613
    private function registerCacheConfiguration(ContainerBuilder $container): void
614
    {
615
        if (!$container->hasParameter('kernel.debug') || !$container->getParameter('kernel.debug')) {
1✔
616
            $container->removeDefinition('api_platform.cache_warmer.cache_pool_clearer');
×
617
        }
618
    }
619

620
    private function registerDoctrineOrmConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
621
    {
622
        if (!$this->isConfigEnabled($container, $config['doctrine'])) {
1✔
623
            return;
×
624
        }
625

626
        // For older versions of doctrine bridge this allows autoconfiguration for filters
627
        if (!$container->has(ManagerRegistry::class)) {
1✔
628
            $container->setAlias(ManagerRegistry::class, 'doctrine');
1✔
629
        }
630

631
        $container->registerForAutoconfiguration(QueryItemExtensionInterface::class)
1✔
632
            ->addTag('api_platform.doctrine.orm.query_extension.item');
1✔
633
        $container->registerForAutoconfiguration(DoctrineQueryCollectionExtensionInterface::class)
1✔
634
            ->addTag('api_platform.doctrine.orm.query_extension.collection');
1✔
635
        $container->registerForAutoconfiguration(DoctrineOrmAbstractFilter::class);
1✔
636

637
        $container->registerForAutoconfiguration(OrmLinksHandlerInterface::class)
1✔
638
            ->addTag('api_platform.doctrine.orm.links_handler');
1✔
639

640
        $loader->load('doctrine_orm.xml');
1✔
641

642
        if ($this->isConfigEnabled($container, $config['eager_loading'])) {
1✔
643
            return;
1✔
644
        }
645

646
        $container->removeAlias(EagerLoadingExtension::class);
×
647
        $container->removeDefinition('api_platform.doctrine.orm.query_extension.eager_loading');
×
648
        $container->removeAlias(FilterEagerLoadingExtension::class);
×
649
        $container->removeDefinition('api_platform.doctrine.orm.query_extension.filter_eager_loading');
×
650
    }
651

652
    private function registerDoctrineMongoDbOdmConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
653
    {
654
        if (!$this->isConfigEnabled($container, $config['doctrine_mongodb_odm'])) {
1✔
655
            return;
1✔
656
        }
657

658
        $container->registerForAutoconfiguration(AggregationItemExtensionInterface::class)
×
659
            ->addTag('api_platform.doctrine_mongodb.odm.aggregation_extension.item');
×
660
        $container->registerForAutoconfiguration(AggregationCollectionExtensionInterface::class)
×
661
            ->addTag('api_platform.doctrine_mongodb.odm.aggregation_extension.collection');
×
662
        $container->registerForAutoconfiguration(DoctrineMongoDbOdmAbstractFilter::class)
×
663
            ->setBindings(['$managerRegistry' => new Reference('doctrine_mongodb')]);
×
664
        $container->registerForAutoconfiguration(OdmLinksHandlerInterface::class)
×
665
            ->addTag('api_platform.doctrine.odm.links_handler');
×
666

667
        $loader->load('doctrine_mongodb_odm.xml');
×
668
    }
669

670
    private function registerHttpCacheConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
671
    {
672
        $loader->load('http_cache.xml');
1✔
673

674
        if (!$this->isConfigEnabled($container, $config['http_cache']['invalidation'])) {
1✔
675
            return;
×
676
        }
677

678
        if ($this->isConfigEnabled($container, $config['doctrine'])) {
1✔
679
            $loader->load('doctrine_orm_http_cache_purger.xml');
1✔
680
        }
681

682
        $loader->load('state/http_cache_purger.xml');
1✔
683
        $loader->load('http_cache_purger.xml');
1✔
684

685
        foreach ($config['http_cache']['invalidation']['scoped_clients'] as $client) {
1✔
686
            $definition = $container->getDefinition($client);
×
687
            $definition->addTag('api_platform.http_cache.http_client');
×
688
        }
689

690
        if (!($urls = $config['http_cache']['invalidation']['urls'])) {
1✔
691
            $urls = $config['http_cache']['invalidation']['varnish_urls'];
1✔
692
        }
693

694
        foreach ($urls as $key => $url) {
1✔
695
            $definition = new Definition(ScopingHttpClient::class, [new Reference('http_client'), $url, ['base_uri' => $url] + $config['http_cache']['invalidation']['request_options']]);
×
696
            $definition->setFactory([ScopingHttpClient::class, 'forBaseUri']);
×
697
            $definition->addTag('api_platform.http_cache.http_client');
×
698
            $container->setDefinition('api_platform.invalidation_http_client.'.$key, $definition);
×
699
        }
700

701
        $serviceName = $config['http_cache']['invalidation']['purger'];
1✔
702

703
        if (!$container->hasDefinition('api_platform.http_cache.purger')) {
1✔
704
            $container->setAlias('api_platform.http_cache.purger', $serviceName);
1✔
705
        }
706
    }
707

708
    /**
709
     * Normalizes the format from config to the one accepted by Symfony HttpFoundation.
710
     */
711
    private function getFormats(array $configFormats): array
712
    {
713
        $formats = [];
1✔
714
        foreach ($configFormats as $format => $value) {
1✔
715
            foreach ($value['mime_types'] as $mimeType) {
1✔
716
                $formats[$format][] = $mimeType;
1✔
717
            }
718
        }
719

720
        return $formats;
1✔
721
    }
722

723
    private function registerValidatorConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
724
    {
725
        if (interface_exists(ValidatorInterface::class)) {
1✔
726
            $loader->load('metadata/validator.xml');
1✔
727
            $loader->load('validator/validator.xml');
1✔
728

729
            if ($this->isConfigEnabled($container, $config['graphql'])) {
1✔
730
                $loader->load('graphql/validator.xml');
1✔
731
            }
732

733
            $loader->load($config['use_symfony_listeners'] ? 'validator/events.xml' : 'validator/state.xml');
1✔
734

735
            $container->registerForAutoconfiguration(ValidationGroupsGeneratorInterface::class)
1✔
736
                ->addTag('api_platform.validation_groups_generator');
1✔
737
            $container->registerForAutoconfiguration(PropertySchemaRestrictionMetadataInterface::class)
1✔
738
                ->addTag('api_platform.metadata.property_schema_restriction');
1✔
739
        }
740

741
        if (!$config['validator']) {
1✔
742
            return;
×
743
        }
744

745
        $container->setParameter('api_platform.validator.serialize_payload_fields', $config['validator']['serialize_payload_fields']);
1✔
746
        $container->setParameter('api_platform.validator.query_parameter_validation', $config['validator']['query_parameter_validation']);
1✔
747

748
        if (!$config['validator']['query_parameter_validation']) {
1✔
749
            $container->removeDefinition('api_platform.listener.view.validate_query_parameters');
×
750
            $container->removeDefinition('api_platform.validator.query_parameter_validator');
×
751
            $container->removeDefinition('api_platform.symfony.parameter_validator');
×
752
        }
753
    }
754

755
    private function registerDataCollectorConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
756
    {
757
        if (!$config['enable_profiler']) {
1✔
758
            return;
×
759
        }
760

761
        $loader->load('data_collector.xml');
1✔
762

763
        if ($container->hasParameter('kernel.debug') && $container->getParameter('kernel.debug')) {
1✔
764
            $loader->load('debug.xml');
1✔
765
        }
766
    }
767

768
    private function registerMercureConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
769
    {
770
        if (!$this->isConfigEnabled($container, $config['mercure'])) {
1✔
771
            return;
×
772
        }
773

774
        $container->setParameter('api_platform.mercure.include_type', $config['mercure']['include_type']);
1✔
775
        $loader->load('state/mercure.xml');
1✔
776

777
        if ($this->isConfigEnabled($container, $config['doctrine'])) {
1✔
778
            $loader->load('doctrine_orm_mercure_publisher.xml');
1✔
779
        }
780
        if ($this->isConfigEnabled($container, $config['doctrine_mongodb_odm'])) {
1✔
781
            $loader->load('doctrine_odm_mercure_publisher.xml');
×
782
        }
783

784
        if ($this->isConfigEnabled($container, $config['graphql'])) {
1✔
785
            $loader->load('graphql_mercure.xml');
1✔
786
        }
787
    }
788

789
    private function registerMessengerConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
790
    {
791
        if (!$this->isConfigEnabled($container, $config['messenger'])) {
1✔
792
            return;
×
793
        }
794

795
        $loader->load('messenger.xml');
1✔
796
    }
797

798
    private function registerElasticsearchConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
799
    {
800
        $enabled = $this->isConfigEnabled($container, $config['elasticsearch']);
1✔
801

802
        $container->setParameter('api_platform.elasticsearch.enabled', $enabled);
1✔
803

804
        if (!$enabled) {
1✔
805
            return;
1✔
806
        }
807

808
        $clientClass = !class_exists(\Elasticsearch\Client::class)
×
809
            // ES v7
×
810
            ? \Elastic\Elasticsearch\Client::class
×
811
            // ES v8 and up
×
812
            : \Elasticsearch\Client::class;
×
813

814
        $clientDefinition = new Definition($clientClass);
×
815
        $container->setDefinition('api_platform.elasticsearch.client', $clientDefinition);
×
816
        $container->registerForAutoconfiguration(RequestBodySearchCollectionExtensionInterface::class)
×
817
            ->addTag('api_platform.elasticsearch.request_body_search_extension.collection');
×
818
        $container->setParameter('api_platform.elasticsearch.hosts', $config['elasticsearch']['hosts']);
×
819
        $loader->load('elasticsearch.xml');
×
820
    }
821

822
    private function registerSecurityConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
823
    {
824
        /** @var string[] $bundles */
825
        $bundles = $container->getParameter('kernel.bundles');
1✔
826

827
        if (!isset($bundles['SecurityBundle'])) {
1✔
828
            return;
×
829
        }
830

831
        $loader->load('security.xml');
1✔
832

833
        $loader->load('state/security.xml');
1✔
834

835
        if (interface_exists(ValidatorInterface::class)) {
1✔
836
            $loader->load('state/security_validator.xml');
1✔
837
        }
838

839
        if ($this->isConfigEnabled($container, $config['graphql'])) {
1✔
840
            $loader->load('graphql/security.xml');
1✔
841
        }
842
    }
843

844
    private function registerOpenApiConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
845
    {
846
        $container->setParameter('api_platform.openapi.termsOfService', $config['openapi']['termsOfService']);
1✔
847
        $container->setParameter('api_platform.openapi.contact.name', $config['openapi']['contact']['name']);
1✔
848
        $container->setParameter('api_platform.openapi.contact.url', $config['openapi']['contact']['url']);
1✔
849
        $container->setParameter('api_platform.openapi.contact.email', $config['openapi']['contact']['email']);
1✔
850
        $container->setParameter('api_platform.openapi.license.name', $config['openapi']['license']['name']);
1✔
851
        $container->setParameter('api_platform.openapi.license.url', $config['openapi']['license']['url']);
1✔
852
        $container->setParameter('api_platform.openapi.overrideResponses', $config['openapi']['overrideResponses']);
1✔
853

854
        $tags = [];
1✔
855
        foreach ($config['openapi']['tags'] as $tag) {
1✔
856
            $tags[] = new Tag($tag['name'], $tag['description'] ?? null);
×
857
        }
858

859
        $container->setParameter('api_platform.openapi.tags', $tags);
1✔
860

861
        $loader->load('json_schema.xml');
1✔
862
    }
863

864
    private function registerMakerConfiguration(ContainerBuilder $container, array $config, XmlFileLoader $loader): void
865
    {
866
        if (!$this->isConfigEnabled($container, $config['maker'])) {
1✔
867
            return;
×
868
        }
869

870
        $loader->load('maker.xml');
1✔
871
    }
872

873
    private function registerArgumentResolverConfiguration(XmlFileLoader $loader): void
874
    {
875
        $loader->load('argument_resolver.xml');
1✔
876
    }
877

878
    private function registerLinkSecurityConfiguration(XmlFileLoader $loader, array $config): void
879
    {
880
        if ($config['enable_link_security']) {
1✔
881
            $loader->load('link_security.xml');
1✔
882
        }
883
    }
884
}
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

© 2025 Coveralls, Inc