mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 12:05:37 +00:00
Merge pull request #3019 from tractorcow/pulls/3.1-dataquery-tables
BUG Fix data query not always joining necessary tables
This commit is contained in:
commit
149b4e4356
@ -149,11 +149,13 @@ class DataQuery {
|
|||||||
}
|
}
|
||||||
|
|
||||||
$query = clone $this->query;
|
$query = clone $this->query;
|
||||||
|
$ancestorTables = ClassInfo::ancestry($this->dataClass, true);
|
||||||
|
|
||||||
// Generate the list of tables to iterate over and the list of columns required by any existing where clauses.
|
// Generate the list of tables to iterate over and the list of columns required
|
||||||
// This second step is skipped if we're fetching the whole dataobject as any required columns will get selected
|
// by any existing where clauses. This second step is skipped if we're fetching
|
||||||
// regardless.
|
// the whole dataobject as any required columns will get selected regardless.
|
||||||
if($queriedColumns) {
|
if($queriedColumns) {
|
||||||
|
// Specifying certain columns allows joining of child tables
|
||||||
$tableClasses = ClassInfo::dataClassesFor($this->dataClass);
|
$tableClasses = ClassInfo::dataClassesFor($this->dataClass);
|
||||||
|
|
||||||
foreach ($query->getWhere() as $where) {
|
foreach ($query->getWhere() as $where) {
|
||||||
@ -163,8 +165,9 @@ class DataQuery {
|
|||||||
if (!in_array($matches[1], $queriedColumns)) $queriedColumns[] = $matches[1];
|
if (!in_array($matches[1], $queriedColumns)) $queriedColumns[] = $matches[1];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
$tableClasses = $ancestorTables;
|
||||||
}
|
}
|
||||||
else $tableClasses = ClassInfo::ancestry($this->dataClass, true);
|
|
||||||
|
|
||||||
$tableNames = array_keys($tableClasses);
|
$tableNames = array_keys($tableClasses);
|
||||||
$baseClass = $tableNames[0];
|
$baseClass = $tableNames[0];
|
||||||
@ -172,30 +175,26 @@ class DataQuery {
|
|||||||
// Iterate over the tables and check what we need to select from them. If any selects are made (or the table is
|
// Iterate over the tables and check what we need to select from them. If any selects are made (or the table is
|
||||||
// required for a select)
|
// required for a select)
|
||||||
foreach($tableClasses as $tableClass) {
|
foreach($tableClasses as $tableClass) {
|
||||||
$joinTable = false;
|
|
||||||
|
|
||||||
// If queriedColumns is set, then check if any of the fields are in this table.
|
// Determine explicit columns to select
|
||||||
|
$selectColumns = null;
|
||||||
if ($queriedColumns) {
|
if ($queriedColumns) {
|
||||||
|
// Restrict queried columns to that on the selected table
|
||||||
$tableFields = DataObject::database_fields($tableClass);
|
$tableFields = DataObject::database_fields($tableClass);
|
||||||
$selectColumns = array();
|
$selectColumns = array_intersect($queriedColumns, array_keys($tableFields));
|
||||||
// Look through columns specifically requested in query (or where clause)
|
}
|
||||||
foreach ($queriedColumns as $queriedColumn) {
|
|
||||||
if (array_key_exists($queriedColumn, $tableFields)) {
|
// If this is a subclass without any explicitly requested columns, omit this from the query
|
||||||
$selectColumns[] = $queriedColumn;
|
if(!in_array($tableClass, $ancestorTables) && empty($selectColumns)) continue;
|
||||||
}
|
|
||||||
}
|
// Select necessary columns (unless an explicitly empty array)
|
||||||
|
if($selectColumns !== array()) {
|
||||||
$this->selectColumnsFromTable($query, $tableClass, $selectColumns);
|
$this->selectColumnsFromTable($query, $tableClass, $selectColumns);
|
||||||
if ($selectColumns && $tableClass != $baseClass) {
|
|
||||||
$joinTable = true;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
$this->selectColumnsFromTable($query, $tableClass);
|
|
||||||
if ($tableClass != $baseClass) $joinTable = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($joinTable) {
|
// Join if not the base table
|
||||||
$query->addLeftJoin($tableClass, "\"$tableClass\".\"ID\" = \"$baseClass\".\"ID\"", $tableClass, 10) ;
|
if($tableClass !== $baseClass) {
|
||||||
|
$query->addLeftJoin($tableClass, "\"$tableClass\".\"ID\" = \"$baseClass\".\"ID\"", $tableClass, 10);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -687,7 +686,8 @@ class DataQuery {
|
|||||||
/**
|
/**
|
||||||
* Query the given field column from the database and return as an array.
|
* Query the given field column from the database and return as an array.
|
||||||
*
|
*
|
||||||
* @param String $field See {@link expressionForField()}.
|
* @param string $field See {@link expressionForField()}.
|
||||||
|
* @return array List of column values for the specified column
|
||||||
*/
|
*/
|
||||||
public function column($field = 'ID') {
|
public function column($field = 'ID') {
|
||||||
$fieldExpression = $this->expressionForField($field);
|
$fieldExpression = $this->expressionForField($field);
|
||||||
|
@ -35,6 +35,7 @@ class DataObjectLazyLoadingTest extends SapphireTest {
|
|||||||
'"DataObjectTest_Team"."ClassName" IS NOT NULL THEN "DataObjectTest_Team"."ClassName" ELSE ' .
|
'"DataObjectTest_Team"."ClassName" IS NOT NULL THEN "DataObjectTest_Team"."ClassName" ELSE ' .
|
||||||
$db->prepStringForDB('DataObjectTest_Team').' END AS "RecordClassName", "DataObjectTest_Team"."Title" '.
|
$db->prepStringForDB('DataObjectTest_Team').' END AS "RecordClassName", "DataObjectTest_Team"."Title" '.
|
||||||
'FROM "DataObjectTest_Team" ' .
|
'FROM "DataObjectTest_Team" ' .
|
||||||
|
'LEFT JOIN "DataObjectTest_SubTeam" ON "DataObjectTest_SubTeam"."ID" = "DataObjectTest_Team"."ID" ' .
|
||||||
'WHERE ("DataObjectTest_Team"."ClassName" IN ('.$db->prepStringForDB('DataObjectTest_SubTeam').'))' .
|
'WHERE ("DataObjectTest_Team"."ClassName" IN ('.$db->prepStringForDB('DataObjectTest_SubTeam').'))' .
|
||||||
' ORDER BY "DataObjectTest_Team"."Title" ASC';
|
' ORDER BY "DataObjectTest_Team"."Title" ASC';
|
||||||
$this->assertEquals($expected, $playerList->sql());
|
$this->assertEquals($expected, $playerList->sql());
|
||||||
@ -62,7 +63,8 @@ class DataObjectLazyLoadingTest extends SapphireTest {
|
|||||||
$expected = 'SELECT DISTINCT "DataObjectTest_Team"."ClassName", "DataObjectTest_Team"."Created", ' .
|
$expected = 'SELECT DISTINCT "DataObjectTest_Team"."ClassName", "DataObjectTest_Team"."Created", ' .
|
||||||
'"DataObjectTest_Team"."LastEdited", "DataObjectTest_Team"."Title", "DataObjectTest_Team"."ID", ' .
|
'"DataObjectTest_Team"."LastEdited", "DataObjectTest_Team"."Title", "DataObjectTest_Team"."ID", ' .
|
||||||
'CASE WHEN "DataObjectTest_Team"."ClassName" IS NOT NULL THEN "DataObjectTest_Team"."ClassName" ELSE ' .
|
'CASE WHEN "DataObjectTest_Team"."ClassName" IS NOT NULL THEN "DataObjectTest_Team"."ClassName" ELSE ' .
|
||||||
$db->prepStringForDB('DataObjectTest_Team').' END AS "RecordClassName" FROM "DataObjectTest_Team" WHERE ' .
|
$db->prepStringForDB('DataObjectTest_Team').' END AS "RecordClassName" FROM "DataObjectTest_Team" ' .
|
||||||
|
'LEFT JOIN "DataObjectTest_SubTeam" ON "DataObjectTest_SubTeam"."ID" = "DataObjectTest_Team"."ID" WHERE ' .
|
||||||
'("DataObjectTest_Team"."ClassName" IN ('.$db->prepStringForDB('DataObjectTest_SubTeam').')) ' .
|
'("DataObjectTest_Team"."ClassName" IN ('.$db->prepStringForDB('DataObjectTest_SubTeam').')) ' .
|
||||||
'ORDER BY "DataObjectTest_Team"."Title" ASC';
|
'ORDER BY "DataObjectTest_Team"."Title" ASC';
|
||||||
$this->assertEquals($expected, $playerList->sql());
|
$this->assertEquals($expected, $playerList->sql());
|
||||||
|
@ -1,11 +1,15 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
class DataQueryTest extends SapphireTest {
|
class DataQueryTest extends SapphireTest {
|
||||||
|
|
||||||
|
protected static $fixture_file = 'DataQueryTest.yml';
|
||||||
|
|
||||||
protected $extraDataObjects = array(
|
protected $extraDataObjects = array(
|
||||||
'DataQueryTest_A',
|
'DataQueryTest_A',
|
||||||
'DataQueryTest_B',
|
'DataQueryTest_B',
|
||||||
|
'DataQueryTest_C',
|
||||||
'DataQueryTest_D',
|
'DataQueryTest_D',
|
||||||
|
'DataQueryTest_E',
|
||||||
);
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -124,6 +128,12 @@ class DataQueryTest extends SapphireTest {
|
|||||||
$dq->sql()
|
$dq->sql()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testDefaultSort() {
|
||||||
|
$query = new DataQuery('DataQueryTest_E');
|
||||||
|
$result = $query->column('Title');
|
||||||
|
$this->assertEquals(array('First', 'Second', 'Last'), $result);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -148,6 +158,10 @@ class DataQueryTest_B extends DataQueryTest_A {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class DataQueryTest_C extends DataObject implements TestOnly {
|
class DataQueryTest_C extends DataObject implements TestOnly {
|
||||||
|
|
||||||
|
private static $db = array(
|
||||||
|
'Title' => 'Varchar'
|
||||||
|
);
|
||||||
|
|
||||||
private static $has_one = array(
|
private static $has_one = array(
|
||||||
'TestA' => 'DataQueryTest_A',
|
'TestA' => 'DataQueryTest_A',
|
||||||
@ -171,3 +185,12 @@ class DataQueryTest_D extends DataObject implements TestOnly {
|
|||||||
'Relation' => 'DataQueryTest_B',
|
'Relation' => 'DataQueryTest_B',
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class DataQueryTest_E extends DataQueryTest_C implements TestOnly {
|
||||||
|
|
||||||
|
private static $db = array(
|
||||||
|
'SortOrder' => 'Int'
|
||||||
|
);
|
||||||
|
|
||||||
|
private static $default_sort = '"DataQueryTest_E"."SortOrder" ASC';
|
||||||
|
}
|
||||||
|
10
tests/model/DataQueryTest.yml
Normal file
10
tests/model/DataQueryTest.yml
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
DataQueryTest_E:
|
||||||
|
query1:
|
||||||
|
Title: 'Last'
|
||||||
|
SortOrder: 3
|
||||||
|
query2:
|
||||||
|
Title: 'First'
|
||||||
|
SortOrder: 1
|
||||||
|
query3:
|
||||||
|
Title: 'Second'
|
||||||
|
SortOrder: 2
|
Loading…
x
Reference in New Issue
Block a user