Fix paginator issues when ordering by a joined column from a to-many association.
Manual merge testcase from #1351
This commit is contained in:
parent
54d7efd92c
commit
ff75a3ad49
6 changed files with 420 additions and 121 deletions
|
@ -13,13 +13,23 @@
|
||||||
|
|
||||||
namespace Doctrine\ORM\Tools\Pagination;
|
namespace Doctrine\ORM\Tools\Pagination;
|
||||||
|
|
||||||
|
use Doctrine\DBAL\Platforms\DB2Platform;
|
||||||
|
use Doctrine\DBAL\Platforms\OraclePlatform;
|
||||||
|
use Doctrine\DBAL\Platforms\PostgreSqlPlatform;
|
||||||
|
use Doctrine\DBAL\Platforms\SQLAnywherePlatform;
|
||||||
|
use Doctrine\DBAL\Platforms\SQLServerPlatform;
|
||||||
use Doctrine\ORM\Query\AST\ArithmeticExpression;
|
use Doctrine\ORM\Query\AST\ArithmeticExpression;
|
||||||
|
use Doctrine\ORM\Query\AST\ArithmeticFactor;
|
||||||
use Doctrine\ORM\Query\AST\ArithmeticTerm;
|
use Doctrine\ORM\Query\AST\ArithmeticTerm;
|
||||||
|
use Doctrine\ORM\Query\AST\Literal;
|
||||||
use Doctrine\ORM\Query\AST\OrderByClause;
|
use Doctrine\ORM\Query\AST\OrderByClause;
|
||||||
|
use Doctrine\ORM\Query\AST\OrderByItem;
|
||||||
use Doctrine\ORM\Query\AST\PartialObjectExpression;
|
use Doctrine\ORM\Query\AST\PartialObjectExpression;
|
||||||
use Doctrine\ORM\Query\AST\PathExpression;
|
use Doctrine\ORM\Query\AST\PathExpression;
|
||||||
use Doctrine\ORM\Query\AST\SelectExpression;
|
use Doctrine\ORM\Query\AST\SelectExpression;
|
||||||
|
use Doctrine\ORM\Query\AST\SimpleArithmeticExpression;
|
||||||
use Doctrine\ORM\Query\Expr\OrderBy;
|
use Doctrine\ORM\Query\Expr\OrderBy;
|
||||||
|
use Doctrine\ORM\Query\Expr\Select;
|
||||||
use Doctrine\ORM\Query\SqlWalker;
|
use Doctrine\ORM\Query\SqlWalker;
|
||||||
use Doctrine\ORM\Query\AST\SelectStatement;
|
use Doctrine\ORM\Query\AST\SelectStatement;
|
||||||
|
|
||||||
|
@ -39,17 +49,17 @@ class LimitSubqueryOutputWalker extends SqlWalker
|
||||||
/**
|
/**
|
||||||
* @var \Doctrine\DBAL\Platforms\AbstractPlatform
|
* @var \Doctrine\DBAL\Platforms\AbstractPlatform
|
||||||
*/
|
*/
|
||||||
private $platform;
|
private $_platform;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @var \Doctrine\ORM\Query\ResultSetMapping
|
* @var \Doctrine\ORM\Query\ResultSetMapping
|
||||||
*/
|
*/
|
||||||
private $rsm;
|
private $_rsm;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @var array
|
* @var array
|
||||||
*/
|
*/
|
||||||
private $queryComponents;
|
private $_queryComponents;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @var int
|
* @var int
|
||||||
|
@ -64,19 +74,19 @@ class LimitSubqueryOutputWalker extends SqlWalker
|
||||||
/**
|
/**
|
||||||
* @var \Doctrine\ORM\EntityManager
|
* @var \Doctrine\ORM\EntityManager
|
||||||
*/
|
*/
|
||||||
private $em;
|
private $_em;
|
||||||
|
|
||||||
/**
|
|
||||||
* @var array
|
|
||||||
*/
|
|
||||||
private $orderByPathExpressions = [];
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The quote strategy.
|
* The quote strategy.
|
||||||
*
|
*
|
||||||
* @var \Doctrine\ORM\Mapping\QuoteStrategy
|
* @var \Doctrine\ORM\Mapping\QuoteStrategy
|
||||||
*/
|
*/
|
||||||
private $quoteStrategy;
|
private $_quoteStrategy;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var array
|
||||||
|
*/
|
||||||
|
private $orderByPathExpressions = [];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor.
|
* Constructor.
|
||||||
|
@ -91,32 +101,143 @@ class LimitSubqueryOutputWalker extends SqlWalker
|
||||||
*/
|
*/
|
||||||
public function __construct($query, $parserResult, array $queryComponents)
|
public function __construct($query, $parserResult, array $queryComponents)
|
||||||
{
|
{
|
||||||
$this->platform = $query->getEntityManager()->getConnection()->getDatabasePlatform();
|
$this->_platform = $query->getEntityManager()->getConnection()->getDatabasePlatform();
|
||||||
$this->rsm = $parserResult->getResultSetMapping();
|
$this->_rsm = $parserResult->getResultSetMapping();
|
||||||
$this->queryComponents = $queryComponents;
|
$this->_queryComponents = $queryComponents;
|
||||||
|
|
||||||
// Reset limit and offset
|
// Reset limit and offset
|
||||||
$this->firstResult = $query->getFirstResult();
|
$this->firstResult = $query->getFirstResult();
|
||||||
$this->maxResults = $query->getMaxResults();
|
$this->maxResults = $query->getMaxResults();
|
||||||
$query->setFirstResult(null)->setMaxResults(null);
|
$query->setFirstResult(null)->setMaxResults(null);
|
||||||
|
|
||||||
$this->em = $query->getEntityManager();
|
$this->_em = $query->getEntityManager();
|
||||||
$this->quoteStrategy = $this->em->getConfiguration()->getQuoteStrategy();
|
$this->_quoteStrategy = $this->_em->getConfiguration()->getQuoteStrategy();
|
||||||
|
|
||||||
parent::__construct($query, $parserResult, $queryComponents);
|
parent::__construct($query, $parserResult, $queryComponents);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if the platform supports the ROW_NUMBER window function.
|
||||||
|
*
|
||||||
|
* @return bool
|
||||||
|
*/
|
||||||
|
private function platformSupportsRowNumber() {
|
||||||
|
return $this->_platform instanceof PostgreSqlPlatform
|
||||||
|
|| $this->_platform instanceof SQLServerPlatform
|
||||||
|
|| $this->_platform instanceof OraclePlatform
|
||||||
|
|| $this->_platform instanceof SQLAnywherePlatform
|
||||||
|
|| $this->_platform instanceof DB2Platform
|
||||||
|
|| (method_exists($this->_platform, "supportsRowNumberFunction")
|
||||||
|
&& $this->_platform->supportsRowNumberFunction());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Rebuilds a select statement's order by clause for use in a
|
||||||
|
* ROW_NUMBER() OVER() expression.
|
||||||
|
*
|
||||||
|
* @param SelectStatement $AST
|
||||||
|
*/
|
||||||
|
private function rebuildOrderByForRowNumber(SelectStatement $AST) {
|
||||||
|
$orderByClause = $AST->orderByClause;
|
||||||
|
$selectAliasToExpressionMap = [];
|
||||||
|
foreach($AST->selectClause->selectExpressions as $selectExpression) {
|
||||||
|
$selectAliasToExpressionMap[$selectExpression->fieldIdentificationVariable] = $selectExpression->expression;
|
||||||
|
}
|
||||||
|
foreach($orderByClause->orderByItems as $orderByItem) {
|
||||||
|
if (is_string($orderByItem->expression) && isset($selectAliasToExpressionMap[$orderByItem->expression])) {
|
||||||
|
$orderByItem->expression = $selectAliasToExpressionMap[$orderByItem->expression];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$func = new RowNumberOverFunction("dctrn_rownum");
|
||||||
|
$func->orderByClause = $AST->orderByClause;
|
||||||
|
$AST->selectClause->selectExpressions[] = new SelectExpression($func, "dctrn_rownum", true);
|
||||||
|
|
||||||
|
// No need for an order by clause, we'll order by rownum in the outer query.
|
||||||
|
$AST->orderByClause = null;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Walks down a SelectStatement AST node, wrapping it in a SELECT DISTINCT.
|
* Walks down a SelectStatement AST node, wrapping it in a SELECT DISTINCT.
|
||||||
*
|
*
|
||||||
* @param SelectStatement $AST
|
* @param SelectStatement $AST
|
||||||
|
*
|
||||||
|
* @return string
|
||||||
|
*
|
||||||
|
* @throws \RuntimeException
|
||||||
|
*/
|
||||||
|
public function walkSelectStatement(SelectStatement $AST)
|
||||||
|
{
|
||||||
|
if ($this->platformSupportsRowNumber()) {
|
||||||
|
return $this->walkSelectStatementWithRowNumber($AST);
|
||||||
|
}
|
||||||
|
return $this->walkSelectStatementWithoutRowNumber($AST);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Walks down a SelectStatement AST node, wrapping it in a SELECT DISTINCT.
|
||||||
|
* This method is for use with platforms which support ROW_NUMBER.
|
||||||
|
*
|
||||||
|
* @param SelectStatement $AST
|
||||||
|
*
|
||||||
|
* @return string
|
||||||
|
*
|
||||||
|
* @throws \RuntimeException
|
||||||
|
*/
|
||||||
|
public function walkSelectStatementWithRowNumber(SelectStatement $AST)
|
||||||
|
{
|
||||||
|
$hasOrderBy = false;
|
||||||
|
$outerOrderBy = " ORDER BY dctrn_minrownum ASC";
|
||||||
|
$orderGroupBy = '';
|
||||||
|
if ($AST->orderByClause instanceof OrderByClause) {
|
||||||
|
$hasOrderBy = true;
|
||||||
|
$this->rebuildOrderByForRowNumber($AST);
|
||||||
|
}
|
||||||
|
|
||||||
|
$innerSql = $this->getInnerSQL($AST);
|
||||||
|
|
||||||
|
$sqlIdentifier = $this->getSQLIdentifier($AST);
|
||||||
|
|
||||||
|
if ($hasOrderBy) {
|
||||||
|
$orderGroupBy = " GROUP BY " . implode(', ', $sqlIdentifier);
|
||||||
|
$sqlIdentifier[] = "MIN(" . $this->walkResultVariable("dctrn_rownum") . ") AS dctrn_minrownum";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Build the counter query
|
||||||
|
$sql = sprintf('SELECT DISTINCT %s FROM (%s) dctrn_result',
|
||||||
|
implode(', ', $sqlIdentifier), $innerSql);
|
||||||
|
|
||||||
|
if ($hasOrderBy) {
|
||||||
|
$sql .= $orderGroupBy . $outerOrderBy;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Apply the limit and offset.
|
||||||
|
$sql = $this->_platform->modifyLimitQuery(
|
||||||
|
$sql, $this->maxResults, $this->firstResult
|
||||||
|
);
|
||||||
|
|
||||||
|
// Add the columns to the ResultSetMapping. It's not really nice but
|
||||||
|
// it works. Preferably I'd clear the RSM or simply create a new one
|
||||||
|
// but that is not possible from inside the output walker, so we dirty
|
||||||
|
// up the one we have.
|
||||||
|
foreach ($sqlIdentifier as $property => $alias) {
|
||||||
|
$this->_rsm->addScalarResult($alias, $property);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $sql;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Walks down a SelectStatement AST node, wrapping it in a SELECT DISTINCT.
|
||||||
|
* This method is for platforms which DO NOT support ROW_NUMBER.
|
||||||
|
*
|
||||||
|
* @param SelectStatement $AST
|
||||||
* @param bool $addMissingItemsFromOrderByToSelect
|
* @param bool $addMissingItemsFromOrderByToSelect
|
||||||
*
|
*
|
||||||
* @return string
|
* @return string
|
||||||
*
|
*
|
||||||
* @throws \RuntimeException
|
* @throws \RuntimeException
|
||||||
*/
|
*/
|
||||||
public function walkSelectStatement(SelectStatement $AST, $addMissingItemsFromOrderByToSelect = true)
|
public function walkSelectStatementWithoutRowNumber(SelectStatement $AST, $addMissingItemsFromOrderByToSelect = true)
|
||||||
{
|
{
|
||||||
// We don't want to call this recursively!
|
// We don't want to call this recursively!
|
||||||
if ($AST->orderByClause instanceof OrderByClause && $addMissingItemsFromOrderByToSelect) {
|
if ($AST->orderByClause instanceof OrderByClause && $addMissingItemsFromOrderByToSelect) {
|
||||||
|
@ -131,74 +252,9 @@ class LimitSubqueryOutputWalker extends SqlWalker
|
||||||
$orderByClause = $AST->orderByClause;
|
$orderByClause = $AST->orderByClause;
|
||||||
$AST->orderByClause = null;
|
$AST->orderByClause = null;
|
||||||
|
|
||||||
// Set every select expression as visible(hidden = false) to
|
$innerSql = $this->getInnerSQL($AST);
|
||||||
// make $AST have scalar mappings properly - this is relevant for referencing selected
|
|
||||||
// fields from outside the subquery, for example in the ORDER BY segment
|
|
||||||
$hiddens = array();
|
|
||||||
|
|
||||||
foreach ($AST->selectClause->selectExpressions as $idx => $expr) {
|
$sqlIdentifier = $this->getSQLIdentifier($AST);
|
||||||
$hiddens[$idx] = $expr->hiddenAliasResultVariable;
|
|
||||||
$expr->hiddenAliasResultVariable = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
$innerSql = parent::walkSelectStatement($AST);
|
|
||||||
|
|
||||||
// Restore orderByClause
|
|
||||||
$AST->orderByClause = $orderByClause;
|
|
||||||
|
|
||||||
// Restore hiddens
|
|
||||||
foreach ($AST->selectClause->selectExpressions as $idx => $expr) {
|
|
||||||
$expr->hiddenAliasResultVariable = $hiddens[$idx];
|
|
||||||
}
|
|
||||||
|
|
||||||
// Find out the SQL alias of the identifier column of the root entity.
|
|
||||||
// It may be possible to make this work with multiple root entities but that
|
|
||||||
// would probably require issuing multiple queries or doing a UNION SELECT.
|
|
||||||
// So for now, it's not supported.
|
|
||||||
|
|
||||||
// Get the root entity and alias from the AST fromClause.
|
|
||||||
$from = $AST->fromClause->identificationVariableDeclarations;
|
|
||||||
if (count($from) !== 1) {
|
|
||||||
throw new \RuntimeException("Cannot count query which selects two FROM components, cannot make distinction");
|
|
||||||
}
|
|
||||||
|
|
||||||
$fromRoot = reset($from);
|
|
||||||
$rootAlias = $fromRoot->rangeVariableDeclaration->aliasIdentificationVariable;
|
|
||||||
$rootClass = $this->queryComponents[$rootAlias]['metadata'];
|
|
||||||
$rootIdentifier = $rootClass->identifier;
|
|
||||||
|
|
||||||
// For every identifier, find out the SQL alias by combing through the ResultSetMapping
|
|
||||||
$sqlIdentifier = array();
|
|
||||||
foreach ($rootIdentifier as $property) {
|
|
||||||
if (isset($rootClass->fieldMappings[$property])) {
|
|
||||||
foreach (array_keys($this->rsm->fieldMappings, $property) as $alias) {
|
|
||||||
if ($this->rsm->columnOwnerMap[$alias] == $rootAlias) {
|
|
||||||
$sqlIdentifier[$property] = $alias;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isset($rootClass->associationMappings[$property])) {
|
|
||||||
$joinColumn = $rootClass->associationMappings[$property]['joinColumns'][0]['name'];
|
|
||||||
|
|
||||||
foreach (array_keys($this->rsm->metaMappings, $joinColumn) as $alias) {
|
|
||||||
if ($this->rsm->columnOwnerMap[$alias] == $rootAlias) {
|
|
||||||
$sqlIdentifier[$property] = $alias;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (count($sqlIdentifier) === 0) {
|
|
||||||
throw new \RuntimeException('The Paginator does not support Queries which only yield ScalarResults.');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (count($rootIdentifier) != count($sqlIdentifier)) {
|
|
||||||
throw new \RuntimeException(sprintf(
|
|
||||||
'Not all identifier properties can be found in the ResultSetMapping: %s',
|
|
||||||
implode(', ', array_diff($rootIdentifier, array_keys($sqlIdentifier)))
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Build the counter query
|
// Build the counter query
|
||||||
$sql = sprintf('SELECT DISTINCT %s FROM (%s) dctrn_result',
|
$sql = sprintf('SELECT DISTINCT %s FROM (%s) dctrn_result',
|
||||||
|
@ -208,7 +264,7 @@ class LimitSubqueryOutputWalker extends SqlWalker
|
||||||
$sql = $this->preserveSqlOrdering($sqlIdentifier, $innerSql, $sql, $orderByClause);
|
$sql = $this->preserveSqlOrdering($sqlIdentifier, $innerSql, $sql, $orderByClause);
|
||||||
|
|
||||||
// Apply the limit and offset.
|
// Apply the limit and offset.
|
||||||
$sql = $this->platform->modifyLimitQuery(
|
$sql = $this->_platform->modifyLimitQuery(
|
||||||
$sql, $this->maxResults, $this->firstResult
|
$sql, $this->maxResults, $this->firstResult
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -217,9 +273,12 @@ class LimitSubqueryOutputWalker extends SqlWalker
|
||||||
// but that is not possible from inside the output walker, so we dirty
|
// but that is not possible from inside the output walker, so we dirty
|
||||||
// up the one we have.
|
// up the one we have.
|
||||||
foreach ($sqlIdentifier as $property => $alias) {
|
foreach ($sqlIdentifier as $property => $alias) {
|
||||||
$this->rsm->addScalarResult($alias, $property);
|
$this->_rsm->addScalarResult($alias, $property);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Restore orderByClause
|
||||||
|
$AST->orderByClause = $orderByClause;
|
||||||
|
|
||||||
return $sql;
|
return $sql;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -241,7 +300,7 @@ class LimitSubqueryOutputWalker extends SqlWalker
|
||||||
// LimitSubqueryOutputWalker::walkPathExpression, which will be called
|
// LimitSubqueryOutputWalker::walkPathExpression, which will be called
|
||||||
// as the select statement is walked. We'll end up with an array of all
|
// as the select statement is walked. We'll end up with an array of all
|
||||||
// path expressions referenced in the query.
|
// path expressions referenced in the query.
|
||||||
$walker->walkSelectStatement($AST, false);
|
$walker->walkSelectStatementWithoutRowNumber($AST, false);
|
||||||
$orderByPathExpressions = $walker->getOrderByPathExpressions();
|
$orderByPathExpressions = $walker->getOrderByPathExpressions();
|
||||||
|
|
||||||
// Get a map of referenced identifiers to field names.
|
// Get a map of referenced identifiers to field names.
|
||||||
|
@ -298,17 +357,13 @@ class LimitSubqueryOutputWalker extends SqlWalker
|
||||||
}
|
}
|
||||||
|
|
||||||
// Rebuild the order by clause to work in the scope of the new select statement
|
// Rebuild the order by clause to work in the scope of the new select statement
|
||||||
/* @var array $sqlOrderColumns an array of items that need to be included in the select list */
|
|
||||||
/* @var array $orderBy an array of rebuilt order by items */
|
/* @var array $orderBy an array of rebuilt order by items */
|
||||||
list($sqlOrderColumns, $orderBy) = $this->rebuildOrderByClauseForOuterScope($orderByClause);
|
$orderBy = $this->rebuildOrderByClauseForOuterScope($orderByClause);
|
||||||
|
|
||||||
// Identifiers are always included in the select list, so there's no need to include them twice
|
|
||||||
$sqlOrderColumns = array_diff($sqlOrderColumns, $sqlIdentifier);
|
|
||||||
|
|
||||||
// Build the select distinct statement
|
// Build the select distinct statement
|
||||||
$sql = sprintf(
|
$sql = sprintf(
|
||||||
'SELECT DISTINCT %s FROM (%s) dctrn_result ORDER BY %s',
|
'SELECT DISTINCT %s FROM (%s) dctrn_result ORDER BY %s',
|
||||||
implode(', ', array_merge($sqlIdentifier, $sqlOrderColumns)),
|
implode(', ', $sqlIdentifier),
|
||||||
$innerSql,
|
$innerSql,
|
||||||
implode(', ', $orderBy)
|
implode(', ', $orderBy)
|
||||||
);
|
);
|
||||||
|
@ -333,8 +388,8 @@ class LimitSubqueryOutputWalker extends SqlWalker
|
||||||
= [];
|
= [];
|
||||||
|
|
||||||
// Generate DQL alias -> SQL table alias mapping
|
// Generate DQL alias -> SQL table alias mapping
|
||||||
foreach(array_keys($this->rsm->aliasMap) as $dqlAlias) {
|
foreach(array_keys($this->_rsm->aliasMap) as $dqlAlias) {
|
||||||
$dqlAliasToClassMap[$dqlAlias] = $class = $this->queryComponents[$dqlAlias]['metadata'];
|
$dqlAliasToClassMap[$dqlAlias] = $class = $this->_queryComponents[$dqlAlias]['metadata'];
|
||||||
$dqlAliasToSqlTableAliasMap[$dqlAlias] = $this->getSQLTableAlias($class->getTableName(), $dqlAlias);
|
$dqlAliasToSqlTableAliasMap[$dqlAlias] = $this->getSQLTableAlias($class->getTableName(), $dqlAlias);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -342,20 +397,12 @@ class LimitSubqueryOutputWalker extends SqlWalker
|
||||||
$fieldSearchPattern = '/(?<![a-z0-9_])%s\.%s(?![a-z0-9_])/i';
|
$fieldSearchPattern = '/(?<![a-z0-9_])%s\.%s(?![a-z0-9_])/i';
|
||||||
|
|
||||||
// Generate search patterns for each field's path expression in the order by clause
|
// Generate search patterns for each field's path expression in the order by clause
|
||||||
foreach($this->rsm->fieldMappings as $fieldAlias => $columnName) {
|
foreach($this->_rsm->fieldMappings as $fieldAlias => $columnName) {
|
||||||
$dqlAliasForFieldAlias = $this->rsm->columnOwnerMap[$fieldAlias];
|
$dqlAliasForFieldAlias = $this->_rsm->columnOwnerMap[$fieldAlias];
|
||||||
$class = $dqlAliasToClassMap[$dqlAliasForFieldAlias];
|
$columnName = $this->_quoteStrategy->getColumnName(
|
||||||
|
|
||||||
// If the field is from a joined child table, we won't be ordering
|
|
||||||
// on it.
|
|
||||||
if (!isset($class->fieldMappings[$columnName])) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
$columnName = $this->quoteStrategy->getColumnName(
|
|
||||||
$columnName,
|
$columnName,
|
||||||
$dqlAliasToClassMap[$dqlAliasForFieldAlias],
|
$dqlAliasToClassMap[$dqlAliasForFieldAlias],
|
||||||
$this->em->getConnection()->getDatabasePlatform()
|
$this->_em->getConnection()->getDatabasePlatform()
|
||||||
);
|
);
|
||||||
|
|
||||||
$sqlTableAliasForFieldAlias = $dqlAliasToSqlTableAliasMap[$dqlAliasForFieldAlias];
|
$sqlTableAliasForFieldAlias = $dqlAliasToSqlTableAliasMap[$dqlAliasForFieldAlias];
|
||||||
|
@ -364,7 +411,6 @@ class LimitSubqueryOutputWalker extends SqlWalker
|
||||||
$replacements[] = $fieldAlias;
|
$replacements[] = $fieldAlias;
|
||||||
}
|
}
|
||||||
|
|
||||||
$complexAddedOrderByAliases = 0;
|
|
||||||
foreach($orderByClause->orderByItems as $orderByItem) {
|
foreach($orderByClause->orderByItems as $orderByItem) {
|
||||||
// Walk order by item to get string representation of it
|
// Walk order by item to get string representation of it
|
||||||
$orderByItemString = $this->walkOrderByItem($orderByItem);
|
$orderByItemString = $this->walkOrderByItem($orderByItem);
|
||||||
|
@ -372,22 +418,10 @@ class LimitSubqueryOutputWalker extends SqlWalker
|
||||||
// Replace path expressions in the order by clause with their column alias
|
// Replace path expressions in the order by clause with their column alias
|
||||||
$orderByItemString = preg_replace($searchPatterns, $replacements, $orderByItemString);
|
$orderByItemString = preg_replace($searchPatterns, $replacements, $orderByItemString);
|
||||||
|
|
||||||
// The order by items are not required to be in the select list on Oracle and PostgreSQL, but
|
|
||||||
// for the sake of simplicity, order by items will be included in the select list on all platforms.
|
|
||||||
// This doesn't impact functionality.
|
|
||||||
$selectListAddition = trim(preg_replace('/([^ ]+) (?:asc|desc)/i', '$1', $orderByItemString));
|
|
||||||
|
|
||||||
// If the expression is an arithmetic expression, we need to create an alias for it.
|
|
||||||
if ($orderByItem->expression instanceof ArithmeticTerm) {
|
|
||||||
$orderByAlias = "ordr_" . $complexAddedOrderByAliases++;
|
|
||||||
$orderByItemString = $orderByAlias . " " . $orderByItem->type;
|
|
||||||
$selectListAddition .= " AS $orderByAlias";
|
|
||||||
}
|
|
||||||
$selectListAdditions[] = $selectListAddition;
|
|
||||||
$orderByItems[] = $orderByItemString;
|
$orderByItems[] = $orderByItemString;
|
||||||
}
|
}
|
||||||
|
|
||||||
return array($selectListAdditions, $orderByItems);
|
return $orderByItems;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -395,7 +429,7 @@ class LimitSubqueryOutputWalker extends SqlWalker
|
||||||
*/
|
*/
|
||||||
public function walkPathExpression($pathExpr)
|
public function walkPathExpression($pathExpr)
|
||||||
{
|
{
|
||||||
if (!in_array($pathExpr, $this->orderByPathExpressions)) {
|
if (!$this->platformSupportsRowNumber() && !in_array($pathExpr, $this->orderByPathExpressions)) {
|
||||||
$this->orderByPathExpressions[] = $pathExpr;
|
$this->orderByPathExpressions[] = $pathExpr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -411,4 +445,90 @@ class LimitSubqueryOutputWalker extends SqlWalker
|
||||||
{
|
{
|
||||||
return $this->orderByPathExpressions;
|
return $this->orderByPathExpressions;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param $AST
|
||||||
|
* @return string
|
||||||
|
* @throws \Doctrine\ORM\OptimisticLockException
|
||||||
|
* @throws \Doctrine\ORM\Query\QueryException
|
||||||
|
*/
|
||||||
|
private function getInnerSQL($AST)
|
||||||
|
{
|
||||||
|
// Set every select expression as visible(hidden = false) to
|
||||||
|
// make $AST have scalar mappings properly - this is relevant for referencing selected
|
||||||
|
// fields from outside the subquery, for example in the ORDER BY segment
|
||||||
|
$hiddens = array();
|
||||||
|
|
||||||
|
foreach ($AST->selectClause->selectExpressions as $idx => $expr) {
|
||||||
|
$hiddens[$idx] = $expr->hiddenAliasResultVariable;
|
||||||
|
$expr->hiddenAliasResultVariable = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$innerSql = parent::walkSelectStatement($AST);
|
||||||
|
|
||||||
|
// Restore hiddens
|
||||||
|
foreach ($AST->selectClause->selectExpressions as $idx => $expr) {
|
||||||
|
$expr->hiddenAliasResultVariable = $hiddens[$idx];
|
||||||
|
}
|
||||||
|
|
||||||
|
return $innerSql;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param $AST
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
private function getSQLIdentifier($AST)
|
||||||
|
{
|
||||||
|
// Find out the SQL alias of the identifier column of the root entity.
|
||||||
|
// It may be possible to make this work with multiple root entities but that
|
||||||
|
// would probably require issuing multiple queries or doing a UNION SELECT.
|
||||||
|
// So for now, it's not supported.
|
||||||
|
|
||||||
|
// Get the root entity and alias from the AST fromClause.
|
||||||
|
$from = $AST->fromClause->identificationVariableDeclarations;
|
||||||
|
if (count($from) !== 1) {
|
||||||
|
throw new \RuntimeException("Cannot count query which selects two FROM components, cannot make distinction");
|
||||||
|
}
|
||||||
|
|
||||||
|
$fromRoot = reset($from);
|
||||||
|
$rootAlias = $fromRoot->rangeVariableDeclaration->aliasIdentificationVariable;
|
||||||
|
$rootClass = $this->_queryComponents[$rootAlias]['metadata'];
|
||||||
|
$rootIdentifier = $rootClass->identifier;
|
||||||
|
|
||||||
|
// For every identifier, find out the SQL alias by combing through the ResultSetMapping
|
||||||
|
$sqlIdentifier = array();
|
||||||
|
foreach ($rootIdentifier as $property) {
|
||||||
|
if (isset($rootClass->fieldMappings[$property])) {
|
||||||
|
foreach (array_keys($this->_rsm->fieldMappings, $property) as $alias) {
|
||||||
|
if ($this->_rsm->columnOwnerMap[$alias] == $rootAlias) {
|
||||||
|
$sqlIdentifier[$property] = $alias;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isset($rootClass->associationMappings[$property])) {
|
||||||
|
$joinColumn = $rootClass->associationMappings[$property]['joinColumns'][0]['name'];
|
||||||
|
|
||||||
|
foreach (array_keys($this->_rsm->metaMappings, $joinColumn) as $alias) {
|
||||||
|
if ($this->_rsm->columnOwnerMap[$alias] == $rootAlias) {
|
||||||
|
$sqlIdentifier[$property] = $alias;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (count($sqlIdentifier) === 0) {
|
||||||
|
throw new \RuntimeException('The Paginator does not support Queries which only yield ScalarResults.');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (count($rootIdentifier) != count($sqlIdentifier)) {
|
||||||
|
throw new \RuntimeException(sprintf(
|
||||||
|
'Not all identifier properties can be found in the ResultSetMapping: %s',
|
||||||
|
implode(', ', array_diff($rootIdentifier, array_keys($sqlIdentifier)))
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
return $sqlIdentifier;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
35
lib/Doctrine/ORM/Tools/Pagination/RowNumberOverFunction.php
Normal file
35
lib/Doctrine/ORM/Tools/Pagination/RowNumberOverFunction.php
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* RowNumberOverFunction.php
|
||||||
|
* Created by William Schaller
|
||||||
|
* Date: 3/27/2015
|
||||||
|
* Time: 11:31 AM
|
||||||
|
*/
|
||||||
|
namespace Doctrine\ORM\Tools\Pagination;
|
||||||
|
|
||||||
|
|
||||||
|
use Doctrine\ORM\Query\AST\Functions\FunctionNode;
|
||||||
|
|
||||||
|
class RowNumberOverFunction extends FunctionNode
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @var \Doctrine\ORM\Query\AST\OrderByClause
|
||||||
|
*/
|
||||||
|
public $orderByClause;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @override
|
||||||
|
*/
|
||||||
|
public function getSql(\Doctrine\ORM\Query\SqlWalker $sqlWalker)
|
||||||
|
{
|
||||||
|
return 'ROW_NUMBER() OVER(' . trim($sqlWalker->walkOrderByClause(
|
||||||
|
$this->orderByClause
|
||||||
|
)) . ')';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @override
|
||||||
|
*/
|
||||||
|
public function parse(\Doctrine\ORM\Query\Parser $parser)
|
||||||
|
{}
|
||||||
|
}
|
|
@ -27,4 +27,9 @@ class Company
|
||||||
* @OneToOne(targetEntity="Logo", mappedBy="company", cascade={"persist"}, orphanRemoval=true)
|
* @OneToOne(targetEntity="Logo", mappedBy="company", cascade={"persist"}, orphanRemoval=true)
|
||||||
*/
|
*/
|
||||||
public $logo;
|
public $logo;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @OneToMany(targetEntity="Department", mappedBy="company", cascade={"persist"}, orphanRemoval=true)
|
||||||
|
*/
|
||||||
|
public $departments;
|
||||||
}
|
}
|
||||||
|
|
30
tests/Doctrine/Tests/Models/Pagination/Department.php
Normal file
30
tests/Doctrine/Tests/Models/Pagination/Department.php
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
<?php
|
||||||
|
namespace Doctrine\Tests\Models\Pagination;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Department
|
||||||
|
*
|
||||||
|
* @package Doctrine\Tests\Models\Pagination
|
||||||
|
*
|
||||||
|
* @author Bill Schaller
|
||||||
|
* @Entity
|
||||||
|
* @Table(name="pagination_department")
|
||||||
|
*/
|
||||||
|
class Department
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @Id @Column(type="integer")
|
||||||
|
* @GeneratedValue
|
||||||
|
*/
|
||||||
|
public $id;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @Column(type="string")
|
||||||
|
*/
|
||||||
|
public $name;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @ManyToOne(targetEntity="Company", inversedBy="departments", cascade={"persist"})
|
||||||
|
*/
|
||||||
|
public $company;
|
||||||
|
}
|
|
@ -8,6 +8,7 @@ use Doctrine\Tests\Models\CMS\CmsUser;
|
||||||
use Doctrine\Tests\Models\CMS\CmsGroup;
|
use Doctrine\Tests\Models\CMS\CmsGroup;
|
||||||
use Doctrine\ORM\Tools\Pagination\Paginator;
|
use Doctrine\ORM\Tools\Pagination\Paginator;
|
||||||
use Doctrine\Tests\Models\Pagination\Company;
|
use Doctrine\Tests\Models\Pagination\Company;
|
||||||
|
use Doctrine\Tests\Models\Pagination\Department;
|
||||||
use Doctrine\Tests\Models\Pagination\Logo;
|
use Doctrine\Tests\Models\Pagination\Logo;
|
||||||
use ReflectionMethod;
|
use ReflectionMethod;
|
||||||
|
|
||||||
|
@ -449,6 +450,106 @@ class PaginationTest extends \Doctrine\Tests\OrmFunctionalTestCase
|
||||||
$this->assertCount(3, $paginator->getIterator());
|
$this->assertCount(3, $paginator->getIterator());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dataProvider useOutputWalkers
|
||||||
|
*/
|
||||||
|
public function testIterateWithFetchJoinOneToManyWithOrderByColumnFromBoth($useOutputWalkers)
|
||||||
|
{
|
||||||
|
$dql = 'SELECT c, d FROM Doctrine\Tests\Models\Pagination\Company c JOIN c.departments d ORDER BY c.name';
|
||||||
|
$dqlAsc = $dql . " ASC, d.name";
|
||||||
|
$dqlDesc = $dql . " DESC, d.name";
|
||||||
|
$this->iterateWithOrderAsc($useOutputWalkers, true, $dqlAsc, "name");
|
||||||
|
$this->iterateWithOrderDesc($useOutputWalkers, true, $dqlDesc, "name");
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testIterateWithFetchJoinOneToManyWithOrderByColumnFromBothWithLimitWithOutputWalker()
|
||||||
|
{
|
||||||
|
$dql = 'SELECT c, d FROM Doctrine\Tests\Models\Pagination\Company c JOIN c.departments d ORDER BY c.name';
|
||||||
|
$dqlAsc = $dql . " ASC, d.name";
|
||||||
|
$dqlDesc = $dql . " DESC, d.name";
|
||||||
|
$this->iterateWithOrderAscWithLimit(true, true, $dqlAsc, "name");
|
||||||
|
$this->iterateWithOrderDescWithLimit(true, true, $dqlDesc, "name");
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testIterateWithFetchJoinOneToManyWithOrderByColumnFromBothWithLimitWithoutOutputWalker()
|
||||||
|
{
|
||||||
|
$this->setExpectedException("RuntimeException", "Cannot select distinct identifiers from query with LIMIT and ORDER BY on a joined entity. Use output walkers.");
|
||||||
|
|
||||||
|
$dql = 'SELECT c, d FROM Doctrine\Tests\Models\Pagination\Company c JOIN c.departments d ORDER BY c.name ASC';
|
||||||
|
// Ascending
|
||||||
|
$query = $this->_em->createQuery($dql);
|
||||||
|
|
||||||
|
// With limit
|
||||||
|
$query->setMaxResults(3);
|
||||||
|
$paginator = new Paginator($query, true);
|
||||||
|
$paginator->setUseOutputWalkers(false);
|
||||||
|
$iter = $paginator->getIterator();
|
||||||
|
iterator_to_array($iter);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dataProvider useOutputWalkers
|
||||||
|
*/
|
||||||
|
public function testIterateWithFetchJoinOneToManyWithOrderByColumnFromRoot($useOutputWalkers)
|
||||||
|
{
|
||||||
|
$dql = 'SELECT c, d FROM Doctrine\Tests\Models\Pagination\Company c JOIN c.departments d ORDER BY c.name';
|
||||||
|
$this->iterateWithOrderAsc($useOutputWalkers, true, $dql, "name");
|
||||||
|
$this->iterateWithOrderDesc($useOutputWalkers, true, $dql, "name");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dataProvider useOutputWalkers
|
||||||
|
*/
|
||||||
|
public function testIterateWithFetchJoinOneToManyWithOrderByColumnFromRootWithLimit($useOutputWalkers)
|
||||||
|
{
|
||||||
|
$dql = 'SELECT c, d FROM Doctrine\Tests\Models\Pagination\Company c JOIN c.departments d ORDER BY c.name';
|
||||||
|
$this->iterateWithOrderAscWithLimit($useOutputWalkers, true, $dql, "name");
|
||||||
|
$this->iterateWithOrderDescWithLimit($useOutputWalkers, true, $dql, "name");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dataProvider useOutputWalkers
|
||||||
|
*/
|
||||||
|
public function testIterateWithFetchJoinOneToManyWithOrderByColumnFromRootWithLimitAndOffset($useOutputWalkers)
|
||||||
|
{
|
||||||
|
$dql = 'SELECT c, d FROM Doctrine\Tests\Models\Pagination\Company c JOIN c.departments d ORDER BY c.name';
|
||||||
|
$this->iterateWithOrderAscWithLimitAndOffset($useOutputWalkers, true, $dql, "name");
|
||||||
|
$this->iterateWithOrderDescWithLimitAndOffset($useOutputWalkers, true, $dql, "name");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dataProvider useOutputWalkers
|
||||||
|
*/
|
||||||
|
public function testIterateWithFetchJoinOneToManyWithOrderByColumnFromJoined($useOutputWalkers)
|
||||||
|
{
|
||||||
|
$dql = 'SELECT c, d FROM Doctrine\Tests\Models\Pagination\Company c JOIN c.departments d ORDER BY d.name';
|
||||||
|
$this->iterateWithOrderAsc($useOutputWalkers, true, $dql, "name");
|
||||||
|
$this->iterateWithOrderDesc($useOutputWalkers, true, $dql, "name");
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testIterateWithFetchJoinOneToManyWithOrderByColumnFromJoinedWithLimitWithOutputWalker()
|
||||||
|
{
|
||||||
|
$dql = 'SELECT c, d FROM Doctrine\Tests\Models\Pagination\Company c JOIN c.departments d ORDER BY d.name';
|
||||||
|
$this->iterateWithOrderAscWithLimit(true, true, $dql, "name");
|
||||||
|
$this->iterateWithOrderDescWithLimit(true, true, $dql, "name");
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testIterateWithFetchJoinOneToManyWithOrderByColumnFromJoinedWithLimitWithoutOutputWalker()
|
||||||
|
{
|
||||||
|
$this->setExpectedException("RuntimeException", "Cannot select distinct identifiers from query with LIMIT and ORDER BY on a joined entity. Use output walkers.");
|
||||||
|
$dql = 'SELECT c, d FROM Doctrine\Tests\Models\Pagination\Company c JOIN c.departments d ORDER BY d.name ASC';
|
||||||
|
|
||||||
|
// Ascending
|
||||||
|
$query = $this->_em->createQuery($dql);
|
||||||
|
|
||||||
|
// With limit
|
||||||
|
$query->setMaxResults(3);
|
||||||
|
$paginator = new Paginator($query, true);
|
||||||
|
$paginator->setUseOutputWalkers(false);
|
||||||
|
$iter = $paginator->getIterator();
|
||||||
|
iterator_to_array($iter);
|
||||||
|
}
|
||||||
|
|
||||||
public function testDetectOutputWalker()
|
public function testDetectOutputWalker()
|
||||||
{
|
{
|
||||||
// This query works using the output walkers but causes an exception using the TreeWalker
|
// This query works using the output walkers but causes an exception using the TreeWalker
|
||||||
|
@ -553,6 +654,12 @@ class PaginationTest extends \Doctrine\Tests\OrmFunctionalTestCase
|
||||||
$company->logo->image_width = 100 + $i;
|
$company->logo->image_width = 100 + $i;
|
||||||
$company->logo->image_height = 100 + $i;
|
$company->logo->image_height = 100 + $i;
|
||||||
$company->logo->company = $company;
|
$company->logo->company = $company;
|
||||||
|
for($j=0;$j<3;$j++) {
|
||||||
|
$department = new Department();
|
||||||
|
$department->name = "name$i$j";
|
||||||
|
$department->company = $company;
|
||||||
|
$company->departments[] = $department;
|
||||||
|
}
|
||||||
$this->_em->persist($company);
|
$this->_em->persist($company);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -270,6 +270,7 @@ abstract class OrmFunctionalTestCase extends OrmTestCase
|
||||||
'pagination' => array(
|
'pagination' => array(
|
||||||
'Doctrine\Tests\Models\Pagination\Company',
|
'Doctrine\Tests\Models\Pagination\Company',
|
||||||
'Doctrine\Tests\Models\Pagination\Logo',
|
'Doctrine\Tests\Models\Pagination\Logo',
|
||||||
|
'Doctrine\Tests\Models\Pagination\Department',
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -521,6 +522,7 @@ abstract class OrmFunctionalTestCase extends OrmTestCase
|
||||||
|
|
||||||
if (isset($this->_usedModelSets['pagination'])) {
|
if (isset($this->_usedModelSets['pagination'])) {
|
||||||
$conn->executeUpdate('DELETE FROM pagination_logo');
|
$conn->executeUpdate('DELETE FROM pagination_logo');
|
||||||
|
$conn->executeUpdate('DELETE FROM pagination_department');
|
||||||
$conn->executeUpdate('DELETE FROM pagination_company');
|
$conn->executeUpdate('DELETE FROM pagination_company');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue