Completed
Pull Request — master (#1922)
by chihiro
33:30
created

CountOutputWalker::walkSelectStatement()   C

Complexity

Conditions 11
Paths 38

Size

Total Lines 51
Code Lines 30

Duplication

Lines 10
Ratio 19.61 %

Code Coverage

Tests 21
CRAP Score 13.5412

Importance

Changes 0
Metric Value
cc 11
eloc 30
nc 38
nop 1
dl 10
loc 51
ccs 21
cts 29
cp 0.7241
crap 13.5412
rs 5.7333
c 0
b 0
f 0

How to fix   Long Method    Complexity   

Long Method

Small methods make your code easier to understand, in particular if combined with a good name. Besides, if your method is small, finding a good name is usually much easier.

For example, if you find yourself adding comments to a method's body, this is usually a good sign to extract the commented part to a new method, and use the comment as a starting point when coming up with a good name for this new method.

Commonly applied refactorings include:

1
<?php
2
/*
3
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
4
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
5
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
6
 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
7
 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
8
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
9
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
10
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
11
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
12
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
13
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
14
 *
15
 * This software consists of voluntary contributions made by many individuals
16
 * and is licensed under the MIT license. For more information, see
17
 * <http://www.doctrine-project.org>.
18
 */
19
namespace Eccube\Doctrine\ORM\Tools\Pagination;
20
use Doctrine\ORM\Query\SqlWalker;
21
use Doctrine\ORM\Query\AST\SelectStatement;
22
/**
23
 * Wraps the query in order to accurately count the root objects.
24
 *
25
 * Given a DQL like `SELECT u FROM User u` it will generate an SQL query like:
26
 * SELECT COUNT(*) (SELECT DISTINCT <id> FROM (<original SQL>))
27
 *
28
 * Works with composite keys but cannot deal with queries that have multiple
29
 * root entities (e.g. `SELECT f, b from Foo, Bar`)
30
 *
31
 * @author Sander Marechal <[email protected]>
32
 */
33
class CountOutputWalker extends SqlWalker
34
{
35
    /**
36
     * @var \Doctrine\DBAL\Platforms\AbstractPlatform
37
     */
38
    private $platform;
0 ignored issues
show
Comprehensibility introduced by
Consider using a different property name as you override a private property of the parent class.
Loading history...
39
    /**
40
     * @var \Doctrine\ORM\Query\ResultSetMapping
41
     */
42
    private $rsm;
0 ignored issues
show
Comprehensibility introduced by
Consider using a different property name as you override a private property of the parent class.
Loading history...
43
    /**
44
     * @var array
45
     */
46
    private $queryComponents;
0 ignored issues
show
Comprehensibility introduced by
Consider using a different property name as you override a private property of the parent class.
Loading history...
47
    /**
48
     * Constructor.
49
     *
50
     * Stores various parameters that are otherwise unavailable
51
     * because Doctrine\ORM\Query\SqlWalker keeps everything private without
52
     * accessors.
53
     *
54
     * @param \Doctrine\ORM\Query              $query
55
     * @param \Doctrine\ORM\Query\ParserResult $parserResult
56
     * @param array                            $queryComponents
57
     */
58 17
    public function __construct($query, $parserResult, array $queryComponents)
59
    {
60 17
        $this->platform = $query->getEntityManager()->getConnection()->getDatabasePlatform();
61 17
        $this->rsm = $parserResult->getResultSetMapping();
62 17
        $this->queryComponents = $queryComponents;
63 17
        parent::__construct($query, $parserResult, $queryComponents);
64
    }
65
    /**
66
     * Walks down a SelectStatement AST node, wrapping it in a COUNT (SELECT DISTINCT).
67
     *
68
     * Note that the ORDER BY clause is not removed. Many SQL implementations (e.g. MySQL)
69
     * are able to cache subqueries. By keeping the ORDER BY clause intact, the limitSubQuery
70
     * that will most likely be executed next can be read from the native SQL cache.
71
     *
72
     * @param SelectStatement $AST
73
     *
74
     * @return string
75
     *
76
     * @throws \RuntimeException
77
     */
78 17
    public function walkSelectStatement(SelectStatement $AST)
79
    {
80 17
        if ($this->platform->getName() === "mssql") {
81
            $AST->orderByClause = null;
82
        }
83 17
        $sql = parent::walkSelectStatement($AST);
84
        // Find out the SQL alias of the identifier column of the root entity
85
        // It may be possible to make this work with multiple root entities but that
86
        // would probably require issuing multiple queries or doing a UNION SELECT
87
        // so for now, It's not supported.
88
        // Get the root entity and alias from the AST fromClause
89 17
        $from = $AST->fromClause->identificationVariableDeclarations;
90 17
        if (count($from) > 1) {
91
            throw new \RuntimeException("Cannot count query which selects two FROM components, cannot make distinction");
92
        }
93 17
        $fromRoot       = reset($from);
94 17
        $rootAlias      = $fromRoot->rangeVariableDeclaration->aliasIdentificationVariable;
95 17
        $rootClass      = $this->queryComponents[$rootAlias]['metadata'];
96 17
        $rootIdentifier = $rootClass->identifier;
97
        // For every identifier, find out the SQL alias by combing through the ResultSetMapping
98 17
        $sqlIdentifier = array();
99 17
        foreach ($rootIdentifier as $property) {
100 17
            if (isset($rootClass->fieldMappings[$property])) {
101 17 View Code Duplication
                foreach (array_keys($this->rsm->fieldMappings, $property) as $alias) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated across your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
102 17
                    if ($this->rsm->columnOwnerMap[$alias] == $rootAlias) {
103 17
                        $sqlIdentifier[$property] = $alias;
104
                    }
105
                }
106
            }
107 17
            if (isset($rootClass->associationMappings[$property])) {
108
                $joinColumn = $rootClass->associationMappings[$property]['joinColumns'][0]['name'];
109 View Code Duplication
                foreach (array_keys($this->rsm->metaMappings, $joinColumn) as $alias) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated across your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
110
                    if ($this->rsm->columnOwnerMap[$alias] == $rootAlias) {
111 17
                        $sqlIdentifier[$property] = $alias;
112
                    }
113
                }
114
            }
115
        }
116 17
        if (count($rootIdentifier) != count($sqlIdentifier)) {
117
            throw new \RuntimeException(sprintf(
118
                'Not all identifier properties can be found in the ResultSetMapping: %s',
119
                implode(', ', array_diff($rootIdentifier, array_keys($sqlIdentifier)))
120
            ));
121
        }
122
        // Build the counter query
123 17
        return sprintf('SELECT %s AS dctrn_count FROM (SELECT DISTINCT %s FROM (%s) dctrn_result) dctrn_table',
124 17
            $this->platform->getCountExpression('*'),
125 17
            implode(', ', $sqlIdentifier),
126
            $sql
127
        );
128
    }
129
}
130