useModelSet('cms'); parent::setUp(); $this->populate(); } /** * @dataProvider useOutputWalkers */ public function testCountSimpleWithoutJoin($useOutputWalkers) { $dql = "SELECT u FROM Doctrine\Tests\Models\CMS\CmsUser u"; $query = $this->_em->createQuery($dql); $paginator = new Paginator($query); $paginator->setUseOutputWalkers($useOutputWalkers); $this->assertCount(3, $paginator); } /** * @dataProvider useOutputWalkers */ public function testCountWithFetchJoin($useOutputWalkers) { $dql = "SELECT u,g FROM Doctrine\Tests\Models\CMS\CmsUser u JOIN u.groups g"; $query = $this->_em->createQuery($dql); $paginator = new Paginator($query); $paginator->setUseOutputWalkers($useOutputWalkers); $this->assertCount(3, $paginator); } public function testCountComplexWithOutputWalker() { $dql = "SELECT g, COUNT(u.id) AS userCount FROM Doctrine\Tests\Models\CMS\CmsGroup g LEFT JOIN g.users u GROUP BY g HAVING COUNT(u.id) > 0"; $query = $this->_em->createQuery($dql); $paginator = new Paginator($query); $paginator->setUseOutputWalkers(true); $this->assertCount(9, $paginator); } /** * @dataProvider useOutputWalkers */ public function testIterateSimpleWithoutJoinFetchJoinHandlingOff($useOutputWalkers) { $dql = "SELECT u FROM Doctrine\Tests\Models\CMS\CmsUser u"; $query = $this->_em->createQuery($dql); $paginator = new Paginator($query, false); $paginator->setUseOutputWalkers($useOutputWalkers); $this->assertCount(3, $paginator->getIterator()); } /** * @dataProvider useOutputWalkers */ public function testIterateSimpleWithoutJoinFetchJoinHandlingOn($useOutputWalkers) { $dql = "SELECT u FROM Doctrine\Tests\Models\CMS\CmsUser u"; $query = $this->_em->createQuery($dql); $paginator = new Paginator($query, true); $paginator->setUseOutputWalkers($useOutputWalkers); $this->assertCount(3, $paginator->getIterator()); } /** * @dataProvider useOutputWalkers */ public function testIterateWithFetchJoin($useOutputWalkers) { $dql = "SELECT u,g FROM Doctrine\Tests\Models\CMS\CmsUser u JOIN u.groups g"; $query = $this->_em->createQuery($dql); $paginator = new Paginator($query, true); $paginator->setUseOutputWalkers($useOutputWalkers); $this->assertCount(3, $paginator->getIterator()); } public function testIterateComplexWithOutputWalker() { $dql = "SELECT g, COUNT(u.id) AS userCount FROM Doctrine\Tests\Models\CMS\CmsGroup g LEFT JOIN g.users u GROUP BY g HAVING COUNT(u.id) > 0"; $query = $this->_em->createQuery($dql); $paginator = new Paginator($query); $paginator->setUseOutputWalkers(true); $this->assertCount(9, $paginator->getIterator()); } public function testDetectOutputWalker() { // This query works using the output walkers but causes an exception using the TreeWalker $dql = "SELECT g, COUNT(u.id) AS userCount FROM Doctrine\Tests\Models\CMS\CmsGroup g LEFT JOIN g.users u GROUP BY g HAVING COUNT(u.id) > 0"; $query = $this->_em->createQuery($dql); // If the Paginator detects the custom output walker it should fall back to using the // Tree walkers for pagination, which leads to an exception. If the query works, the output walkers were used $query->setHint(Query::HINT_CUSTOM_OUTPUT_WALKER, 'Doctrine\ORM\Query\SqlWalker'); $paginator = new Paginator($query); $this->setExpectedException( 'RuntimeException', 'Cannot count query that uses a HAVING clause. Use the output walkers for pagination' ); count($paginator); } public function testCloneQuery() { $dql = "SELECT u FROM Doctrine\Tests\Models\CMS\CmsUser u"; $query = $this->_em->createQuery($dql); $paginator = new Paginator($query); $paginator->getIterator(); $this->assertTrue($query->getParameters()->isEmpty()); } public function testQueryWalkerIsKept() { $dql = "SELECT u FROM Doctrine\Tests\Models\CMS\CmsUser u"; $query = $this->_em->createQuery($dql); $query->setHint(Query::HINT_CUSTOM_TREE_WALKERS, array('Doctrine\Tests\ORM\Functional\CustomPaginationTestTreeWalker')); $paginator = new Paginator($query, true); $paginator->setUseOutputWalkers(false); $this->assertCount(1, $paginator->getIterator()); $this->assertEquals(1, $paginator->count()); } public function testCountQueryStripsParametersInSelect() { /** @var $query Query */ $query = $this->_em->createQuery( 'SELECT u, (CASE WHEN u.id < :vipMaxId THEN 1 ELSE 0 END) AS hidden promotedFirst FROM Doctrine\Tests\Models\CMS\CmsUser u WHERE u.id < :id or 1=1 ORDER BY promotedFirst' ); $query->setParameter('vipMaxId', 10); $query->setParameter('id', 100); $query->setFirstResult(null)->setMaxResults(null); $paginator = new Paginator($query); // $getCountQuery = new ReflectionMethod($paginator, 'getCountQuery'); $getCountQuery->setAccessible(true); $countQuery = $getCountQuery->invoke($paginator); $this->assertEquals(2, count($countQuery->getParameters())); $this->assertEquals(3, $paginator->count()); $query->setHint(Query::HINT_CUSTOM_OUTPUT_WALKER, 'Doctrine\ORM\Query\SqlWalker'); $paginator = new Paginator($query); $countQuery = $getCountQuery->invoke($paginator); //if select part of query is replaced with count(...) paginator should remove parameters from query object not used in new query. $this->assertEquals(1, count($countQuery->getParameters())); $this->assertEquals(3, $paginator->count()); } public function populate() { for ($i = 0; $i < 3; $i++) { $user = new CmsUser(); $user->name = "Name$i"; $user->username = "username$i"; $user->status = "active"; $this->_em->persist($user); for ($j = 0; $j < 3; $j++) {; $group = new CmsGroup(); $group->name = "group$j"; $user->addGroup($group); $this->_em->persist($group); } } $this->_em->flush(); } public function useOutputWalkers() { return array( array(true), array(false), ); } } class CustomPaginationTestTreeWalker extends Query\TreeWalkerAdapter { public function walkSelectStatement(Query\AST\SelectStatement $selectStatement) { $condition = new Query\AST\ConditionalPrimary(); $path = new Query\AST\PathExpression(Query\AST\PathExpression::TYPE_STATE_FIELD, 'u', 'name'); $path->type = Query\AST\PathExpression::TYPE_STATE_FIELD; $condition->simpleConditionalExpression = new Query\AST\ComparisonExpression( $path, '=', new Query\AST\Literal(Query\AST\Literal::STRING, 'Name1') ); $selectStatement->whereClause = new Query\AST\WhereClause($condition); } }