silverstripe-framework/tests/php/ORM/DataQueryTest.php

333 lines
13 KiB
PHP
Raw Normal View History

<?php
2016-10-14 14:30:05 +13:00
namespace SilverStripe\ORM\Tests;
use SilverStripe\ORM\DataQuery;
use SilverStripe\ORM\DataObject;
use SilverStripe\ORM\DB;
use SilverStripe\Dev\SapphireTest;
2016-10-14 14:30:05 +13:00
use SilverStripe\Security\Member;
2016-11-13 20:35:43 +13:00
/**
* @skipUpgrade
*/
class DataQueryTest extends SapphireTest {
2014-08-15 18:53:05 +12:00
protected static $fixture_file = 'DataQueryTest.yml';
protected $extraDataObjects = array(
2016-10-14 14:30:05 +13:00
DataQueryTest\ObjectA::class,
DataQueryTest\ObjectB::class,
DataQueryTest\ObjectC::class,
DataQueryTest\ObjectD::class,
DataQueryTest\ObjectE::class,
DataQueryTest\ObjectF::class,
DataQueryTest\ObjectG::class,
2016-11-13 20:35:43 +13:00
SQLSelectTest\TestObject::class,
SQLSelectTest\TestBase::class,
SQLSelectTest\TestChild::class,
);
public function testSortByJoinedFieldRetainsSourceInformation() {
2016-10-14 14:30:05 +13:00
$bar = new DataQueryTest\ObjectC();
$bar->Title = "Bar";
$bar->write();
2016-10-14 14:30:05 +13:00
$foo = new DataQueryTest\ObjectB();
$foo->Title = "Foo";
$foo->TestC = $bar->ID;
$foo->write();
2016-10-14 14:30:05 +13:00
$query = new DataQuery(DataQueryTest\ObjectB::class);
$result = $query->leftJoin(
2016-11-13 20:35:43 +13:00
'DataQueryTest_C',
"\"DataQueryTest_B\".\"TestCID\" = \"DataQueryTest_B\".\"ID\""
)->sort('"DataQueryTest_B"."Title"', 'ASC');
$result = $result->execute()->record();
$this->assertEquals('Foo', $result['Title']);
}
/**
* Test the leftJoin() and innerJoin method of the DataQuery object
*/
public function testJoins() {
2016-10-14 14:30:05 +13:00
$dq = new DataQuery(Member::class);
$dq->innerJoin("Group_Members", "\"Group_Members\".\"MemberID\" = \"Member\".\"ID\"");
$this->assertSQLContains("INNER JOIN \"Group_Members\" ON \"Group_Members\".\"MemberID\" = \"Member\".\"ID\"",
$dq->sql($parameters));
2016-10-14 14:30:05 +13:00
$dq = new DataQuery(Member::class);
$dq->leftJoin("Group_Members", "\"Group_Members\".\"MemberID\" = \"Member\".\"ID\"");
$this->assertSQLContains("LEFT JOIN \"Group_Members\" ON \"Group_Members\".\"MemberID\" = \"Member\".\"ID\"",
$dq->sql($parameters));
}
public function testApplyRelation() {
// Test applyRelation with two has_ones pointing to the same class
2016-10-14 14:30:05 +13:00
$dq = new DataQuery(DataQueryTest\ObjectB::class);
$dq->applyRelation('TestC');
2016-11-13 20:35:43 +13:00
$this->assertTrue($dq->query()->isJoinedTo('DataQueryTest_C'));
$this->assertContains('"DataQueryTest_C"."ID" = "DataQueryTest_B"."TestCID"', $dq->sql());
2016-10-14 14:30:05 +13:00
$dq = new DataQuery(DataQueryTest\ObjectB::class);
$dq->applyRelation('TestCTwo');
2016-11-13 20:35:43 +13:00
$this->assertTrue($dq->query()->isJoinedTo('DataQueryTest_C'));
$this->assertContains('"DataQueryTest_C"."ID" = "DataQueryTest_B"."TestCTwoID"', $dq->sql());
}
public function testApplyReplationDeepInheretence() {
//test has_one relation
2016-10-14 14:30:05 +13:00
$newDQ = new DataQuery(DataQueryTest\ObjectE::class);
//apply a relation to a relation from an ancestor class
$newDQ->applyRelation('TestA');
2016-11-13 20:35:43 +13:00
$this->assertTrue($newDQ->query()->isJoinedTo('DataQueryTest_C'));
2014-11-19 11:05:07 +13:00
$this->assertContains('"DataQueryTest_A"."ID" = "DataQueryTest_C"."TestAID"', $newDQ->sql($params));
//test many_many relation
//test many_many with separate inheritance
2016-10-14 14:30:05 +13:00
$newDQ = new DataQuery(DataQueryTest\ObjectC::class);
$baseDBTable = DataObject::getSchema()->baseDataTable(DataQueryTest\ObjectC::class);
$newDQ->applyRelation('ManyTestAs');
//check we are "joined" to the DataObject's table (there is no distinction between FROM or JOIN clauses)
$this->assertTrue($newDQ->query()->isJoinedTo($baseDBTable));
//check we are explicitly selecting "FROM" the DO's table
$this->assertContains("FROM \"$baseDBTable\"", $newDQ->sql());
//test many_many with shared inheritance
2016-10-14 14:30:05 +13:00
$newDQ = new DataQuery(DataQueryTest\ObjectE::class);
$baseDBTable = DataObject::getSchema()->baseDataTable(DataQueryTest\ObjectE::class);
//check we are "joined" to the DataObject's table (there is no distinction between FROM or JOIN clauses)
$this->assertTrue($newDQ->query()->isJoinedTo($baseDBTable));
//check we are explicitly selecting "FROM" the DO's table
$this->assertContains("FROM \"$baseDBTable\"", $newDQ->sql(), 'The FROM clause is missing from the query');
$newDQ->applyRelation('ManyTestGs');
//confirm we are still joined to the base table
$this->assertTrue($newDQ->query()->isJoinedTo($baseDBTable));
//double check it is the "FROM" clause
$this->assertContains("FROM \"$baseDBTable\"", $newDQ->sql(), 'The FROM clause has been removed from the query');
//another (potentially less crude check) for checking "FROM" clause
$fromTables = $newDQ->query()->getFrom();
$this->assertEquals('"' . $baseDBTable . '"', $fromTables[$baseDBTable]);
}
public function testRelationReturn() {
2016-10-14 14:30:05 +13:00
$dq = new DataQuery(DataQueryTest\ObjectC::class);
$this->assertEquals(DataQueryTest\ObjectA::class, $dq->applyRelation('TestA'),
'DataQuery::applyRelation should return the name of the related object.');
2016-10-14 14:30:05 +13:00
$this->assertEquals(DataQueryTest\ObjectA::class, $dq->applyRelation('TestAs'),
'DataQuery::applyRelation should return the name of the related object.');
2016-10-14 14:30:05 +13:00
$this->assertEquals(DataQueryTest\ObjectA::class, $dq->applyRelation('ManyTestAs'),
'DataQuery::applyRelation should return the name of the related object.');
2016-10-14 14:30:05 +13:00
$this->assertEquals(DataQueryTest\ObjectB::class, $dq->applyRelation('TestB'),
'DataQuery::applyRelation should return the name of the related object.');
2016-10-14 14:30:05 +13:00
$this->assertEquals(DataQueryTest\ObjectB::class, $dq->applyRelation('TestBs'),
'DataQuery::applyRelation should return the name of the related object.');
2016-10-14 14:30:05 +13:00
$this->assertEquals(DataQueryTest\ObjectB::class, $dq->applyRelation('ManyTestBs'),
'DataQuery::applyRelation should return the name of the related object.');
2016-10-14 14:30:05 +13:00
$newDQ = new DataQuery(DataQueryTest\ObjectE::class);
$this->assertEquals(DataQueryTest\ObjectA::class, $newDQ->applyRelation('TestA'),
'DataQuery::applyRelation should return the name of the related object.');
}
public function testRelationOrderWithCustomJoin() {
2016-10-14 14:30:05 +13:00
$dataQuery = new DataQuery(DataQueryTest\ObjectB::class);
2016-11-13 20:35:43 +13:00
$dataQuery->innerJoin('DataQueryTest_D', '"DataQueryTest_D"."RelationID" = "DataQueryTest_B"."ID"');
$dataQuery->execute();
}
public function testDisjunctiveGroup() {
2016-10-14 14:30:05 +13:00
$dq = new DataQuery(DataQueryTest\ObjectA::class);
$dq->where('DataQueryTest_A.ID = 2');
$subDq = $dq->disjunctiveGroup();
$subDq->where('DataQueryTest_A.Name = \'John\'');
$subDq->where('DataQueryTest_A.Name = \'Bob\'');
$this->assertSQLContains(
2014-08-15 18:53:05 +12:00
"WHERE (DataQueryTest_A.ID = 2) AND ((DataQueryTest_A.Name = 'John') OR (DataQueryTest_A.Name = 'Bob'))",
$dq->sql($parameters)
2012-10-03 18:08:34 +02:00
);
}
public function testConjunctiveGroup() {
2016-10-14 14:30:05 +13:00
$dq = new DataQuery(DataQueryTest\ObjectA::class);
$dq->where('DataQueryTest_A.ID = 2');
$subDq = $dq->conjunctiveGroup();
$subDq->where('DataQueryTest_A.Name = \'John\'');
$subDq->where('DataQueryTest_A.Name = \'Bob\'');
$this->assertSQLContains(
2014-08-15 18:53:05 +12:00
"WHERE (DataQueryTest_A.ID = 2) AND ((DataQueryTest_A.Name = 'John') AND (DataQueryTest_A.Name = 'Bob'))",
$dq->sql($parameters)
2012-10-03 18:08:34 +02:00
);
}
/**
* @todo Test paramaterised
*/
public function testNestedGroups() {
2016-10-14 14:30:05 +13:00
$dq = new DataQuery(DataQueryTest\ObjectA::class);
$dq->where('DataQueryTest_A.ID = 2');
$subDq = $dq->disjunctiveGroup();
$subDq->where('DataQueryTest_A.Name = \'John\'');
$subSubDq = $subDq->conjunctiveGroup();
$subSubDq->where('DataQueryTest_A.Age = 18');
$subSubDq->where('DataQueryTest_A.Age = 50');
$subDq->where('DataQueryTest_A.Name = \'Bob\'');
$this->assertSQLContains(
2012-10-03 18:08:34 +02:00
"WHERE (DataQueryTest_A.ID = 2) AND ((DataQueryTest_A.Name = 'John') OR ((DataQueryTest_A.Age = 18) "
2014-08-15 18:53:05 +12:00
. "AND (DataQueryTest_A.Age = 50)) OR (DataQueryTest_A.Name = 'Bob'))",
$dq->sql($parameters)
2012-10-03 18:08:34 +02:00
);
}
public function testEmptySubgroup() {
2016-10-14 14:30:05 +13:00
$dq = new DataQuery(DataQueryTest\ObjectA::class);
$dq->conjunctiveGroup();
// Empty groups should have no where condition at all
$this->assertSQLNotContains('WHERE', $dq->sql($parameters));
}
public function testSubgroupHandoff() {
2016-10-14 14:30:05 +13:00
$dq = new DataQuery(DataQueryTest\ObjectA::class);
$subDq = $dq->disjunctiveGroup();
$orgDq = clone $dq;
$subDq->sort('"DataQueryTest_A"."Name"');
$orgDq->sort('"DataQueryTest_A"."Name"');
$this->assertSQLEquals($dq->sql($parameters), $orgDq->sql($parameters));
$subDq->limit(5, 7);
$orgDq->limit(5, 7);
$this->assertSQLEquals($dq->sql($parameters), $orgDq->sql($parameters));
}
2014-08-15 18:53:05 +12:00
public function testOrderByMultiple() {
2016-11-13 20:35:43 +13:00
$dq = new DataQuery(SQLSelectTest\TestObject::class);
$dq = $dq->sort('"Name" ASC, MID("Name", 8, 1) DESC');
$this->assertContains(
2015-07-31 17:18:49 +12:00
'ORDER BY "SQLSelectTest_DO"."Name" ASC, "_SortColumn0" DESC',
$dq->sql($parameters)
);
}
2014-08-15 18:53:05 +12:00
public function testDefaultSort() {
2016-10-14 14:30:05 +13:00
$query = new DataQuery(DataQueryTest\ObjectE::class);
$result = $query->column('Title');
$this->assertEquals(array('First', 'Second', 'Last'), $result);
}
public function testDistinct() {
2016-10-14 14:30:05 +13:00
$query = new DataQuery(DataQueryTest\ObjectE::class);
2014-11-19 11:05:07 +13:00
$this->assertContains('SELECT DISTINCT', $query->sql($params), 'Query is set as distinct by default');
$query = $query->distinct(false);
2014-11-19 11:05:07 +13:00
$this->assertNotContains('SELECT DISTINCT', $query->sql($params), 'Query does not contain distinct');
$query = $query->distinct(true);
2014-11-19 11:05:07 +13:00
$this->assertContains('SELECT DISTINCT', $query->sql($params), 'Query contains distinct');
}
public function testComparisonClauseInt() {
DB::query("INSERT INTO \"DataQueryTest_F\" (\"SortOrder\") VALUES (2)");
2016-10-14 14:30:05 +13:00
$query = new DataQuery(DataQueryTest\ObjectF::class);
$query->where(DB::get_conn()->comparisonClause('"SortOrder"', '2'));
$this->assertGreaterThan(0, $query->count(), "Couldn't find SortOrder");
$this->resetDBSchema(true);
}
public function testComparisonClauseDateFull() {
DB::query("INSERT INTO \"DataQueryTest_F\" (\"MyDate\") VALUES ('1988-03-04 06:30')");
2016-10-14 14:30:05 +13:00
$query = new DataQuery(DataQueryTest\ObjectF::class);
$query->where(DB::get_conn()->comparisonClause('"MyDate"', '1988-03-04%'));
$this->assertGreaterThan(0, $query->count(), "Couldn't find MyDate");
$this->resetDBSchema(true);
}
public function testComparisonClauseDateStartsWith() {
DB::query("INSERT INTO \"DataQueryTest_F\" (\"MyDate\") VALUES ('1988-03-04 06:30')");
2016-10-14 14:30:05 +13:00
$query = new DataQuery(DataQueryTest\ObjectF::class);
$query->where(DB::get_conn()->comparisonClause('"MyDate"', '1988%'));
$this->assertGreaterThan(0, $query->count(), "Couldn't find MyDate");
$this->resetDBSchema(true);
}
public function testComparisonClauseDateStartsPartial() {
DB::query("INSERT INTO \"DataQueryTest_F\" (\"MyDate\") VALUES ('1988-03-04 06:30')");
2016-10-14 14:30:05 +13:00
$query = new DataQuery(DataQueryTest\ObjectF::class);
$query->where(DB::get_conn()->comparisonClause('"MyDate"', '%03-04%'));
$this->assertGreaterThan(0, $query->count(), "Couldn't find MyDate");
$this->resetDBSchema(true);
}
public function testComparisonClauseTextCaseInsensitive() {
DB::query("INSERT INTO \"DataQueryTest_F\" (\"MyString\") VALUES ('HelloWorld')");
2016-10-14 14:30:05 +13:00
$query = new DataQuery(DataQueryTest\ObjectF::class);
$query->where(DB::get_conn()->comparisonClause('"MyString"', 'helloworld'));
$this->assertGreaterThan(0, $query->count(), "Couldn't find MyString");
$this->resetDBSchema(true);
}
public function testComparisonClauseTextCaseSensitive() {
DB::query("INSERT INTO \"DataQueryTest_F\" (\"MyString\") VALUES ('HelloWorld')");
2016-10-14 14:30:05 +13:00
$query = new DataQuery(DataQueryTest\ObjectF::class);
$query->where(DB::get_conn()->comparisonClause('"MyString"', 'HelloWorld', false, false, true));
$this->assertGreaterThan(0, $query->count(), "Couldn't find MyString");
2016-10-14 14:30:05 +13:00
$query2 = new DataQuery(DataQueryTest\ObjectF::class);
$query2->where(DB::get_conn()->comparisonClause('"MyString"', 'helloworld', false, false, true));
$this->assertEquals(0, $query2->count(), "Found mystring. Shouldn't be able too.");
$this->resetDBSchema(true);
}
/**
* Tests that getFinalisedQuery can include all tables
*/
public function testConditionsIncludeTables() {
// Including filter on parent table only doesn't pull in second
2016-10-14 14:30:05 +13:00
$query = new DataQuery(DataQueryTest\ObjectC::class);
$query->sort('"SortOrder"');
$query->where(array(
'"DataQueryTest_C"."Title" = ?' => array('First')
));
$result = $query->getFinalisedQuery(array('Title'));
$from = $result->getFrom();
2016-11-13 20:35:43 +13:00
$this->assertContains('DataQueryTest_C', array_keys($from));
$this->assertNotContains('DataQueryTest_E', array_keys($from));
// Including filter on sub-table requires it
2016-10-14 14:30:05 +13:00
$query = new DataQuery(DataQueryTest\ObjectC::class);
$query->sort('"SortOrder"');
$query->where(array(
'"DataQueryTest_C"."Title" = ? OR "DataQueryTest_E"."SortOrder" > ?' => array(
'First', 2
)
));
$result = $query->getFinalisedQuery(array('Title'));
$from = $result->getFrom();
// Check that including "SortOrder" prompted inclusion of DataQueryTest_E table
2016-11-13 20:35:43 +13:00
$this->assertContains('DataQueryTest_C', array_keys($from));
$this->assertContains('DataQueryTest_E', array_keys($from));
$arrayResult = iterator_to_array($result->execute());
$first = array_shift($arrayResult);
$this->assertNotNull($first);
$this->assertEquals('First', $first['Title']);
$second = array_shift($arrayResult);
$this->assertNotNull($second);
$this->assertEquals('Last', $second['Title']);
$this->assertEmpty(array_shift($arrayResult));
}
}