2012-03-06 19:24:44 +04:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* Doctrine ORM
|
|
|
|
*
|
|
|
|
* LICENSE
|
|
|
|
*
|
|
|
|
* This source file is subject to the new BSD license that is bundled
|
|
|
|
* with this package in the file LICENSE.txt.
|
|
|
|
* If you did not receive a copy of the license and are unable to
|
|
|
|
* obtain it through the world-wide-web, please send an email
|
|
|
|
* to kontakt@beberlei.de so I can send you a copy immediately.
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace Doctrine\ORM\Tools\Pagination;
|
|
|
|
|
2012-10-12 15:53:20 +04:00
|
|
|
use Doctrine\ORM\Query\SqlWalker;
|
|
|
|
use Doctrine\ORM\Query\AST\SelectStatement;
|
2012-03-06 19:24:44 +04:00
|
|
|
|
|
|
|
/**
|
2012-12-14 17:13:22 +04:00
|
|
|
* Wraps the query in order to accurately count the root objects.
|
2012-03-06 19:24:44 +04:00
|
|
|
*
|
|
|
|
* Given a DQL like `SELECT u FROM User u` it will generate an SQL query like:
|
|
|
|
* SELECT COUNT(*) (SELECT DISTINCT <id> FROM (<original SQL>))
|
|
|
|
*
|
|
|
|
* Works with composite keys but cannot deal with queries that have multiple
|
|
|
|
* root entities (e.g. `SELECT f, b from Foo, Bar`)
|
|
|
|
*
|
|
|
|
* @author Sander Marechal <s.marechal@jejik.com>
|
|
|
|
*/
|
2012-03-12 11:33:35 +04:00
|
|
|
class CountOutputWalker extends SqlWalker
|
2012-03-06 19:24:44 +04:00
|
|
|
{
|
|
|
|
/**
|
2012-09-21 03:20:06 +04:00
|
|
|
* @var \Doctrine\DBAL\Platforms\AbstractPlatform
|
2012-03-06 19:24:44 +04:00
|
|
|
*/
|
|
|
|
private $platform;
|
|
|
|
|
|
|
|
/**
|
2012-09-21 03:20:06 +04:00
|
|
|
* @var \Doctrine\ORM\Query\ResultSetMapping
|
2012-03-06 19:24:44 +04:00
|
|
|
*/
|
|
|
|
private $rsm;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
private $queryComponents;
|
|
|
|
|
|
|
|
/**
|
2012-12-14 17:13:22 +04:00
|
|
|
* Constructor.
|
|
|
|
*
|
|
|
|
* Stores various parameters that are otherwise unavailable
|
2012-03-06 19:24:44 +04:00
|
|
|
* because Doctrine\ORM\Query\SqlWalker keeps everything private without
|
|
|
|
* accessors.
|
|
|
|
*
|
2012-12-14 17:13:22 +04:00
|
|
|
* @param \Doctrine\ORM\Query $query
|
2012-09-21 03:20:06 +04:00
|
|
|
* @param \Doctrine\ORM\Query\ParserResult $parserResult
|
2012-12-14 17:13:22 +04:00
|
|
|
* @param array $queryComponents
|
2012-03-06 19:24:44 +04:00
|
|
|
*/
|
|
|
|
public function __construct($query, $parserResult, array $queryComponents)
|
|
|
|
{
|
|
|
|
$this->platform = $query->getEntityManager()->getConnection()->getDatabasePlatform();
|
|
|
|
$this->rsm = $parserResult->getResultSetMapping();
|
|
|
|
$this->queryComponents = $queryComponents;
|
|
|
|
|
|
|
|
parent::__construct($query, $parserResult, $queryComponents);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2012-12-14 17:13:22 +04:00
|
|
|
* Walks down a SelectStatement AST node, wrapping it in a COUNT (SELECT DISTINCT).
|
2012-03-06 19:24:44 +04:00
|
|
|
*
|
|
|
|
* Note that the ORDER BY clause is not removed. Many SQL implementations (e.g. MySQL)
|
|
|
|
* are able to cache subqueries. By keeping the ORDER BY clause intact, the limitSubQuery
|
|
|
|
* that will most likely be executed next can be read from the native SQL cache.
|
|
|
|
*
|
|
|
|
* @param SelectStatement $AST
|
2012-12-14 17:13:22 +04:00
|
|
|
*
|
2012-03-06 19:24:44 +04:00
|
|
|
* @return string
|
2012-12-14 17:13:22 +04:00
|
|
|
*
|
|
|
|
* @throws \RuntimeException
|
2012-03-06 19:24:44 +04:00
|
|
|
*/
|
|
|
|
public function walkSelectStatement(SelectStatement $AST)
|
|
|
|
{
|
|
|
|
$sql = parent::walkSelectStatement($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");
|
|
|
|
}
|
|
|
|
|
2012-05-27 20:33:35 +04:00
|
|
|
$rootAlias = $from[0]->rangeVariableDeclaration->aliasIdentificationVariable;
|
|
|
|
$rootClass = $this->queryComponents[$rootAlias]['metadata'];
|
|
|
|
$rootIdentifier = $rootClass->identifier;
|
2012-03-06 19:24:44 +04:00
|
|
|
|
|
|
|
// For every identifier, find out the SQL alias by combing through the ResultSetMapping
|
|
|
|
$sqlIdentifier = array();
|
|
|
|
foreach ($rootIdentifier as $property) {
|
2012-05-27 20:33:35 +04:00
|
|
|
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;
|
|
|
|
}
|
2012-03-06 19:24:44 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
2012-05-27 15:22:48 +04:00
|
|
|
return sprintf('SELECT %s AS dctrn_count FROM (SELECT DISTINCT %s FROM (%s) dctrn_result) dctrn_table',
|
2012-03-06 19:24:44 +04:00
|
|
|
$this->platform->getCountExpression('*'),
|
|
|
|
implode(', ', $sqlIdentifier),
|
|
|
|
$sql
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|