From d52f0c1bd6e12577613e3bc1e9c980697f568d37 Mon Sep 17 00:00:00 2001 From: Jan Skrasek Date: Thu, 25 Feb 2021 18:05:18 +0100 Subject: [PATCH] dbal builder: rewrite WHERE condition for has-many rel into JOIN clause and aggretaion in HAVING This commit changes how we handle has-many relationship query construciton. Until now, every has-many relationship was simply left-joined and filtered via where. The duplication was solved by group-by statement. From now on, the joined table is filtered already in the JOIN clause and the where-part is simply aggretation count of the joined table > 0. This allow moving the where clause to having clause, where it may be combined with other aggregation filtering condition. --- .../Functions/BaseCompareFunction.php | 22 +-- .../Helpers/DbalQueryBuilderHelper.php | 135 ++++++++++++++---- .../Dbal/RelationshipMapperOneHasMany.php | 17 ++- .../Collection/collection.aggregation.phpt | 16 +++ .../Dbal/DbalValueOperatorFunctionTest.phpt | 4 +- 5 files changed, 150 insertions(+), 44 deletions(-) diff --git a/src/Collection/Functions/BaseCompareFunction.php b/src/Collection/Functions/BaseCompareFunction.php index a5418f18..d9a6e766 100644 --- a/src/Collection/Functions/BaseCompareFunction.php +++ b/src/Collection/Functions/BaseCompareFunction.php @@ -46,16 +46,20 @@ public function processQueryBuilderExpression( { assert(count($args) === 2); - $expression = $helper->processPropertyExpr($builder, $args[0]); - - if ($expression->valueNormalizer !== null) { - $cb = $expression->valueNormalizer; - $value = $cb($args[1]); - } else { - $value = $args[1]; - } + return $helper->processPropertyExpr( + $builder, + $args[0], + function (DbalExpressionResult $expression) use ($args): DbalExpressionResult { + if ($expression->valueNormalizer !== null) { + $cb = $expression->valueNormalizer; + $value = $cb($args[1]); + } else { + $value = $args[1]; + } - return $this->evaluateInDb($expression, $value); + return $this->evaluateInDb($expression, $value); + } + ); } diff --git a/src/Collection/Helpers/DbalQueryBuilderHelper.php b/src/Collection/Helpers/DbalQueryBuilderHelper.php index 80ccaee7..8c3f309f 100644 --- a/src/Collection/Helpers/DbalQueryBuilderHelper.php +++ b/src/Collection/Helpers/DbalQueryBuilderHelper.php @@ -6,6 +6,7 @@ use Nette\Utils\Arrays; use Nextras\Dbal\Platforms\Data\Column; use Nextras\Dbal\QueryBuilder\QueryBuilder; +use Nextras\Orm\Collection\Functions\ConjunctionOperatorFunction; use Nextras\Orm\Collection\Functions\IQueryBuilderFunction; use Nextras\Orm\Collection\ICollection; use Nextras\Orm\Entity\Embeddable\EmbeddableContainer; @@ -14,6 +15,7 @@ use Nextras\Orm\Entity\Reflection\PropertyMetadata; use Nextras\Orm\Entity\Reflection\PropertyRelationshipMetadata as Relationship; use Nextras\Orm\Exception\InvalidArgumentException; +use Nextras\Orm\Exception\InvalidStateException; use Nextras\Orm\Exception\NotSupportedException; use Nextras\Orm\Mapper\Dbal\Conventions\IConventions; use Nextras\Orm\Mapper\Dbal\DbalMapper; @@ -82,22 +84,43 @@ public function __construct(IModel $model, IRepository $repository, DbalMapper $ /** + * Processes a property expression represented by either string or collection function array expression. + * + * If you provide expression mutator, the has-many relationship processing for string property expression uses this + * mutator's result for the JOIN clause condition and the returned result value is constructed as an aggregation + * check if the wanted table was actually joined at least once. Not providing mutator processes the expression + * without JOIN clause modification. + * * @param string|mixed[] $expr + * @phpstan-param (callable(DbalExpressionResult): DbalExpressionResult)|null $joinExpressionMutator callback processing expression */ - public function processPropertyExpr(QueryBuilder $builder, $expr): DbalExpressionResult + public function processPropertyExpr( + QueryBuilder $builder, + $expr, + ?callable $joinExpressionMutator = null + ): DbalExpressionResult { if (is_array($expr)) { $function = array_shift($expr); $collectionFunction = $this->getCollectionFunction($function); - return $collectionFunction->processQueryBuilderExpression($this, $builder, $expr); + $expression = $collectionFunction->processQueryBuilderExpression($this, $builder, $expr); + if ($joinExpressionMutator != null) { + return $joinExpressionMutator($expression); + } else { + return $expression; + } } [$tokens, $sourceEntity] = $this->repository->getConditionParser()->parsePropertyExpr($expr); - return $this->processTokens($tokens, $sourceEntity, $builder); + return $this->processTokens($tokens, $sourceEntity, $builder, $joinExpressionMutator); } /** + * Processes an array expression when the first argument at 0 is a collection function name + * and the rest are function argument. If the function name is not present, an implicit + * {@link ConjunctionOperatorFunction} is used. + * * @phpstan-param array|array|list $expr */ public function processFilterFunction(QueryBuilder $builder, array $expr): DbalExpressionResult @@ -211,8 +234,14 @@ private function getCollectionFunction(string $name): IQueryBuilderFunction /** * @param array $tokens * @param class-string<\Nextras\Orm\Entity\IEntity>|null $sourceEntity + * @phpstan-param (callable(DbalExpressionResult): DbalExpressionResult)|null $joinExpressionMutator */ - private function processTokens(array $tokens, ?string $sourceEntity, QueryBuilder $builder): DbalExpressionResult + private function processTokens( + array $tokens, + ?string $sourceEntity, + QueryBuilder $builder, + ?callable $joinExpressionMutator + ): DbalExpressionResult { $lastToken = array_pop($tokens); assert($lastToken !== null); @@ -226,6 +255,8 @@ private function processTokens(array $tokens, ?string $sourceEntity, QueryBuilde $propertyPrefixTokens = ""; $makeDistinct = false; + $joins = []; + foreach ($tokens as $tokenIndex => $token) { $property = $currentEntityMetadata->getProperty($token); if ($property->relationship !== null) { @@ -236,7 +267,7 @@ private function processTokens(array $tokens, ?string $sourceEntity, QueryBuilde $currentMapper, ] = $this->processRelationship( $tokens, - $builder, + $joins, $property, $currentConventions, $currentMapper, @@ -245,10 +276,12 @@ private function processTokens(array $tokens, ?string $sourceEntity, QueryBuilde $tokenIndex, $makeDistinct ); + } elseif ($property->wrapper === EmbeddableContainer::class) { assert($property->args !== null); $currentEntityMetadata = $property->args[EmbeddableContainer::class]['metadata']; $propertyPrefixTokens .= "$token->"; + } else { throw new InvalidArgumentException("Entity {$currentEntityMetadata->className}::\${$token} does not contain a relationship or an embeddable."); } @@ -272,7 +305,7 @@ private function processTokens(array $tokens, ?string $sourceEntity, QueryBuilde $propertyPrefixTokens ); - return new DbalExpressionResult( + $expression = new DbalExpressionResult( ['%column', $column], false, $propertyMetadata, @@ -280,18 +313,54 @@ function ($value) use ($propertyMetadata, $currentConventions) { return $this->normalizeValue($value, $propertyMetadata, $currentConventions); } ); + + if ($makeDistinct && $joinExpressionMutator !== null) { + $joinLast = array_pop($joins); + foreach ($joins as [$target, $on]) { + $builder->joinLeft($target, $on); + } + + /** @var DbalExpressionResult $joinExpressionResult */ + $joinExpressionResult = $joinExpressionMutator($expression); + $joinExpression = array_shift($joinExpressionResult->args); + + $tableName = $currentMapper->getTableName(); + $tableAlias = $joinLast[2]; + $primaryKey = $currentConventions->getStoragePrimaryKey()[0]; + + $builder->joinLeft( + "[$tableName] as [$tableAlias]", + "({$joinLast[1]}) AND $joinExpression", + ...$joinExpressionResult->args + ); + $builder->addGroupBy("%table.%column", $tableAlias, $primaryKey); + return new DbalExpressionResult(['COUNT(%table.%column) > 0', $tableAlias, $primaryKey], true); + + } elseif ($joinExpressionMutator !== null) { + foreach ($joins as [$target, $on]) { + $builder->joinLeft($target, $on); + } + return $joinExpressionMutator($expression); + + } else { + foreach ($joins as [$target, $on]) { + $builder->joinLeft($target, $on); + } + return $expression; + } } /** * @param array $tokens + * @phpstan-param list $joins * @param DbalMapper $currentMapper * @param mixed $token * @return array{string, IConventions, EntityMetadata, DbalMapper} */ private function processRelationship( array $tokens, - QueryBuilder $builder, + array &$joins, PropertyMetadata $property, IConventions $currentConventions, DbalMapper $currentMapper, @@ -305,59 +374,62 @@ private function processRelationship( $targetMapper = $this->model->getRepository($property->relationship->repository)->getMapper(); assert($targetMapper instanceof DbalMapper); - $targetConvetions = $targetMapper->getConventions(); + $targetConventions = $targetMapper->getConventions(); $targetEntityMetadata = $property->relationship->entityMetadata; $relType = $property->relationship->type; - if ($relType === Relationship::ONE_HAS_MANY) { + + if ($relType === Relationship::ONE_HAS_ONE && !$property->relationship->isMain) { assert($property->relationship->property !== null); - $toColumn = $targetConvetions->convertEntityToStorageKey($property->relationship->property); + $toColumn = $targetConventions->convertEntityToStorageKey($property->relationship->property); $fromColumn = $currentConventions->getStoragePrimaryKey()[0]; + + } elseif ($relType === Relationship::ONE_HAS_ONE || $relType === Relationship::MANY_HAS_ONE) { + $toColumn = $targetConventions->getStoragePrimaryKey()[0]; + $fromColumn = $currentConventions->convertEntityToStorageKey($token); + + } elseif ($relType === Relationship::ONE_HAS_MANY) { $makeDistinct = true; - } elseif ($relType === Relationship::ONE_HAS_ONE && !$property->relationship->isMain) { assert($property->relationship->property !== null); - $toColumn = $targetConvetions->convertEntityToStorageKey($property->relationship->property); + $toColumn = $targetConventions->convertEntityToStorageKey($property->relationship->property); $fromColumn = $currentConventions->getStoragePrimaryKey()[0]; } elseif ($relType === Relationship::MANY_HAS_MANY) { - $toColumn = $targetConvetions->getStoragePrimaryKey()[0]; - $fromColumn = $currentConventions->getStoragePrimaryKey()[0]; $makeDistinct = true; - if ($property->relationship->isMain) { - [ - $joinTable, - [$inColumn, $outColumn], - ] = $currentMapper->getManyHasManyParameters($property, $targetMapper); + $toColumn = $targetConventions->getStoragePrimaryKey()[0]; + $fromColumn = $currentConventions->getStoragePrimaryKey()[0]; + if ($property->relationship->isMain) { + [$joinTable, [$inColumn, $outColumn]] = + $currentMapper->getManyHasManyParameters($property, $targetMapper); } else { assert($property->relationship->property !== null); - $sourceProperty = $targetEntityMetadata->getProperty($property->relationship->property); - [ - $joinTable, - [$outColumn, $inColumn], - ] = $targetMapper->getManyHasManyParameters($sourceProperty, $currentMapper); + [$joinTable, [$outColumn, $inColumn]] = + $targetMapper->getManyHasManyParameters($sourceProperty, $currentMapper); } $joinAlias = self::getAlias($joinTable, array_slice($tokens, 0, $tokenIndex)); - $builder->joinLeft("[$joinTable] AS [$joinAlias]", "[$currentAlias.$fromColumn] = [$joinAlias.$inColumn]"); + $joins[] = ["[$joinTable] AS [$joinAlias]", "[$currentAlias.$fromColumn] = [$joinAlias.$inColumn]"]; $currentAlias = $joinAlias; $fromColumn = $outColumn; } else { - $toColumn = $targetConvetions->getStoragePrimaryKey()[0]; - $fromColumn = $currentConventions->convertEntityToStorageKey($token); + throw new InvalidStateException('Should not happen.'); } $targetTable = $targetMapper->getTableName(); $targetAlias = self::getAlias($tokens[$tokenIndex], array_slice($tokens, 0, $tokenIndex)); + $joins[] = [ + "[$targetTable] as [$targetAlias]", + "[$currentAlias.$fromColumn] = [$targetAlias.$toColumn]", + $targetAlias, + ]; - $builder->joinLeft("[$targetTable] AS [$targetAlias]", "[$currentAlias.$fromColumn] = [$targetAlias.$toColumn]"); - - return [$targetAlias, $targetConvetions, $targetEntityMetadata, $targetMapper]; + return [$targetAlias, $targetConventions, $targetEntityMetadata, $targetMapper]; } @@ -399,6 +471,9 @@ private function toColumnExpr( */ private function makeDistinct(QueryBuilder $builder, DbalMapper $mapper): void { + $isGrouped = $builder->getClause('group')[0] ?? null; + if ($isGrouped !== null) return; + $baseTable = $builder->getFromAlias(); if ($this->platformName === 'mssql') { $tableName = $mapper->getTableName(); diff --git a/src/Mapper/Dbal/RelationshipMapperOneHasMany.php b/src/Mapper/Dbal/RelationshipMapperOneHasMany.php index 88111512..095c470f 100644 --- a/src/Mapper/Dbal/RelationshipMapperOneHasMany.php +++ b/src/Mapper/Dbal/RelationshipMapperOneHasMany.php @@ -17,6 +17,7 @@ use Nextras\Orm\Mapper\IRelationshipMapper; use function array_merge; use function array_unique; +use function array_unshift; use function assert; use function count; use function implode; @@ -254,9 +255,9 @@ private function fetchCounts(QueryBuilder $builder, array $values): array $sourceTable = $builder->getFromAlias(); $builder = clone $builder; - $builder->select('%column', "{$sourceTable}.{$this->joinStorageKey}"); if ($builder->hasLimitOffsetClause()) { + $builder->select('%column', "{$sourceTable}.{$this->joinStorageKey}"); $result = $this->processMultiCountResult($builder, $values); } else { @@ -274,11 +275,19 @@ private function fetchCounts(QueryBuilder $builder, array $values): array throw new InvalidStateException('Unable to detect column for count query.'); } - $builder->addSelect('COUNT(DISTINCT %column) AS [count]', $targetColumn); + $builder->addSelect('%column AS [count]', $targetColumn); $builder->andWhere('%column IN %any', "{$sourceTable}.{$this->joinStorageKey}", $values); - $builder->groupBy('%column', "{$sourceTable}.{$this->joinStorageKey}"); $builder->orderBy(null); - $result = $this->connection->queryArgs($builder->getQuerySql(), $builder->getQueryParameters()); + + $boxingBuilder = $this->connection->createQueryBuilder(); + $boxingBuilder->addSelect('%column, COUNT(DISTINCT [count]) as [count]', $this->joinStorageKey); + $boxingBuilder->groupBy('%column', $this->joinStorageKey); + + $args = $builder->getQueryParameters(); + array_unshift($args, $builder->getQuerySql()); + $boxingBuilder->from('(%ex)', 'temp', $args); + + $result = $this->connection->queryByQueryBuilder($boxingBuilder); } $counts = []; diff --git a/tests/cases/integration/Collection/collection.aggregation.phpt b/tests/cases/integration/Collection/collection.aggregation.phpt index d4a3c251..7a5b2dcf 100644 --- a/tests/cases/integration/Collection/collection.aggregation.phpt +++ b/tests/cases/integration/Collection/collection.aggregation.phpt @@ -169,6 +169,22 @@ class CollectionAggregationTest extends DataTestCase ->fetchPairs(null, 'id'); Assert::same([1], $userIds); } + + + public function testAggerationWithNoAggregateCondition(): void + { + $users = $this->orm->authors->findBy([ + ICollection::OR, + ['books->title' => 'Book 1'], + [ + CompareSmallerThanEqualsFunction::class, + [CountAggregateFunction::class, 'tagFollowers->tag'], + 2, + ], + ]); + Assert::same(2, $users->count()); + Assert::same(2, $users->countStored()); + } } diff --git a/tests/cases/unit/Mapper/Dbal/DbalValueOperatorFunctionTest.phpt b/tests/cases/unit/Mapper/Dbal/DbalValueOperatorFunctionTest.phpt index 67b99efa..f50627bc 100644 --- a/tests/cases/unit/Mapper/Dbal/DbalValueOperatorFunctionTest.phpt +++ b/tests/cases/unit/Mapper/Dbal/DbalValueOperatorFunctionTest.phpt @@ -33,7 +33,9 @@ class DbalValueOperatorFunctionTest extends TestCase $expressionResult = new DbalExpressionResult(['%column', 'books.id']); $helper = Mockery::mock(DbalQueryBuilderHelper::class); - $helper->shouldReceive('processPropertyExpr')->once()->andReturn($expressionResult); + $helper->shouldReceive('processPropertyExpr')->once()->andReturnUsing(function ($_, $__, $expressionResultCb) use ($expressionResult) { + return $expressionResultCb($expressionResult); + }); $builder = Mockery::mock(QueryBuilder::class); $builder->shouldReceive('getFromAlias')->andReturn('books');