2009-08-13 14:13:06 +04:00
< ? php
/*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* " AS IS " AND ANY EXPRESS OR IMPLIED WARRANTIES , INCLUDING , BUT NOT
* LIMITED TO , THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED . IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT , INDIRECT , INCIDENTAL ,
* SPECIAL , EXEMPLARY , OR CONSEQUENTIAL DAMAGES ( INCLUDING , BUT NOT
* LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES ; LOSS OF USE ,
* DATA , OR PROFITS ; OR BUSINESS INTERRUPTION ) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY , WHETHER IN CONTRACT , STRICT LIABILITY , OR TORT
* ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE .
*
* This software consists of voluntary contributions made by many individuals
* and is licensed under the LGPL . For more information , see
* < http :// www . doctrine - project . org >.
*/
namespace Doctrine\Tests\ORM\Functional ;
use Doctrine\ORM\Query ;
/**
* Test case for custom AST walking and modification .
*
* @ author Roman Borschel < roman @ code - factory . org >
* @ license http :// www . opensource . org / licenses / lgpl - license . php LGPL
* @ link http :// www . doctrine - project . org
* @ since 2.0
*/
2011-10-29 22:42:44 +04:00
class CustomTreeWalkersTest extends \Doctrine\Tests\OrmTestCase
2009-08-13 14:13:06 +04:00
{
2011-10-29 22:42:44 +04:00
private $_em ;
protected function setUp ()
{
$this -> _em = $this -> _getTestEntityManager ();
2009-08-13 14:13:06 +04:00
}
2010-04-30 05:15:36 +04:00
2012-10-06 13:15:07 +04:00
public function generateSql ( $dqlToBeTested , $treeWalkers , $outputWalker )
2010-04-30 05:15:36 +04:00
{
2012-10-06 13:15:07 +04:00
$query = $this -> _em -> createQuery ( $dqlToBeTested );
$query -> setHint ( Query :: HINT_CUSTOM_TREE_WALKERS , $treeWalkers )
2014-02-06 02:27:17 +04:00
-> useQueryCache ( false );
2012-10-06 13:15:07 +04:00
if ( $outputWalker ) {
$query -> setHint ( Query :: HINT_CUSTOM_OUTPUT_WALKER , $outputWalker );
}
return $query -> getSql ();
}
2010-04-30 05:15:36 +04:00
2012-10-06 13:15:07 +04:00
public function assertSqlGeneration ( $dqlToBeTested , $sqlToBeConfirmed , $treeWalkers = array (), $outputWalker = null )
{
try {
$this -> assertEquals ( $sqlToBeConfirmed , $this -> generateSql ( $dqlToBeTested , $treeWalkers , $outputWalker ));
2010-04-30 05:15:36 +04:00
} catch ( \Exception $e ) {
2010-07-23 08:55:33 +04:00
$this -> fail ( $e -> getMessage () . ' at "' . $e -> getFile () . '" on line ' . $e -> getLine ());
2010-04-30 05:15:36 +04:00
}
}
2009-08-13 14:13:06 +04:00
public function testSupportsQueriesWithoutWhere ()
2010-07-23 08:55:33 +04:00
{
$this -> assertSqlGeneration (
'select u from Doctrine\Tests\Models\CMS\CmsUser u' ,
2014-03-21 22:39:51 +04:00
" SELECT c0_.id AS id_0, c0_.status AS status_1, c0_.username AS username_2, c0_.name AS name_3, c0_.email_id AS email_id_4 FROM cms_users c0_ WHERE c0_.id = 1 " ,
2012-10-06 13:15:07 +04:00
array ( 'Doctrine\Tests\ORM\Functional\CustomTreeWalker' )
2010-07-23 08:55:33 +04:00
);
}
public function testSupportsQueriesWithMultipleConditionalExpressions ()
2009-08-13 14:13:06 +04:00
{
2010-04-30 05:15:36 +04:00
$this -> assertSqlGeneration (
'select u from Doctrine\Tests\Models\CMS\CmsUser u where u.name = :name or u.name = :otherName' ,
2014-03-21 22:39:51 +04:00
" SELECT c0_.id AS id_0, c0_.status AS status_1, c0_.username AS username_2, c0_.name AS name_3, c0_.email_id AS email_id_4 FROM cms_users c0_ WHERE (c0_.name = ? OR c0_.name = ?) AND c0_.id = 1 " ,
2012-10-06 13:15:07 +04:00
array ( 'Doctrine\Tests\ORM\Functional\CustomTreeWalker' )
2010-04-30 05:15:36 +04:00
);
}
2011-10-29 22:42:44 +04:00
2010-07-23 08:55:33 +04:00
public function testSupportsQueriesWithSimpleConditionalExpression ()
2010-04-30 05:15:36 +04:00
{
$this -> assertSqlGeneration (
'select u from Doctrine\Tests\Models\CMS\CmsUser u where u.name = :name' ,
2014-03-21 22:39:51 +04:00
" SELECT c0_.id AS id_0, c0_.status AS status_1, c0_.username AS username_2, c0_.name AS name_3, c0_.email_id AS email_id_4 FROM cms_users c0_ WHERE c0_.name = ? AND c0_.id = 1 " ,
2012-10-06 13:15:07 +04:00
array ( 'Doctrine\Tests\ORM\Functional\CustomTreeWalker' )
2010-04-30 05:15:36 +04:00
);
}
2012-10-06 13:15:07 +04:00
public function testSetUnknownQueryComponentThrowsException ()
{
$this -> setExpectedException ( " Doctrine \ ORM \ Query \ QueryException " , " Invalid query component given for DQL alias 'x', requires 'metadata', 'parent', 'relation', 'map', 'nestingLevel' and 'token' keys. " );
$this -> generateSql (
'select u from Doctrine\Tests\Models\CMS\CmsUser u' ,
array (),
__NAMESPACE__ . '\\AddUnknownQueryComponentWalker'
);
}
2014-02-06 02:27:17 +04:00
2014-02-06 11:53:34 +04:00
public function testSupportsSeveralHintsQueries ()
2014-02-06 02:27:17 +04:00
{
$this -> assertSqlGeneration (
'select u from Doctrine\Tests\Models\CMS\CmsUser u' ,
2014-03-21 22:39:51 +04:00
" SELECT c0_.id AS id_0, c0_.status AS status_1, c0_.username AS username_2, c0_.name AS name_3, c1_.id AS id_4, c1_.country AS country_5, c1_.zip AS zip_6, c1_.city AS city_7, c0_.email_id AS email_id_8, c1_.user_id AS user_id_9 FROM cms_users c0_ LEFT JOIN cms_addresses c1_ ON c0_.id = c1_.user_id WHERE c0_.id = 1 " ,
2014-02-06 02:27:17 +04:00
array ( 'Doctrine\Tests\ORM\Functional\CustomTreeWalkerJoin' , 'Doctrine\Tests\ORM\Functional\CustomTreeWalker' )
);
}
2012-10-06 13:15:07 +04:00
}
class AddUnknownQueryComponentWalker extends Query\SqlWalker
{
public function walkSelectStatement ( Query\AST\SelectStatement $selectStatement )
{
parent :: walkSelectStatement ( $selectStatement );
$this -> setQueryComponent ( 'x' , array ());
}
2009-08-13 14:13:06 +04:00
}
class CustomTreeWalker extends Query\TreeWalkerAdapter
{
public function walkSelectStatement ( Query\AST\SelectStatement $selectStatement )
{
// Get the DQL aliases of all the classes we want to modify
$dqlAliases = array ();
2010-04-30 05:15:36 +04:00
2009-08-13 14:13:06 +04:00
foreach ( $this -> _getQueryComponents () as $dqlAlias => $comp ) {
// Hard-coded check just for demonstration: We want to modify the query if
// it involves the CmsUser class.
if ( $comp [ 'metadata' ] -> name == 'Doctrine\Tests\Models\CMS\CmsUser' ) {
$dqlAliases [] = $dqlAlias ;
}
}
2011-10-29 22:42:44 +04:00
2009-08-13 14:13:06 +04:00
// Create our conditions for all involved classes
$factors = array ();
foreach ( $dqlAliases as $alias ) {
2010-07-23 08:55:33 +04:00
$pathExpr = new Query\AST\PathExpression ( Query\AST\PathExpression :: TYPE_STATE_FIELD , $alias , 'id' );
2009-08-13 14:13:06 +04:00
$pathExpr -> type = Query\AST\PathExpression :: TYPE_STATE_FIELD ;
$comparisonExpr = new Query\AST\ComparisonExpression ( $pathExpr , '=' , 1 );
$condPrimary = new Query\AST\ConditionalPrimary ;
$condPrimary -> simpleConditionalExpression = $comparisonExpr ;
$factor = new Query\AST\ConditionalFactor ( $condPrimary );
$factors [] = $factor ;
}
2011-10-29 22:42:44 +04:00
2010-04-30 05:15:36 +04:00
if (( $whereClause = $selectStatement -> whereClause ) !== null ) {
2009-08-13 14:13:06 +04:00
// There is already a WHERE clause, so append the conditions
2010-04-30 05:15:36 +04:00
$condExpr = $whereClause -> conditionalExpression ;
// Since Phase 1 AST optimizations were included, we need to re-add the ConditionalExpression
if ( ! ( $condExpr instanceof Query\AST\ConditionalExpression )) {
$condExpr = new Query\AST\ConditionalExpression ( array ( $condExpr ));
$whereClause -> conditionalExpression = $condExpr ;
}
2011-10-29 22:42:44 +04:00
2010-04-30 05:15:36 +04:00
$existingTerms = $whereClause -> conditionalExpression -> conditionalTerms ;
2011-10-29 22:42:44 +04:00
2009-08-13 14:13:06 +04:00
if ( count ( $existingTerms ) > 1 ) {
// More than one term, so we need to wrap all these terms in a single root term
// i.e: "WHERE u.name = :foo or u.other = :bar" => "WHERE (u.name = :foo or u.other = :bar) AND <our condition>"
2011-10-29 22:42:44 +04:00
2009-08-13 14:13:06 +04:00
$primary = new Query\AST\ConditionalPrimary ;
$primary -> conditionalExpression = new Query\AST\ConditionalExpression ( $existingTerms );
$existingFactor = new Query\AST\ConditionalFactor ( $primary );
$term = new Query\AST\ConditionalTerm ( array_merge ( array ( $existingFactor ), $factors ));
2011-10-29 22:42:44 +04:00
2009-08-13 14:13:06 +04:00
$selectStatement -> whereClause -> conditionalExpression -> conditionalTerms = array ( $term );
} else {
// Just one term so we can simply append our factors to that term
$singleTerm = $selectStatement -> whereClause -> conditionalExpression -> conditionalTerms [ 0 ];
2010-04-30 05:15:36 +04:00
// Since Phase 1 AST optimizations were included, we need to re-add the ConditionalExpression
if ( ! ( $singleTerm instanceof Query\AST\ConditionalTerm )) {
$singleTerm = new Query\AST\ConditionalTerm ( array ( $singleTerm ));
$selectStatement -> whereClause -> conditionalExpression -> conditionalTerms [ 0 ] = $singleTerm ;
}
2009-08-13 14:13:06 +04:00
$singleTerm -> conditionalFactors = array_merge ( $singleTerm -> conditionalFactors , $factors );
$selectStatement -> whereClause -> conditionalExpression -> conditionalTerms = array ( $singleTerm );
}
} else {
// Create a new WHERE clause with our factors
$term = new Query\AST\ConditionalTerm ( $factors );
$condExpr = new Query\AST\ConditionalExpression ( array ( $term ));
$whereClause = new Query\AST\WhereClause ( $condExpr );
$selectStatement -> whereClause = $whereClause ;
}
}
}
2014-02-06 02:27:17 +04:00
class CustomTreeWalkerJoin extends Query\TreeWalkerAdapter
{
public function walkSelectStatement ( Query\AST\SelectStatement $selectStatement )
{
foreach ( $selectStatement -> fromClause -> identificationVariableDeclarations as $identificationVariableDeclaration ) {
if ( $identificationVariableDeclaration -> rangeVariableDeclaration -> abstractSchemaName == 'Doctrine\Tests\Models\CMS\CmsUser' ) {
$identificationVariableDeclaration -> joins [] = new Query\AST\Join (
Query\AST\Join :: JOIN_TYPE_LEFT ,
new Query\AST\JoinAssociationDeclaration (
new Query\AST\JoinAssociationPathExpression (
$identificationVariableDeclaration -> rangeVariableDeclaration -> aliasIdentificationVariable ,
'address'
),
$identificationVariableDeclaration -> rangeVariableDeclaration -> aliasIdentificationVariable . 'a' ,
null
)
);
$selectStatement -> selectClause -> selectExpressions [] =
new Query\AST\SelectExpression (
$identificationVariableDeclaration -> rangeVariableDeclaration -> aliasIdentificationVariable . 'a' ,
null ,
false
);
$meta1 = $this -> _getQuery () -> getEntityManager () -> getClassMetadata ( 'Doctrine\Tests\Models\CMS\CmsUser' );
$meta = $this -> _getQuery () -> getEntityManager () -> getClassMetadata ( 'Doctrine\Tests\Models\CMS\CmsAddress' );
$this -> setQueryComponent ( $identificationVariableDeclaration -> rangeVariableDeclaration -> aliasIdentificationVariable . 'a' ,
array (
'metadata' => $meta ,
'parent' => $identificationVariableDeclaration -> rangeVariableDeclaration -> aliasIdentificationVariable ,
'relation' => $meta1 -> getAssociationMapping ( 'address' ),
'map' => null ,
'nestingLevel' => 0 ,
'token' => null
)
);
}
}
}
}