-
-
Notifications
You must be signed in to change notification settings - Fork 877
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(doctrine): doctrine filters like laravel eloquent filters
- Loading branch information
1 parent
b458b55
commit aef306c
Showing
39 changed files
with
1,724 additions
and
60 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
25 changes: 25 additions & 0 deletions
25
src/Doctrine/Common/Filter/ManagerRegistryAwareInterface.php
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,25 @@ | ||
<?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\Doctrine\Common\Filter; | ||
|
||
use Doctrine\Persistence\ManagerRegistry; | ||
|
||
interface ManagerRegistryAwareInterface | ||
{ | ||
public function hasManagerRegistry(): bool; | ||
|
||
public function getManagerRegistry(): ManagerRegistry; | ||
|
||
public function setManagerRegistry(ManagerRegistry $managerRegistry): void; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -13,9 +13,11 @@ | |
|
||
namespace ApiPlatform\Doctrine\Odm\Filter; | ||
|
||
use ApiPlatform\Doctrine\Common\Filter\ManagerRegistryAwareInterface; | ||
use ApiPlatform\Doctrine\Common\Filter\PropertyAwareFilterInterface; | ||
use ApiPlatform\Doctrine\Common\PropertyHelperTrait; | ||
use ApiPlatform\Doctrine\Odm\PropertyHelperTrait as MongoDbOdmPropertyHelperTrait; | ||
use ApiPlatform\Metadata\Exception\RuntimeException; | ||
use ApiPlatform\Metadata\Operation; | ||
use Doctrine\ODM\MongoDB\Aggregation\Builder; | ||
use Doctrine\Persistence\ManagerRegistry; | ||
|
@@ -30,14 +32,18 @@ | |
* | ||
* @author Alan Poulain <[email protected]> | ||
*/ | ||
abstract class AbstractFilter implements FilterInterface, PropertyAwareFilterInterface | ||
abstract class AbstractFilter implements FilterInterface, PropertyAwareFilterInterface, ManagerRegistryAwareInterface | ||
{ | ||
use MongoDbOdmPropertyHelperTrait; | ||
use PropertyHelperTrait; | ||
protected LoggerInterface $logger; | ||
|
||
public function __construct(protected ManagerRegistry $managerRegistry, ?LoggerInterface $logger = null, protected ?array $properties = null, protected ?NameConverterInterface $nameConverter = null) | ||
{ | ||
public function __construct( | ||
protected ?ManagerRegistry $managerRegistry = null, | ||
?LoggerInterface $logger = null, | ||
protected ?array $properties = null, | ||
protected ?NameConverterInterface $nameConverter = null, | ||
) { | ||
$this->logger = $logger ?? new NullLogger(); | ||
} | ||
|
||
|
@@ -56,18 +62,35 @@ public function apply(Builder $aggregationBuilder, string $resourceClass, ?Opera | |
*/ | ||
abstract protected function filterProperty(string $property, $value, Builder $aggregationBuilder, string $resourceClass, ?Operation $operation = null, array &$context = []): void; | ||
|
||
protected function getManagerRegistry(): ManagerRegistry | ||
public function hasManagerRegistry(): bool | ||
{ | ||
return $this->managerRegistry instanceof ManagerRegistry; | ||
} | ||
|
||
public function getManagerRegistry(): ManagerRegistry | ||
{ | ||
if (!$this->hasManagerRegistry()) { | ||
throw new RuntimeException('ManagerRegistry must be initialized before accessing it.'); | ||
} | ||
|
||
return $this->managerRegistry; | ||
} | ||
|
||
protected function getProperties(): ?array | ||
public function setManagerRegistry(ManagerRegistry $managerRegistry): void | ||
{ | ||
$this->managerRegistry = $managerRegistry; | ||
} | ||
|
||
/** | ||
* @return array<string, mixed>|null | ||
*/ | ||
public function getProperties(): ?array | ||
{ | ||
return $this->properties; | ||
} | ||
|
||
/** | ||
* @param string[] $properties | ||
* @param array<string, mixed> $properties | ||
*/ | ||
public function setProperties(array $properties): void | ||
{ | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -14,7 +14,9 @@ | |
namespace ApiPlatform\Doctrine\Odm\Filter; | ||
|
||
use ApiPlatform\Doctrine\Common\Filter\BooleanFilterTrait; | ||
use ApiPlatform\Metadata\JsonSchemaFilterInterface; | ||
use ApiPlatform\Metadata\Operation; | ||
use ApiPlatform\Metadata\Parameter; | ||
use Doctrine\ODM\MongoDB\Aggregation\Builder; | ||
use Doctrine\ODM\MongoDB\Types\Type as MongoDbType; | ||
|
||
|
@@ -104,7 +106,7 @@ | |
* @author Teoh Han Hui <[email protected]> | ||
* @author Alan Poulain <[email protected]> | ||
*/ | ||
final class BooleanFilter extends AbstractFilter | ||
final class BooleanFilter extends AbstractFilter implements JsonSchemaFilterInterface | ||
{ | ||
use BooleanFilterTrait; | ||
|
||
|
@@ -139,4 +141,12 @@ protected function filterProperty(string $property, $value, Builder $aggregation | |
|
||
$aggregationBuilder->match()->field($matchField)->equals($value); | ||
} | ||
|
||
/** | ||
* @return array<string, string> | ||
*/ | ||
public function getSchema(Parameter $parameter): array | ||
{ | ||
return ['type' => 'boolean']; | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -16,7 +16,12 @@ | |
use ApiPlatform\Doctrine\Common\Filter\DateFilterInterface; | ||
use ApiPlatform\Doctrine\Common\Filter\DateFilterTrait; | ||
use ApiPlatform\Metadata\Exception\InvalidArgumentException; | ||
use ApiPlatform\Metadata\JsonSchemaFilterInterface; | ||
use ApiPlatform\Metadata\OpenApiParameterFilterInterface; | ||
use ApiPlatform\Metadata\Operation; | ||
use ApiPlatform\Metadata\Parameter; | ||
use ApiPlatform\Metadata\QueryParameter; | ||
use ApiPlatform\OpenApi\Model\Parameter as OpenApiParameter; | ||
use Doctrine\ODM\MongoDB\Aggregation\Builder; | ||
use Doctrine\ODM\MongoDB\Types\Type as MongoDbType; | ||
|
||
|
@@ -117,7 +122,7 @@ | |
* @author Théo FIDRY <[email protected]> | ||
* @author Alan Poulain <[email protected]> | ||
*/ | ||
final class DateFilter extends AbstractFilter implements DateFilterInterface | ||
final class DateFilter extends AbstractFilter implements DateFilterInterface, JsonSchemaFilterInterface, OpenApiParameterFilterInterface | ||
{ | ||
use DateFilterTrait; | ||
|
||
|
@@ -237,4 +242,25 @@ private function addMatch(Builder $aggregationBuilder, string $field, string $op | |
|
||
$aggregationBuilder->match()->addAnd($aggregationBuilder->matchExpr()->field($field)->operator($operatorValue[$operator], $value)); | ||
} | ||
|
||
/** | ||
* @return array<string, string> | ||
*/ | ||
public function getSchema(Parameter $parameter): array | ||
{ | ||
return ['type' => 'date']; | ||
} | ||
|
||
public function getOpenApiParameters(Parameter $parameter): OpenApiParameter|array|null | ||
{ | ||
$in = $parameter instanceof QueryParameter ? 'query' : 'header'; | ||
$key = $parameter->getKey(); | ||
|
||
return [ | ||
new OpenApiParameter(name: $key.'[after]', in: $in), | ||
new OpenApiParameter(name: $key.'[before]', in: $in), | ||
new OpenApiParameter(name: $key.'[strictly_after]', in: $in), | ||
new OpenApiParameter(name: $key.'[strictly_before]', in: $in), | ||
]; | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -15,7 +15,12 @@ | |
|
||
use ApiPlatform\Doctrine\Common\Filter\ExistsFilterInterface; | ||
use ApiPlatform\Doctrine\Common\Filter\ExistsFilterTrait; | ||
use ApiPlatform\Metadata\JsonSchemaFilterInterface; | ||
use ApiPlatform\Metadata\OpenApiParameterFilterInterface; | ||
use ApiPlatform\Metadata\Operation; | ||
use ApiPlatform\Metadata\Parameter; | ||
use ApiPlatform\Metadata\QueryParameter; | ||
use ApiPlatform\OpenApi\Model\Parameter as OpenApiParameter; | ||
use Doctrine\ODM\MongoDB\Aggregation\Builder; | ||
use Doctrine\ODM\MongoDB\Mapping\ClassMetadata; | ||
use Doctrine\Persistence\ManagerRegistry; | ||
|
@@ -107,12 +112,16 @@ | |
* @author Teoh Han Hui <[email protected]> | ||
* @author Alan Poulain <[email protected]> | ||
*/ | ||
final class ExistsFilter extends AbstractFilter implements ExistsFilterInterface | ||
final class ExistsFilter extends AbstractFilter implements ExistsFilterInterface, JsonSchemaFilterInterface, OpenApiParameterFilterInterface | ||
{ | ||
use ExistsFilterTrait; | ||
|
||
public function __construct(ManagerRegistry $managerRegistry, ?LoggerInterface $logger = null, ?array $properties = null, string $existsParameterName = self::QUERY_PARAMETER_KEY, ?NameConverterInterface $nameConverter = null) | ||
public function __construct(?ManagerRegistry $managerRegistry = null, ?LoggerInterface $logger = null, ?array $properties = null, string $existsParameterName = self::QUERY_PARAMETER_KEY, ?NameConverterInterface $nameConverter = null) | ||
{ | ||
if (\is_array($properties) && \is_int(key($properties))) { | ||
$properties = array_flip($properties); | ||
} | ||
|
||
parent::__construct($managerRegistry, $logger, $properties, $nameConverter); | ||
|
||
$this->existsParameterName = $existsParameterName; | ||
|
@@ -123,6 +132,12 @@ public function __construct(ManagerRegistry $managerRegistry, ?LoggerInterface $ | |
*/ | ||
public function apply(Builder $aggregationBuilder, string $resourceClass, ?Operation $operation = null, array &$context = []): void | ||
{ | ||
$parameter = $context['parameter'] ?? null; | ||
|
||
if (null !== ($value = $context['filters'][$parameter?->getProperty()] ?? null)) { | ||
$this->filterProperty($this->denormalizePropertyName($parameter->getProperty()), $value, $aggregationBuilder, $resourceClass, $operation, $context); | ||
} | ||
|
||
foreach ($context['filters'][$this->existsParameterName] ?? [] as $property => $value) { | ||
$this->filterProperty($this->denormalizePropertyName($property), $value, $aggregationBuilder, $resourceClass, $operation, $context); | ||
} | ||
|
@@ -167,4 +182,20 @@ protected function isNullableField(string $property, string $resourceClass): boo | |
|
||
return $metadata instanceof ClassMetadata && $metadata->hasField($field) ? $metadata->isNullable($field) : false; | ||
} | ||
|
||
public function getSchema(Parameter $parameter): array | ||
{ | ||
return ['type' => 'boolean']; | ||
} | ||
|
||
public function getOpenApiParameters(Parameter $parameter): OpenApiParameter|array|null | ||
{ | ||
$property = $parameter->getProperty(); | ||
$in = $parameter instanceof QueryParameter ? 'query' : 'header'; | ||
|
||
return new OpenApiParameter( | ||
name: "exists[$property]", | ||
in: $in | ||
); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -14,7 +14,9 @@ | |
namespace ApiPlatform\Doctrine\Odm\Filter; | ||
|
||
use ApiPlatform\Doctrine\Common\Filter\NumericFilterTrait; | ||
use ApiPlatform\Metadata\JsonSchemaFilterInterface; | ||
use ApiPlatform\Metadata\Operation; | ||
use ApiPlatform\Metadata\Parameter; | ||
use Doctrine\ODM\MongoDB\Aggregation\Builder; | ||
use Doctrine\ODM\MongoDB\Types\Type as MongoDbType; | ||
|
||
|
@@ -104,7 +106,7 @@ | |
* @author Teoh Han Hui <[email protected]> | ||
* @author Alan Poulain <[email protected]> | ||
*/ | ||
final class NumericFilter extends AbstractFilter | ||
final class NumericFilter extends AbstractFilter implements JsonSchemaFilterInterface | ||
{ | ||
use NumericFilterTrait; | ||
|
||
|
@@ -163,4 +165,9 @@ protected function getType(?string $doctrineType = null): string | |
|
||
return 'int'; | ||
} | ||
|
||
public function getSchema(Parameter $parameter): array | ||
{ | ||
return ['type' => 'numeric']; | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -15,7 +15,11 @@ | |
|
||
use ApiPlatform\Doctrine\Common\Filter\RangeFilterInterface; | ||
use ApiPlatform\Doctrine\Common\Filter\RangeFilterTrait; | ||
use ApiPlatform\Metadata\OpenApiParameterFilterInterface; | ||
use ApiPlatform\Metadata\Operation; | ||
use ApiPlatform\Metadata\Parameter; | ||
use ApiPlatform\Metadata\QueryParameter; | ||
use ApiPlatform\OpenApi\Model\Parameter as OpenApiParameter; | ||
use Doctrine\ODM\MongoDB\Aggregation\Builder; | ||
|
||
/** | ||
|
@@ -104,7 +108,7 @@ | |
* @author Lee Siong Chan <[email protected]> | ||
* @author Alan Poulain <[email protected]> | ||
*/ | ||
final class RangeFilter extends AbstractFilter implements RangeFilterInterface | ||
final class RangeFilter extends AbstractFilter implements RangeFilterInterface, OpenApiParameterFilterInterface | ||
{ | ||
use RangeFilterTrait; | ||
|
||
|
@@ -204,4 +208,17 @@ protected function addMatch(Builder $aggregationBuilder, string $field, string $ | |
break; | ||
} | ||
} | ||
|
||
public function getOpenApiParameters(Parameter $parameter): OpenApiParameter|array|null | ||
{ | ||
$in = $parameter instanceof QueryParameter ? 'query' : 'header'; | ||
$key = $parameter->getKey(); | ||
|
||
return [ | ||
new OpenApiParameter(name: $key.'[gt]', in: $in), | ||
new OpenApiParameter(name: $key.'[lt]', in: $in), | ||
new OpenApiParameter(name: $key.'[gte]', in: $in), | ||
new OpenApiParameter(name: $key.'[lte]', in: $in), | ||
]; | ||
} | ||
} |
Oops, something went wrong.