Skip to content

Commit

Permalink
Merge 3.4
Browse files Browse the repository at this point in the history
  • Loading branch information
soyuka committed Oct 26, 2024
2 parents 4f65ef2 + 8bdf044 commit 4ef8b6a
Show file tree
Hide file tree
Showing 8 changed files with 315 additions and 65 deletions.
6 changes: 6 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -152,6 +152,12 @@ Notes:

* [0d5f35683](https://github.com/api-platform/core/commit/0d5f356839eb6aa9f536044abe4affa736553e76) feat(laravel): laravel component (#5882)

## v3.4.5

### Bug fixes

* [fc8fa00a1](https://github.com/api-platform/core/commit/fc8fa00a19320b65547a60537261959c11f8e6a8) fix(hydra): iri template when using query parameter (#6742)

## v3.4.4

### Bug fixes
Expand Down
4 changes: 2 additions & 2 deletions src/Hydra/Serializer/CollectionFiltersNormalizer.php
Original file line number Diff line number Diff line change
Expand Up @@ -153,7 +153,7 @@ private function getSearch(string $resourceClass, array $parts, array $filters,
continue;
}

if (!($property = $parameter->getProperty()) && ($filterId = $parameter->getFilter()) && ($filter = $this->getFilter($filterId))) {
if (($filterId = $parameter->getFilter()) && ($filter = $this->getFilter($filterId))) {
foreach ($filter->getDescription($resourceClass) as $variable => $description) {
// This is a practice induced by PHP and is not necessary when implementing URI template
if (str_ends_with((string) $variable, '[]')) {
Expand All @@ -174,7 +174,7 @@ private function getSearch(string $resourceClass, array $parts, array $filters,
continue;
}

if (!$property) {
if (!($property = $parameter->getProperty())) {
continue;
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -155,16 +155,12 @@ private function addFilterMetadata(Parameter $parameter): Parameter
return $parameter;
}

if (null === $parameter->getSchema() && $filter instanceof JsonSchemaFilterInterface) {
if ($schema = $filter->getSchema($parameter)) {
$parameter = $parameter->withSchema($schema);
}
if (null === $parameter->getSchema() && $filter instanceof JsonSchemaFilterInterface && $schema = $filter->getSchema($parameter)) {
$parameter = $parameter->withSchema($schema);
}

if (null === $parameter->getOpenApi() && $filter instanceof OpenApiParameterFilterInterface) {
if ($openApiParameter = $filter->getOpenApiParameters($parameter)) {
$parameter = $parameter->withOpenApi($openApiParameter);
}
if (null === $parameter->getOpenApi() && $filter instanceof OpenApiParameterFilterInterface && $openApiParameter = $filter->getOpenApiParameters($parameter)) {
$parameter = $parameter->withOpenApi($openApiParameter);
}

return $parameter;
Expand Down Expand Up @@ -194,10 +190,6 @@ private function setDefaults(string $key, Parameter $parameter, string $resource
$parameter = $parameter->withSchema($schema);
}

if (null === $parameter->getProperty() && ($property = $description[$key]['property'] ?? null)) {
$parameter = $parameter->withProperty($property);
}

$currentKey = $key;
if (null === $parameter->getProperty() && isset($properties[$key])) {
$parameter = $parameter->withProperty($key);
Expand All @@ -212,17 +204,83 @@ private function setDefaults(string $key, Parameter $parameter, string $resource
$parameter = $parameter->withExtraProperties(['_query_property' => $eloquentRelation['foreign_key']] + $parameter->getExtraProperties());
}

if (null === $parameter->getRequired() && ($required = $description[$key]['required'] ?? null)) {
$parameter = $parameter->withRequired($required);
}
$schema = $parameter->getSchema() ?? (($openApi = $parameter->getOpenApi()) ? $openApi->getSchema() : null);

if (null === $parameter->getOpenApi() && ($openApi = $description[$key]['openapi'] ?? null) && $openApi instanceof OpenApiParameter) {
$parameter = $parameter->withOpenApi($openApi);
// Only add validation if the Symfony Validator is installed
if (interface_exists(ValidatorInterface::class) && !$parameter->getConstraints()) {
$parameter = $this->addSchemaValidation($parameter, $schema, $parameter->getRequired() ?? $description['required'] ?? false, $parameter->getOpenApi() ?: null);
}

return $this->addFilterMetadata($parameter);
}

private function addSchemaValidation(Parameter $parameter, ?array $schema = null, bool $required = false, ?OpenApiParameter $openApi = null): Parameter
{
$assertions = [];

if ($required && false !== ($allowEmptyValue = $openApi?->getAllowEmptyValue())) {
$assertions[] = new NotNull(message: \sprintf('The parameter "%s" is required.', $parameter->getKey()));

Check failure on line 222 in src/Metadata/Resource/Factory/ParameterResourceMetadataCollectionFactory.php

View workflow job for this annotation

GitHub Actions / PHPStan (PHP 8.3)

Instantiated class ApiPlatform\Metadata\Resource\Factory\NotNull not found.
}

if (false === ($allowEmptyValue ?? $openApi?->getAllowEmptyValue())) {
$assertions[] = new NotBlank(allowNull: !$required);

Check failure on line 226 in src/Metadata/Resource/Factory/ParameterResourceMetadataCollectionFactory.php

View workflow job for this annotation

GitHub Actions / PHPStan (PHP 8.3)

Instantiated class ApiPlatform\Metadata\Resource\Factory\NotBlank not found.
}

if (isset($schema['exclusiveMinimum'])) {
$assertions[] = new GreaterThan(value: $schema['exclusiveMinimum']);

Check failure on line 230 in src/Metadata/Resource/Factory/ParameterResourceMetadataCollectionFactory.php

View workflow job for this annotation

GitHub Actions / PHPStan (PHP 8.3)

Instantiated class ApiPlatform\Metadata\Resource\Factory\GreaterThan not found.
}

if (isset($schema['exclusiveMaximum'])) {
$assertions[] = new LessThan(value: $schema['exclusiveMaximum']);

Check failure on line 234 in src/Metadata/Resource/Factory/ParameterResourceMetadataCollectionFactory.php

View workflow job for this annotation

GitHub Actions / PHPStan (PHP 8.3)

Instantiated class ApiPlatform\Metadata\Resource\Factory\LessThan not found.
}

if (isset($schema['minimum'])) {
$assertions[] = new GreaterThanOrEqual(value: $schema['minimum']);

Check failure on line 238 in src/Metadata/Resource/Factory/ParameterResourceMetadataCollectionFactory.php

View workflow job for this annotation

GitHub Actions / PHPStan (PHP 8.3)

Instantiated class ApiPlatform\Metadata\Resource\Factory\GreaterThanOrEqual not found.
}

if (isset($schema['maximum'])) {
$assertions[] = new LessThanOrEqual(value: $schema['maximum']);

Check failure on line 242 in src/Metadata/Resource/Factory/ParameterResourceMetadataCollectionFactory.php

View workflow job for this annotation

GitHub Actions / PHPStan (PHP 8.3)

Instantiated class ApiPlatform\Metadata\Resource\Factory\LessThanOrEqual not found.
}

if (isset($schema['pattern'])) {
$assertions[] = new Regex($schema['pattern']);

Check failure on line 246 in src/Metadata/Resource/Factory/ParameterResourceMetadataCollectionFactory.php

View workflow job for this annotation

GitHub Actions / PHPStan (PHP 8.3)

Instantiated class ApiPlatform\Metadata\Resource\Factory\Regex not found.
}

if (isset($schema['maxLength']) || isset($schema['minLength'])) {
$assertions[] = new Length(min: $schema['minLength'] ?? null, max: $schema['maxLength'] ?? null);

Check failure on line 250 in src/Metadata/Resource/Factory/ParameterResourceMetadataCollectionFactory.php

View workflow job for this annotation

GitHub Actions / PHPStan (PHP 8.3)

Instantiated class ApiPlatform\Metadata\Resource\Factory\Length not found.
}

if (isset($schema['minItems']) || isset($schema['maxItems'])) {
$assertions[] = new Count(min: $schema['minItems'] ?? null, max: $schema['maxItems'] ?? null);

Check failure on line 254 in src/Metadata/Resource/Factory/ParameterResourceMetadataCollectionFactory.php

View workflow job for this annotation

GitHub Actions / PHPStan (PHP 8.3)

Instantiated class ApiPlatform\Metadata\Resource\Factory\Count not found.
}

if (isset($schema['multipleOf'])) {
$assertions[] = new DivisibleBy(value: $schema['multipleOf']);

Check failure on line 258 in src/Metadata/Resource/Factory/ParameterResourceMetadataCollectionFactory.php

View workflow job for this annotation

GitHub Actions / PHPStan (PHP 8.3)

Instantiated class ApiPlatform\Metadata\Resource\Factory\DivisibleBy not found.
}

if ($schema['uniqueItems'] ?? false) {
$assertions[] = new Unique();
}

if (isset($schema['enum'])) {
$assertions[] = new Choice(choices: $schema['enum']);
}

if (isset($schema['type']) && 'array' === $schema['type']) {
$assertions[] = new Type(type: 'array');
}

if (!$assertions) {
return $parameter;
}

if (1 === \count($assertions)) {
return $parameter->withConstraints($assertions[0]);
}

return $parameter->withConstraints($assertions);
}

private function getFilterClass(Operation $operation): ?string
{
$stateOptions = $operation->getStateOptions();
Expand Down
121 changes: 90 additions & 31 deletions src/OpenApi/Factory/OpenApiFactory.php
Original file line number Diff line number Diff line change
Expand Up @@ -261,12 +261,26 @@ private function collectPaths(ApiResource $resource, ResourceMetadataCollection
}
}

$entityClass = $this->getFilterClass($operation);
$openapiParameters = $openapiOperation->getParameters();
foreach ($operation->getParameters() ?? [] as $key => $p) {
if (false === $p->getOpenApi()) {
continue;
}

if (($f = $p->getFilter()) && \is_string($f) && $this->filterLocator->has($f)) {
$filter = $this->filterLocator->get($f);
foreach ($filter->getDescription($entityClass) as $name => $description) {
if ($prop = $p->getProperty()) {
$name = str_replace($prop, $key, $name);
}

$openapiParameters[] = $this->getFilterParameter($name, $description, $operation->getShortName(), $f);
}

continue;
}

$in = $p instanceof HeaderParameterInterface ? 'header' : 'query';
$defaultParameter = new Parameter($key, $in, $p->getDescription() ?? "$resourceShortName $key", $p->getRequired() ?? false, false, false, $p->getSchema() ?? ['type' => 'string']);

Expand Down Expand Up @@ -557,57 +571,102 @@ private function getLinks(ResourceMetadataCollection $resourceMetadataCollection
private function getFiltersParameters(CollectionOperationInterface|HttpOperation $operation): array
{
$parameters = [];

$resourceFilters = $operation->getFilters();
$entityClass = $this->getFilterClass($operation);

foreach ($resourceFilters ?? [] as $filterId) {
if (!$this->filterLocator->has($filterId)) {
continue;
}

$filter = $this->filterLocator->get($filterId);
$entityClass = $operation->getClass();
if ($options = $operation->getStateOptions()) {
if ($options instanceof DoctrineOptions && $options->getEntityClass()) {
$entityClass = $options->getEntityClass();
}
foreach ($filter->getDescription($entityClass) as $name => $description) {
$parameters[] = $this->getFilterParameter($name, $description, $operation->getShortName(), $filterId);
}
}

if ($options instanceof DoctrineODMOptions && $options->getDocumentClass()) {
$entityClass = $options->getDocumentClass();
}
return $parameters;
}

private function getFilterClass(HttpOperation $operation): ?string
{
$entityClass = $operation->getClass();
if ($options = $operation->getStateOptions()) {
if ($options instanceof DoctrineOptions && $options->getEntityClass()) {
return $options->getEntityClass();
}

if ($options instanceof DoctrineODMOptions && $options->getDocumentClass()) {
return $options->getDocumentClass();
}
}

foreach ($filter->getDescription($entityClass) as $name => $data) {
$schema = $data['schema'] ?? [];
return $entityClass;
}

if (isset($data['type']) && \in_array($data['type'] ?? null, Type::$builtinTypes, true) && !isset($schema['type'])) {
$schema += $this->getType(new Type($data['type'], false, null, $data['is_collection'] ?? false));
}
private function getFilterParameter(string $name, array $description, string $shortName, string $filter): Parameter
{
if (isset($description['swagger'])) {
trigger_deprecation('api-platform/core', '4.0', \sprintf('Using the "swagger" field of the %s::getDescription() (%s) is deprecated.', $filter, $shortName));
}

if (!isset($schema['type'])) {
$schema['type'] = 'string';
}
if (!isset($description['openapi']) || $description['openapi'] instanceof Parameter) {
$schema = $description['schema'] ?? [];

if (isset($description['type']) && \in_array($description['type'], Type::$builtinTypes, true) && !isset($schema['type'])) {
$schema += $this->jsonSchemaTypeFactory ? $this->jsonSchemaTypeFactory->getType(new Type($description['type'], false, null, $description['is_collection'] ?? false)) : $this->getType(new Type($description['type'], false, null, $description['is_collection'] ?? false));
}

$style = 'array' === ($schema['type'] ?? null) && \in_array(
$data['type'],
[Type::BUILTIN_TYPE_ARRAY, Type::BUILTIN_TYPE_OBJECT],
true
) ? 'deepObject' : 'form';
if (!isset($schema['type'])) {
$schema['type'] = 'string';
}

$parameter = isset($data['openapi']) && $data['openapi'] instanceof Parameter ? $data['openapi'] : new Parameter(in: 'query', name: $name, style: $style, explode: $data['is_collection'] ?? false);
$style = 'array' === ($schema['type'] ?? null) && \in_array(
$description['type'],
[Type::BUILTIN_TYPE_ARRAY, Type::BUILTIN_TYPE_OBJECT],
true
) ? 'deepObject' : 'form';

if ('' === $parameter->getDescription() && ($description = $data['description'] ?? '')) {
$parameter = $parameter->withDescription($description);
}
$parameter = isset($description['openapi']) && $description['openapi'] instanceof Parameter ? $description['openapi'] : new Parameter(in: 'query', name: $name, style: $style, explode: $description['is_collection'] ?? false);

if (false === $parameter->getRequired() && false !== ($required = $data['required'] ?? false)) {
$parameter = $parameter->withRequired($required);
}
if ('' === $parameter->getDescription() && ($str = $description['description'] ?? '')) {
$parameter = $parameter->withDescription($str);
}

$parameters[] = $parameter->withSchema($schema);
if (false === $parameter->getRequired() && false !== ($required = $description['required'] ?? false)) {
$parameter = $parameter->withRequired($required);
}

return $parameter->withSchema($schema);
}

return $parameters;
trigger_deprecation('api-platform/core', '4.0', \sprintf('Not using "%s" on the "openapi" field of the %s::getDescription() (%s) is deprecated.', Parameter::class, $filter, $shortName));
if ($this->jsonSchemaTypeFactory) {
$schema = $description['schema'] ?? (\in_array($description['type'], Type::$builtinTypes, true) ? $this->jsonSchemaTypeFactory->getType(new Type($description['type'], false, null, $description['is_collection'] ?? false), 'openapi') : ['type' => 'string']);
} else {
$schema = $description['schema'] ?? (\in_array($description['type'], Type::$builtinTypes, true) ? $this->getType(new Type($description['type'], false, null, $description['is_collection'] ?? false)) : ['type' => 'string']);
}

return new Parameter(
$name,
'query',
$description['description'] ?? '',
$description['required'] ?? false,
$description['openapi']['deprecated'] ?? false,
$description['openapi']['allowEmptyValue'] ?? true,
$schema,
'array' === $schema['type'] && \in_array(
$description['type'],
[Type::BUILTIN_TYPE_ARRAY, Type::BUILTIN_TYPE_OBJECT],
true
) ? 'deepObject' : 'form',
$description['openapi']['explode'] ?? ('array' === $schema['type']),
$description['openapi']['allowReserved'] ?? false,
$description['openapi']['example'] ?? null,
isset(
$description['openapi']['examples']
) ? new \ArrayObject($description['openapi']['examples']) : null
);
}

private function getPaginationParameters(CollectionOperationInterface|HttpOperation $operation): array
Expand Down
3 changes: 2 additions & 1 deletion src/OpenApi/Tests/Factory/OpenApiFactoryTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -879,7 +879,8 @@ public function testInvoke(): void
'type' => 'string',
'enum' => ['asc', 'desc'],
]),
]
],
deprecated: false
), $filteredPath->getGet());

$paginatedPath = $paths->getPath('/paginated');
Expand Down
36 changes: 36 additions & 0 deletions tests/Fixtures/TestBundle/ApiResource/FilterWithStateOptions.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
<?php

/*
* This file is part of the API Platform project.
*
* (c) Kévin Dunglas <[email protected]>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/

declare(strict_types=1);

namespace ApiPlatform\Tests\Fixtures\TestBundle\ApiResource;

use ApiPlatform\Doctrine\Orm\Filter\DateFilter;
use ApiPlatform\Doctrine\Orm\State\CollectionProvider;
use ApiPlatform\Doctrine\Orm\State\Options;
use ApiPlatform\Metadata\ApiFilter;
use ApiPlatform\Metadata\GetCollection;
use ApiPlatform\Metadata\QueryParameter;
use ApiPlatform\Tests\Fixtures\TestBundle\Entity\FilterWithStateOptionsEntity;

#[GetCollection(
uriTemplate: 'filter_with_state_options',
stateOptions: new Options(entityClass: FilterWithStateOptionsEntity::class),
parameters: ['date' => new QueryParameter(filter: 'filter_with_state_options_date', property: 'dummyDate')],
provider: CollectionProvider::class
)]
#[ApiFilter(DateFilter::class, alias: 'filter_with_state_options_date', properties: ['dummyDate' => DateFilter::EXCLUDE_NULL])]
final class FilterWithStateOptions
{
public function __construct(public readonly string $id, public readonly \DateTimeImmutable $dummyDate, public readonly string $name)
{
}
}
32 changes: 32 additions & 0 deletions tests/Fixtures/TestBundle/Entity/FilterWithStateOptionsEntity.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
<?php

/*
* This file is part of the API Platform project.
*
* (c) Kévin Dunglas <[email protected]>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/

declare(strict_types=1);

namespace ApiPlatform\Tests\Fixtures\TestBundle\Entity;

use Doctrine\ORM\Mapping as ORM;

#[ORM\Entity]
class FilterWithStateOptionsEntity
{
public function __construct(
#[ORM\Column(type: 'integer')]
#[ORM\Id]
#[ORM\GeneratedValue(strategy: 'AUTO')]
public ?int $id = null,
#[ORM\Column(type: 'date_immutable', nullable: true)]
public ?\DateTimeImmutable $dummyDate = null,
#[ORM\Column(type: 'string', nullable: true)]
public ?string $name = null,
) {
}
}
Loading

0 comments on commit 4ef8b6a

Please sign in to comment.