diff --git a/lib/Doctrine/Hydrate.php b/lib/Doctrine/Hydrate.php index 240611ff4..5d4e35a40 100644 --- a/lib/Doctrine/Hydrate.php +++ b/lib/Doctrine/Hydrate.php @@ -361,8 +361,11 @@ class Doctrine_Hydrate extends Doctrine_Object implements Serializable if ( ! isset($this->parts[$name])) { throw new Doctrine_Hydrate_Exception('Unknown query part ' . $name); } - $this->parts[$name][] = $part; - + if (is_array($part)) { + $this->parts[$name] = array_merge($this->parts[$name], $part); + } else { + $this->parts[$name][] = $part; + } return $this; } /** @@ -419,7 +422,11 @@ class Doctrine_Hydrate extends Doctrine_Object implements Serializable } if ($name !== 'limit' && $name !== 'offset') { - $this->parts[$name] = array($part); + if (is_array($part)) { + $this->parts[$name] = $part; + } else { + $this->parts[$name] = array($part); + } } else { $this->parts[$name] = $part; } diff --git a/lib/Doctrine/Query.php b/lib/Doctrine/Query.php index eec1a5d01..382e9b946 100644 --- a/lib/Doctrine/Query.php +++ b/lib/Doctrine/Query.php @@ -848,19 +848,41 @@ class Doctrine_Query extends Doctrine_Query_Abstract implements Countable // initialize the base of the subquery $subquery = 'SELECT DISTINCT ' . $primaryKey; - if ($this->_conn->getAttribute(Doctrine::ATTR_DRIVER_NAME) == 'pgsql') { - // pgsql needs the order by fields to be preserved in select clause + $driverName = $this->_conn->getAttribute(Doctrine::ATTR_DRIVER_NAME); - foreach ($this->parts['orderby'] as $part) { - $e = explode(' ', $part); - // don't add primarykey column (its already in the select clause) - if ($e[0] !== $primaryKey) { - $subquery .= ', ' . $e[0]; + + + foreach ($this->parts['orderby'] as $part) { + $part = trim($part); + $aggregate = false; + + foreach ($this->parts['select'] as $select) { + $e = explode(' AS ', trim($select)); + + if (count($e) > 1) { + if ($part === $e[1]) { + $part = $select; + $aggregate = true; + break; + } + } + } + + // don't add primarykey column (its already in the select clause) + if ($part !== $primaryKey) { + if ($driverName == 'mysql') { + if ($aggregate) { + $subquery .= ', ' . $part; + } + } elseif ($driverName == 'pgsql') { + // pgsql needs the order by fields to be preserved in select clause + $subquery .= ', ' . $part; } } } + $subquery .= ' FROM'; @@ -889,23 +911,32 @@ class Doctrine_Query extends Doctrine_Query_Abstract implements Countable $parts = Doctrine_Tokenizer::quoteExplode($subquery, ' ', "'", "'"); - foreach($parts as $k => $part) { - if(strpos($part, "'") !== false) { + foreach ($parts as $k => $part) { + if (strpos($part, "'") !== false) { continue; } - if($this->hasTableAlias($part)) { + if ($this->hasTableAlias($part)) { $parts[$k] = $this->generateNewTableAlias($part); } - if(strpos($part, '.') !== false) { - $e = explode('.', $part); + $separator = false; + + if (strpos($part, '.') !== false) { + $separator = '.'; + } + if (strpos($part, '__') !== false) { + $separator = '__'; + } + + if ($separator) { + $e = explode($separator, $part); $trimmed = ltrim($e[0], '( '); $pos = strpos($e[0], $trimmed); $e[0] = substr($e[0], 0, $pos) . $this->generateNewTableAlias($trimmed); - $parts[$k] = implode('.', $e); + $parts[$k] = implode($separator, $e); } } $subquery = implode(' ', $parts); diff --git a/lib/Doctrine/Query/Orderby.php b/lib/Doctrine/Query/Orderby.php index 5e6566541..1049b2234 100644 --- a/lib/Doctrine/Query/Orderby.php +++ b/lib/Doctrine/Query/Orderby.php @@ -69,6 +69,6 @@ class Doctrine_Query_Orderby extends Doctrine_Query_Part } $ret[] = $r; } - return implode(', ', $ret); + return $ret; } } diff --git a/tests/Query/MysqlSubqueryTestCase.php b/tests/Query/MysqlSubqueryTestCase.php index 075780905..0e94c3fd0 100644 --- a/tests/Query/MysqlSubqueryTestCase.php +++ b/tests/Query/MysqlSubqueryTestCase.php @@ -1,32 +1,72 @@ . + */ + +/** + * Doctrine_Query_MysqlSubquery_TestCase + * + * @package Doctrine + * @author Konsta Vesterinen + * @license http://www.opensource.org/licenses/lgpl-license.php LGPL + * @category Object Relational Mapping + * @link www.phpdoctrine.com + * @since 1.0 + * @version $Revision$ + */ class Doctrine_Query_MysqlSubquery_TestCase extends Doctrine_UnitTestCase { - public function testInit() + public function setUp() { $this->dbh = new Doctrine_Adapter_Mock('mysql'); $this->conn = Doctrine_Manager::getInstance()->openConnection($this->dbh); } - - public function testGetLimitSubqueryOrderBy() + + public function testGetLimitSubqueryWithOrderByOnAggregateValues() { $q = new Doctrine_Query(); $q->select('u.name, COUNT(DISTINCT a.id) num_albums'); $q->from('User u, u.Album a'); $q->orderby('num_albums'); $q->groupby('u.id'); - // this causes getLimitSubquery() to be used, and it fails $q->limit(5); - - try { - $q->execute(); - } catch (Doctrine_Exception $e) { - $this->fail(); - } - - $queries = $this->dbh->getAll(); - - $this->assertEqual($queries[0], 'SELECT DISTINCT e2.id, COUNT(DISTINCT a.id) AS a__0 FROM entity e2 LEFT JOIN album a2 ON e2.id = a2.user_id WHERE (e2.type = 0) GROUP BY e2.id ORDER BY a__0 LIMIT 5'); + $q->execute(); + $this->dbh->pop(); + + $this->assertEqual($this->dbh->pop(), 'SELECT DISTINCT e2.id, COUNT(DISTINCT a2.id) AS a2__0 FROM entity e2 LEFT JOIN album a2 ON e2.id = a2.user_id WHERE (e2.type = 0) GROUP BY e2.id ORDER BY a2__0 LIMIT 5'); } -} \ No newline at end of file + + public function testGetLimitSubqueryWithOrderByOnAggregateValuesAndColumns() + { + $q = new Doctrine_Query(); + $q->select('u.name, COUNT(DISTINCT a.id) num_albums'); + $q->from('User u, u.Album a'); + $q->orderby('num_albums, u.name'); + $q->groupby('u.id'); + $q->limit(5); + + $q->execute(); + + $this->dbh->pop(); + + $this->assertEqual($this->dbh->pop(), 'SELECT DISTINCT e2.id, COUNT(DISTINCT a2.id) AS a2__0 FROM entity e2 LEFT JOIN album a2 ON e2.id = a2.user_id WHERE (e2.type = 0) GROUP BY e2.id ORDER BY a2__0, e2.name LIMIT 5'); + } +} diff --git a/tests/Query/PgsqlSubqueryTestCase.php b/tests/Query/PgsqlSubqueryTestCase.php new file mode 100644 index 000000000..e9167b0d1 --- /dev/null +++ b/tests/Query/PgsqlSubqueryTestCase.php @@ -0,0 +1,73 @@ +. + */ + +/** + * Doctrine_Query_PgsqlSubquery_TestCase + * + * @package Doctrine + * @author Konsta Vesterinen + * @license http://www.opensource.org/licenses/lgpl-license.php LGPL + * @category Object Relational Mapping + * @link www.phpdoctrine.com + * @since 1.0 + * @version $Revision$ + */ +class Doctrine_Query_PgsqlSubquery_TestCase extends Doctrine_UnitTestCase +{ + public function setUp() + { + $this->dbh = new Doctrine_Adapter_Mock('pgsql'); + $this->conn = Doctrine_Manager::getInstance()->openConnection($this->dbh); + } + /** + public function testGetLimitSubqueryWithOrderByOnAggregateValues() + { + $q = new Doctrine_Query(); + $q->select('u.name, COUNT(DISTINCT a.id) num_albums'); + $q->from('User u, u.Album a'); + $q->orderby('num_albums'); + $q->groupby('u.id'); + $q->limit(5); + + $q->execute(); + + $this->dbh->pop(); + + $this->assertEqual($this->dbh->pop(), 'SELECT DISTINCT e2.id, COUNT(DISTINCT a2.id) AS a2__0 FROM entity e2 LEFT JOIN album a2 ON e2.id = a2.user_id WHERE (e2.type = 0) GROUP BY e2.id ORDER BY a2__0 LIMIT 5'); + } + + public function testGetLimitSubqueryWithOrderByOnAggregateValuesAndColumns() + { + $q = new Doctrine_Query(); + $q->select('u.name, COUNT(DISTINCT a.id) num_albums'); + $q->from('User u, u.Album a'); + $q->orderby('num_albums, u.name'); + $q->groupby('u.id'); + $q->limit(5); + + $q->execute(); + + $this->dbh->pop(); + + $this->assertEqual($this->dbh->pop(), 'SELECT DISTINCT e2.id, COUNT(DISTINCT a2.id) AS a2__0, e2.name FROM entity e2 LEFT JOIN album a2 ON e2.id = a2.user_id WHERE (e2.type = 0) GROUP BY e2.id ORDER BY a2__0, e2.name LIMIT 5'); + } + */ +} diff --git a/tests/TemplateTestCase.php b/tests/TemplateTestCase.php new file mode 100644 index 000000000..591fceda3 --- /dev/null +++ b/tests/TemplateTestCase.php @@ -0,0 +1,39 @@ +. + */ + +/** + * Doctrine_Template_TestCase + * + * @package Doctrine + * @author Konsta Vesterinen + * @license http://www.opensource.org/licenses/lgpl-license.php LGPL + * @category Object Relational Mapping + * @link www.phpdoctrine.com + * @since 1.0 + * @version $Revision$ + */ +class Doctrine_Template_TestCase extends Doctrine_UnitTestCase +{ + public function testTemplateRelationsSupportConcreteInheritance() + { + $blog = new Blog(); + } +} diff --git a/tests/run.php b/tests/run.php index 90abc9ad9..f31558634 100644 --- a/tests/run.php +++ b/tests/run.php @@ -50,11 +50,9 @@ spl_autoload_register(array('Doctrine', 'autoload')); spl_autoload_register('autoload'); require_once dirname(__FILE__) . '/../models/location.php'; +require_once dirname(__FILE__) . '/../models/Blog.php'; require_once dirname(__FILE__) . '/classes.php'; -/** -require_once dirname(__FILE__) . '/../vendor/simpletest/unit_tester.php'; -require_once dirname(__FILE__) . '/../vendor/simpletest/reporter.php'; -*/ + require_once dirname(__FILE__) . '/Test.php'; require_once dirname(__FILE__) . '/UnitTestCase.php'; @@ -72,7 +70,7 @@ $test = new GroupTest('Doctrine Framework Unit Tests'); $test->addTestCase(new Doctrine_Ticket330_TestCase()); */ -/** */ + /** */ // Connection drivers (not yet fully tested) $test->addTestCase(new Doctrine_Connection_Pgsql_TestCase()); $test->addTestCase(new Doctrine_Connection_Oracle_TestCase()); @@ -304,9 +302,12 @@ $test->addTestCase(new Doctrine_Query_TestCase()); $test->addTestCase(new Doctrine_Ticket364_TestCase()); +$test->addTestCase(new Doctrine_Template_TestCase()); + $test->addTestCase(new Doctrine_Query_MysqlSubquery_TestCase()); -/** */ +$test->addTestCase(new Doctrine_Query_PgsqlSubquery_TestCase()); + //$test->addTestCase(new Doctrine_IntegrityAction_TestCase()); //$test->addTestCase(new Doctrine_AuditLog_TestCase());