Skip to content

Commit

Permalink
dbal builder: rewrite WHERE condition for has-many rel into JOIN clau…
Browse files Browse the repository at this point in the history
…se and aggretaion in HAVING (#496)

This commit changes how we handle has-many relationship query construction.
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 aggregation 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.
  • Loading branch information
hrach authored Mar 26, 2021
1 parent 9b5ed59 commit 880b998
Show file tree
Hide file tree
Showing 5 changed files with 150 additions and 44 deletions.
22 changes: 13 additions & 9 deletions src/Collection/Functions/BaseCompareFunction.php
Original file line number Diff line number Diff line change
Expand Up @@ -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);
}
);
}


Expand Down
135 changes: 105 additions & 30 deletions src/Collection/Helpers/DbalQueryBuilderHelper.php
Original file line number Diff line number Diff line change
Expand Up @@ -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;
Expand All @@ -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;
Expand Down Expand Up @@ -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<string, mixed>|array<int|string, mixed>|list<mixed> $expr
*/
public function processFilterFunction(QueryBuilder $builder, array $expr): DbalExpressionResult
Expand Down Expand Up @@ -211,8 +234,14 @@ private function getCollectionFunction(string $name): IQueryBuilderFunction
/**
* @param array<string> $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);
Expand All @@ -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) {
Expand All @@ -236,7 +267,7 @@ private function processTokens(array $tokens, ?string $sourceEntity, QueryBuilde
$currentMapper,
] = $this->processRelationship(
$tokens,
$builder,
$joins,
$property,
$currentConventions,
$currentMapper,
Expand All @@ -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.");
}
Expand All @@ -272,26 +305,62 @@ private function processTokens(array $tokens, ?string $sourceEntity, QueryBuilde
$propertyPrefixTokens
);

return new DbalExpressionResult(
$expression = new DbalExpressionResult(
['%column', $column],
false,
$propertyMetadata,
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<string> $tokens
* @phpstan-param list<array{string, string, string}> $joins
* @param DbalMapper<IEntity> $currentMapper
* @param mixed $token
* @return array{string, IConventions, EntityMetadata, DbalMapper<IEntity>}
*/
private function processRelationship(
array $tokens,
QueryBuilder $builder,
array &$joins,
PropertyMetadata $property,
IConventions $currentConventions,
DbalMapper $currentMapper,
Expand All @@ -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];
}


Expand Down Expand Up @@ -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();
Expand Down
17 changes: 13 additions & 4 deletions src/Mapper/Dbal/RelationshipMapperOneHasMany.php
Original file line number Diff line number Diff line change
Expand Up @@ -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;
Expand Down Expand Up @@ -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 {
Expand All @@ -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 = [];
Expand Down
16 changes: 16 additions & 0 deletions tests/cases/integration/Collection/collection.aggregation.phpt
Original file line number Diff line number Diff line change
Expand Up @@ -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());
}
}


Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -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');
Expand Down

0 comments on commit 880b998

Please sign in to comment.