2009-11-22 18:16:38 +13:00
|
|
|
<?php
|
|
|
|
|
|
|
|
class DataQueryTest extends SapphireTest {
|
2014-04-09 08:46:52 +12:00
|
|
|
|
|
|
|
protected static $fixture_file = 'DataQueryTest.yml';
|
2012-12-10 18:11:07 +13:00
|
|
|
|
|
|
|
protected $extraDataObjects = array(
|
|
|
|
'DataQueryTest_A',
|
|
|
|
'DataQueryTest_B',
|
2014-04-09 08:46:52 +12:00
|
|
|
'DataQueryTest_C',
|
2012-12-10 18:11:07 +13:00
|
|
|
'DataQueryTest_D',
|
2014-04-09 08:46:52 +12:00
|
|
|
'DataQueryTest_E',
|
2014-08-05 14:46:06 +12:00
|
|
|
'DataQueryTest_F',
|
2012-12-10 18:11:07 +13:00
|
|
|
);
|
|
|
|
|
2014-08-26 14:56:45 +12:00
|
|
|
|
|
|
|
public function testSortByJoinedFieldRetainsSourceInformation() {
|
|
|
|
$bar = new DataQueryTest_C();
|
|
|
|
$bar->Title = "Bar";
|
|
|
|
$bar->write();
|
|
|
|
|
|
|
|
$foo = new DataQueryTest_B();
|
|
|
|
$foo->Title = "Foo";
|
|
|
|
$foo->TestC = $bar->ID;
|
|
|
|
$foo->write();
|
|
|
|
|
|
|
|
$query = new DataQuery('DataQueryTest_B');
|
|
|
|
$result = $query->leftJoin(
|
|
|
|
'DataQueryTest_C',
|
|
|
|
"\"DataQueryTest_B\".\"TestCID\" = \"DataQueryTest_B\".\"ID\""
|
|
|
|
)->sort('"DataQueryTest_B"."Title"', 'ASC');
|
|
|
|
|
|
|
|
$result = $result->execute()->record();
|
|
|
|
$this->assertEquals('Foo', $result['Title']);
|
|
|
|
}
|
|
|
|
|
2009-11-22 18:16:38 +13:00
|
|
|
/**
|
2011-10-29 17:11:27 +13:00
|
|
|
* Test the leftJoin() and innerJoin method of the DataQuery object
|
2009-11-22 18:16:38 +13:00
|
|
|
*/
|
2012-09-19 12:07:39 +02:00
|
|
|
public function testJoins() {
|
2009-11-22 18:16:38 +13:00
|
|
|
$dq = new DataQuery('Member');
|
2011-10-29 17:11:27 +13:00
|
|
|
$dq->innerJoin("Group_Members", "\"Group_Members\".\"MemberID\" = \"Member\".\"ID\"");
|
2012-09-27 09:34:00 +12:00
|
|
|
$this->assertContains("INNER JOIN \"Group_Members\" ON \"Group_Members\".\"MemberID\" = \"Member\".\"ID\"",
|
|
|
|
$dq->sql());
|
2011-10-29 17:11:27 +13:00
|
|
|
|
|
|
|
$dq = new DataQuery('Member');
|
|
|
|
$dq->leftJoin("Group_Members", "\"Group_Members\".\"MemberID\" = \"Member\".\"ID\"");
|
2012-09-27 09:34:00 +12:00
|
|
|
$this->assertContains("LEFT JOIN \"Group_Members\" ON \"Group_Members\".\"MemberID\" = \"Member\".\"ID\"",
|
|
|
|
$dq->sql());
|
2009-11-22 18:16:38 +13:00
|
|
|
}
|
2012-06-25 11:34:02 +12:00
|
|
|
|
2014-11-10 17:32:58 +01:00
|
|
|
public function testApplyReplationDeepInheretence() {
|
|
|
|
$newDQ = new DataQuery('DataQueryTest_E');
|
|
|
|
//apply a relation to a relation from an ancestor class
|
|
|
|
$newDQ->applyRelation('TestA');
|
|
|
|
$this->assertTrue($newDQ->query()->isJoinedTo('DataQueryTest_C'));
|
|
|
|
$this->assertContains('"DataQueryTest_A"."ID" = "DataQueryTest_C"."TestAID"', $newDQ->sql());
|
|
|
|
}
|
|
|
|
|
2012-09-19 12:07:39 +02:00
|
|
|
public function testRelationReturn() {
|
2012-06-25 11:34:02 +12:00
|
|
|
$dq = new DataQuery('DataQueryTest_C');
|
2012-09-27 09:34:00 +12:00
|
|
|
$this->assertEquals('DataQueryTest_A', $dq->applyRelation('TestA'),
|
|
|
|
'DataQuery::applyRelation should return the name of the related object.');
|
|
|
|
$this->assertEquals('DataQueryTest_A', $dq->applyRelation('TestAs'),
|
|
|
|
'DataQuery::applyRelation should return the name of the related object.');
|
|
|
|
$this->assertEquals('DataQueryTest_A', $dq->applyRelation('ManyTestAs'),
|
|
|
|
'DataQuery::applyRelation should return the name of the related object.');
|
2012-06-25 11:34:02 +12:00
|
|
|
|
2012-09-27 09:34:00 +12:00
|
|
|
$this->assertEquals('DataQueryTest_B', $dq->applyRelation('TestB'),
|
|
|
|
'DataQuery::applyRelation should return the name of the related object.');
|
|
|
|
$this->assertEquals('DataQueryTest_B', $dq->applyRelation('TestBs'),
|
|
|
|
'DataQuery::applyRelation should return the name of the related object.');
|
|
|
|
$this->assertEquals('DataQueryTest_B', $dq->applyRelation('ManyTestBs'),
|
|
|
|
'DataQuery::applyRelation should return the name of the related object.');
|
2014-11-10 17:32:58 +01:00
|
|
|
$newDQ = new DataQuery('DataQueryTest_E');
|
|
|
|
$this->assertEquals('DataQueryTest_A', $newDQ->applyRelation('TestA'),
|
|
|
|
'DataQuery::applyRelation should return the name of the related object.');
|
2012-06-25 11:34:02 +12:00
|
|
|
}
|
2012-09-20 23:28:54 +12:00
|
|
|
|
2012-12-10 18:11:07 +13:00
|
|
|
public function testRelationOrderWithCustomJoin() {
|
|
|
|
$dataQuery = new DataQuery('DataQueryTest_B');
|
|
|
|
$dataQuery->innerJoin('DataQueryTest_D', '"DataQueryTest_D"."RelationID" = "DataQueryTest_B"."ID"');
|
|
|
|
$dataQuery->execute();
|
|
|
|
}
|
2012-12-14 10:57:28 +01:00
|
|
|
|
2012-09-20 23:28:54 +12:00
|
|
|
public function testDisjunctiveGroup() {
|
|
|
|
$dq = new DataQuery('DataQueryTest_A');
|
|
|
|
|
|
|
|
$dq->where('DataQueryTest_A.ID = 2');
|
|
|
|
$subDq = $dq->disjunctiveGroup();
|
|
|
|
$subDq->where('DataQueryTest_A.Name = \'John\'');
|
|
|
|
$subDq->where('DataQueryTest_A.Name = \'Bob\'');
|
|
|
|
|
2012-10-03 18:08:34 +02:00
|
|
|
$this->assertContains(
|
|
|
|
"WHERE (DataQueryTest_A.ID = 2) AND ((DataQueryTest_A.Name = 'John') OR (DataQueryTest_A.Name = 'Bob'))",
|
|
|
|
$dq->sql()
|
|
|
|
);
|
2012-09-20 23:28:54 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testConjunctiveGroup() {
|
|
|
|
$dq = new DataQuery('DataQueryTest_A');
|
|
|
|
|
|
|
|
$dq->where('DataQueryTest_A.ID = 2');
|
|
|
|
$subDq = $dq->conjunctiveGroup();
|
|
|
|
$subDq->where('DataQueryTest_A.Name = \'John\'');
|
|
|
|
$subDq->where('DataQueryTest_A.Name = \'Bob\'');
|
|
|
|
|
2012-10-03 18:08:34 +02:00
|
|
|
$this->assertContains(
|
|
|
|
"WHERE (DataQueryTest_A.ID = 2) AND ((DataQueryTest_A.Name = 'John') AND (DataQueryTest_A.Name = 'Bob'))",
|
|
|
|
$dq->sql()
|
|
|
|
);
|
2012-09-20 23:28:54 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testNestedGroups() {
|
|
|
|
$dq = new DataQuery('DataQueryTest_A');
|
|
|
|
|
|
|
|
$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\'');
|
|
|
|
|
2012-10-03 18:08:34 +02:00
|
|
|
$this->assertContains(
|
|
|
|
"WHERE (DataQueryTest_A.ID = 2) AND ((DataQueryTest_A.Name = 'John') OR ((DataQueryTest_A.Age = 18) "
|
|
|
|
. "AND (DataQueryTest_A.Age = 50)) OR (DataQueryTest_A.Name = 'Bob'))",
|
|
|
|
$dq->sql()
|
|
|
|
);
|
2012-09-20 23:28:54 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testEmptySubgroup() {
|
|
|
|
$dq = new DataQuery('DataQueryTest_A');
|
|
|
|
$dq->conjunctiveGroup();
|
|
|
|
|
|
|
|
$this->assertContains('WHERE (1=1)', $dq->sql());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testSubgroupHandoff() {
|
|
|
|
$dq = new DataQuery('DataQueryTest_A');
|
|
|
|
$subDq = $dq->disjunctiveGroup();
|
|
|
|
|
|
|
|
$orgDq = clone $dq;
|
|
|
|
|
|
|
|
$subDq->sort('"DataQueryTest_A"."Name"');
|
|
|
|
$orgDq->sort('"DataQueryTest_A"."Name"');
|
|
|
|
|
|
|
|
$this->assertEquals($dq->sql(), $orgDq->sql());
|
|
|
|
|
|
|
|
$subDq->limit(5, 7);
|
|
|
|
$orgDq->limit(5, 7);
|
|
|
|
|
|
|
|
$this->assertEquals($dq->sql(), $orgDq->sql());
|
|
|
|
}
|
2013-10-03 14:20:31 +13:00
|
|
|
|
|
|
|
public function testOrderByMultiple() {
|
|
|
|
$dq = new DataQuery('SQLQueryTest_DO');
|
|
|
|
$dq = $dq->sort('"Name" ASC, MID("Name", 8, 1) DESC');
|
|
|
|
$this->assertContains(
|
|
|
|
'ORDER BY "Name" ASC, "_SortColumn0" DESC',
|
|
|
|
$dq->sql()
|
|
|
|
);
|
|
|
|
}
|
2014-04-09 08:46:52 +12:00
|
|
|
|
|
|
|
public function testDefaultSort() {
|
|
|
|
$query = new DataQuery('DataQueryTest_E');
|
|
|
|
$result = $query->column('Title');
|
|
|
|
$this->assertEquals(array('First', 'Second', 'Last'), $result);
|
|
|
|
}
|
2014-09-03 18:53:47 +12:00
|
|
|
|
|
|
|
public function testDistinct() {
|
|
|
|
$query = new DataQuery('DataQueryTest_E');
|
|
|
|
$this->assertContains('SELECT DISTINCT', $query->sql(), 'Query is set as distinct by default');
|
|
|
|
|
|
|
|
$query = $query->distinct(false);
|
|
|
|
$this->assertNotContains('SELECT DISTINCT', $query->sql(), 'Query does not contain distinct');
|
|
|
|
|
|
|
|
$query = $query->distinct(true);
|
|
|
|
$this->assertContains('SELECT DISTINCT', $query->sql(), 'Query contains distinct');
|
2014-08-05 14:46:06 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testComparisonClauseInt() {
|
|
|
|
DB::query("INSERT INTO \"DataQueryTest_F\" (\"SortOrder\") VALUES (2)");
|
|
|
|
$query = new DataQuery('DataQueryTest_F');
|
|
|
|
$query->where(DB::getConn()->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')");
|
|
|
|
$query = new DataQuery('DataQueryTest_F');
|
|
|
|
$query->where(DB::getConn()->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')");
|
|
|
|
$query = new DataQuery('DataQueryTest_F');
|
|
|
|
$query->where(DB::getConn()->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')");
|
|
|
|
$query = new DataQuery('DataQueryTest_F');
|
|
|
|
$query->where(DB::getConn()->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')");
|
|
|
|
$query = new DataQuery('DataQueryTest_F');
|
|
|
|
$query->where(DB::getConn()->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')");
|
|
|
|
$query = new DataQuery('DataQueryTest_F');
|
|
|
|
$query->where(DB::getConn()->comparisonClause('"MyString"', 'HelloWorld', false, false, true));
|
|
|
|
$this->assertGreaterThan(0, $query->count(), "Couldn't find MyString");
|
|
|
|
|
|
|
|
$query2 = new DataQuery('DataQueryTest_F');
|
|
|
|
$query2->where(DB::getConn()->comparisonClause('"MyString"', 'helloworld', false, false, true));
|
|
|
|
$this->assertEquals(0, $query2->count(), "Found mystring. Shouldn't be able too.");
|
|
|
|
$this->resetDBSchema(true);
|
2014-09-03 18:53:47 +12:00
|
|
|
}
|
|
|
|
|
2012-06-25 11:34:02 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class DataQueryTest_A extends DataObject implements TestOnly {
|
2014-08-26 14:56:45 +12:00
|
|
|
|
2013-03-21 19:48:54 +01:00
|
|
|
private static $db = array(
|
2012-06-25 11:34:02 +12:00
|
|
|
'Name' => 'Varchar',
|
|
|
|
);
|
|
|
|
|
2013-03-21 19:48:54 +01:00
|
|
|
private static $has_one = array(
|
2012-06-25 11:34:02 +12:00
|
|
|
'TestC' => 'DataQueryTest_C',
|
|
|
|
);
|
2009-11-22 18:16:38 +13:00
|
|
|
}
|
|
|
|
|
2014-08-26 14:56:45 +12:00
|
|
|
class DataQueryTest_B extends DataObject implements TestOnly {
|
|
|
|
|
2013-03-21 19:48:54 +01:00
|
|
|
private static $db = array(
|
2012-06-25 11:34:02 +12:00
|
|
|
'Title' => 'Varchar',
|
|
|
|
);
|
|
|
|
|
2013-03-21 19:48:54 +01:00
|
|
|
private static $has_one = array(
|
2012-06-25 11:34:02 +12:00
|
|
|
'TestC' => 'DataQueryTest_C',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
class DataQueryTest_C extends DataObject implements TestOnly {
|
2014-04-09 08:46:52 +12:00
|
|
|
|
|
|
|
private static $db = array(
|
|
|
|
'Title' => 'Varchar'
|
|
|
|
);
|
2012-12-10 18:11:07 +13:00
|
|
|
|
2013-03-21 19:48:54 +01:00
|
|
|
private static $has_one = array(
|
2012-06-25 11:34:02 +12:00
|
|
|
'TestA' => 'DataQueryTest_A',
|
|
|
|
'TestB' => 'DataQueryTest_B',
|
|
|
|
);
|
|
|
|
|
2013-03-21 19:48:54 +01:00
|
|
|
private static $has_many = array(
|
2012-06-25 11:34:02 +12:00
|
|
|
'TestAs' => 'DataQueryTest_A',
|
|
|
|
'TestBs' => 'DataQueryTest_B',
|
|
|
|
);
|
|
|
|
|
2013-03-21 19:48:54 +01:00
|
|
|
private static $many_many = array(
|
2012-06-25 11:34:02 +12:00
|
|
|
'ManyTestAs' => 'DataQueryTest_A',
|
|
|
|
'ManyTestBs' => 'DataQueryTest_B',
|
|
|
|
);
|
|
|
|
}
|
2012-12-10 18:11:07 +13:00
|
|
|
|
|
|
|
class DataQueryTest_D extends DataObject implements TestOnly {
|
|
|
|
|
2013-03-21 19:48:54 +01:00
|
|
|
private static $has_one = array(
|
2012-12-10 18:11:07 +13:00
|
|
|
'Relation' => 'DataQueryTest_B',
|
|
|
|
);
|
|
|
|
}
|
2014-04-09 08:46:52 +12:00
|
|
|
|
|
|
|
class DataQueryTest_E extends DataQueryTest_C implements TestOnly {
|
|
|
|
|
|
|
|
private static $db = array(
|
|
|
|
'SortOrder' => 'Int'
|
|
|
|
);
|
|
|
|
|
|
|
|
private static $default_sort = '"DataQueryTest_E"."SortOrder" ASC';
|
|
|
|
}
|
2014-08-05 14:46:06 +12:00
|
|
|
|
|
|
|
class DataQueryTest_F extends DataObject implements TestOnly {
|
|
|
|
|
|
|
|
private static $db = array(
|
|
|
|
'SortOrder' => 'Int',
|
|
|
|
'MyDate' => 'SS_Datetime',
|
|
|
|
'MyString' => 'Text'
|
|
|
|
);
|
|
|
|
}
|