mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 14:05:37 +02:00
API CHANGE: Add SQLQuery::prepareSelect(), to further remove the need for direct property access.
API CHANGE: Change the format of SQLQuery::$select to use aliases as keys.
This commit is contained in:
parent
a8e8a6060a
commit
2207e3d978
@ -598,7 +598,7 @@ class DataQuery {
|
||||
$subSelect= $subtractQuery->getFinalisedQuery();
|
||||
$fieldExpression = $this->expressionForField($field, $subSelect);
|
||||
$subSelect->clearSelect();
|
||||
$subSelect->selectField($fieldExpression);
|
||||
$subSelect->selectField($fieldExpression, $field);
|
||||
$this->where($this->expressionForField($field, $this).' NOT IN ('.$subSelect->sql().')');
|
||||
|
||||
return $this;
|
||||
@ -624,7 +624,7 @@ class DataQuery {
|
||||
$originalSelect = $query->itemisedSelect();
|
||||
$fieldExpression = $this->expressionForField($field, $query);
|
||||
$query->clearSelect();
|
||||
$query->selectField($fieldExpression);
|
||||
$query->selectField($fieldExpression, $field);
|
||||
$this->ensureSelectContainsOrderbyColumns($query, $originalSelect);
|
||||
|
||||
return $query->execute()->column($field);
|
||||
|
@ -728,9 +728,10 @@ abstract class SS_Database {
|
||||
|
||||
if($sqlQuery->delete) {
|
||||
$text = "DELETE ";
|
||||
} else if($sqlQuery->select) {
|
||||
$text = "SELECT $distinct" . implode(", ", $sqlQuery->select);
|
||||
} else {
|
||||
$text = "SELECT $distinct" . $sqlQuery->prepareSelect();
|
||||
}
|
||||
|
||||
if($sqlQuery->from) $text .= " FROM " . implode(" ", $sqlQuery->from);
|
||||
if($sqlQuery->where) $text .= " WHERE (" . $sqlQuery->prepareWhere(). ")";
|
||||
if($sqlQuery->groupby) $text .= " GROUP BY " . $sqlQuery->prepareGroupBy();
|
||||
|
@ -844,8 +844,8 @@ class MySQLDatabase extends SS_Database {
|
||||
|
||||
// Make column selection lists
|
||||
$select = array(
|
||||
'SiteTree' => array("ClassName","$baseClasses[SiteTree].ID","ParentID","Title","MenuTitle","URLSegment","Content","LastEdited","Created","_utf8'' AS Filename", "_utf8'' AS Name", "$relevance[SiteTree] AS Relevance", "CanViewType"),
|
||||
'File' => array("ClassName","$baseClasses[File].ID","_utf8'' AS ParentID","Title","_utf8'' AS MenuTitle","_utf8'' AS URLSegment","Content","LastEdited","Created","Filename","Name","$relevance[File] AS Relevance","NULL AS CanViewType"),
|
||||
'SiteTree' => array("ClassName","$baseClasses[SiteTree].\"ID\"","ParentID","Title","MenuTitle","URLSegment","Content","LastEdited","Created","Filename" => "_utf8''", "Name" => "_utf8''", "Relevance" => $relevance['SiteTree'], "CanViewType"),
|
||||
'File' => array("ClassName","$baseClasses[File].\"ID\"","ParentID" => "_utf8''","Title","MenuTitle" => "_utf8''","URLSegment" => "_utf8''","Content","LastEdited","Created","Filename","Name", "Relevance" => $relevance['File'], "CanViewType" => "NULL"),
|
||||
);
|
||||
|
||||
// Process and combine queries
|
||||
@ -856,7 +856,7 @@ class MySQLDatabase extends SS_Database {
|
||||
|
||||
// There's no need to do all that joining
|
||||
$query->from = array(str_replace(array('"','`'),'',$baseClasses[$class]) => $baseClasses[$class]);
|
||||
$query->select = $select[$class];
|
||||
$query->select($select[$class]);
|
||||
$query->orderby = null;
|
||||
|
||||
$querySQLs[] = $query->sql();
|
||||
|
@ -137,11 +137,14 @@ class SQLQuery {
|
||||
*/
|
||||
public function select($fields) {
|
||||
if (func_num_args() > 1) {
|
||||
$this->select = func_get_args();
|
||||
} else {
|
||||
$this->select = is_array($fields) ? $fields : array($fields);
|
||||
$fields = func_get_args();
|
||||
} else if(!is_array($fields)) {
|
||||
$fields = array($fields);
|
||||
}
|
||||
|
||||
$this->select = array();
|
||||
$this->selectMore($fields);
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
@ -151,14 +154,20 @@ class SQLQuery {
|
||||
* @param array|string
|
||||
*/
|
||||
public function selectMore($fields) {
|
||||
if(func_num_args() > 1) $fields = func_get_args();
|
||||
|
||||
if(is_array($fields)) {
|
||||
foreach($fields as $field) {
|
||||
$this->select[] = $field;
|
||||
if (func_num_args() > 1) {
|
||||
$fields = func_get_args();
|
||||
} else if(!is_array($fields)) {
|
||||
$fields = array($fields);
|
||||
}
|
||||
|
||||
$this->select = array();
|
||||
foreach($fields as $idx => $field) {
|
||||
if(preg_match('/^(.*) +AS +"?([^"]*)"?/i', $field, $matches)) {
|
||||
Deprecation::notice("3.0", "Use selectField() to specify column aliases");
|
||||
$this->selectField($matches[1], $matches[2]);
|
||||
} else {
|
||||
$this->select[] = $fields;
|
||||
$this->selectField($field, is_numeric($idx) ? null : $idx);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@ -169,13 +178,11 @@ class SQLQuery {
|
||||
* @param $alias The alias of that field
|
||||
*/
|
||||
public function selectField($field, $alias = null) {
|
||||
// Let's not put unnecessary aliases into the query
|
||||
if($alias && preg_match('/"' . preg_quote($alias) . '"$/', $field)) {
|
||||
$alias = null;
|
||||
if(!$alias) {
|
||||
if(preg_match('/"([^"]+)"$/', $field, $matches)) $alias = $matches[1];
|
||||
else $alias = $field;
|
||||
}
|
||||
|
||||
if($alias) $this->select[] = "$field AS \"$alias\"";
|
||||
else $this->select[] = $field;
|
||||
$this->select[$alias] = $field;
|
||||
}
|
||||
|
||||
/**
|
||||
@ -185,13 +192,7 @@ class SQLQuery {
|
||||
* @todo This should be refactored after $this->select is changed to make that easier
|
||||
*/
|
||||
public function expressionForField($field) {
|
||||
foreach($this->select as $sel) {
|
||||
if(preg_match('/AS +"?([^"]*)"?/i', $sel, $matches)) $selField = $matches[1];
|
||||
else if(preg_match('/"([^"]*)"\."([^"]*)"/', $sel, $matches)) $selField = $matches[2];
|
||||
else if(preg_match('/"?([^"]*)"?/', $sel, $matches)) $selField = $matches[2];
|
||||
if($selField == $field) return $sel;
|
||||
}
|
||||
return null;
|
||||
return isset($this->select[$field]) ? $this->select[$field] : null;
|
||||
}
|
||||
|
||||
/**
|
||||
@ -385,7 +386,7 @@ class SQLQuery {
|
||||
$clause = trim($clause);
|
||||
$column = "_SortColumn{$i}";
|
||||
|
||||
$this->select(sprintf("%s AS \"%s\"", $clause, $column));
|
||||
$this->selectField($clause, $column);
|
||||
$this->orderby($column, $dir, false);
|
||||
$i++;
|
||||
}
|
||||
@ -587,21 +588,21 @@ class SQLQuery {
|
||||
* E.g., 'Title' => '"SiteTree"."Title"'.
|
||||
*/
|
||||
public function itemisedSelect() {
|
||||
$output = array();
|
||||
foreach($this->select as $item) {
|
||||
$split = preg_split('/ AS /i', $item, 2);
|
||||
if(isset($split[1])) {
|
||||
$source = $split[0];
|
||||
$alias = $split[1];
|
||||
} else {
|
||||
$source = $item;
|
||||
$alias = substr($item,strrpos($item,'.')+1);
|
||||
return $this->select;
|
||||
}
|
||||
// Drop quoting
|
||||
$alias = preg_replace('/^"(.*)"$/','\\1', $alias);
|
||||
$output[$alias] = $source;
|
||||
|
||||
/**
|
||||
* Returns the WHERE clauses ready for inserting into a query.
|
||||
* @return string
|
||||
*/
|
||||
public function prepareSelect() {
|
||||
$clauses = array();
|
||||
foreach($this->select as $alias => $field) {
|
||||
// Don't include redundant aliases.
|
||||
if($alias === $field || preg_match('/"' . preg_quote($alias) . '"$/', $field)) $clauses[] = $field;
|
||||
else $clauses[] = "$field AS \"$alias\"";
|
||||
}
|
||||
return $output;
|
||||
return implode(", ", $clauses);
|
||||
}
|
||||
|
||||
/**
|
||||
@ -757,7 +758,7 @@ class SQLQuery {
|
||||
if($column == null) {
|
||||
if($this->groupby) {
|
||||
$countQuery = new SQLQuery();
|
||||
$countQuery->select = array("count(*)");
|
||||
$countQuery->select("count(*)");
|
||||
$countQuery->from = array('(' . $clone->sql() . ') all_distinct');
|
||||
|
||||
return $countQuery->execute()->value();
|
||||
@ -775,21 +776,11 @@ class SQLQuery {
|
||||
|
||||
/**
|
||||
* Returns true if this query can be sorted by the given field.
|
||||
* Note that the implementation of this method is a little crude at the moment, it wil return
|
||||
* "false" more often that is strictly necessary.
|
||||
*/
|
||||
function canSortBy($fieldName) {
|
||||
$fieldName = preg_replace('/(\s+?)(A|DE)SC$/', '', $fieldName);
|
||||
|
||||
$sql = $this->sql();
|
||||
|
||||
$selects = $this->select;
|
||||
foreach($selects as $i => $sel) {
|
||||
if (preg_match('/"(.*)"\."(.*)"/', $sel, $matches)) $selects[$i] = $matches[2];
|
||||
}
|
||||
|
||||
$SQL_fieldName = Convert::raw2sql($fieldName);
|
||||
return (in_array($SQL_fieldName,$selects) || stripos($sql,"AS {$SQL_fieldName}"));
|
||||
return isset($this->select[$fieldName]);
|
||||
}
|
||||
|
||||
|
||||
@ -831,11 +822,9 @@ class SQLQuery {
|
||||
|
||||
/**
|
||||
* Return a new SQLQuery that calls the given aggregate functions on this data.
|
||||
* @param $columns An aggregate expression, such as 'MAX("Balance")', or an array of them.
|
||||
* @param $column An aggregate expression, such as 'MAX("Balance")', or a set of them.
|
||||
*/
|
||||
function aggregate($columns) {
|
||||
if(!is_array($columns)) $columns = array($columns);
|
||||
|
||||
function aggregate($column) {
|
||||
if($this->groupby || $this->limit) {
|
||||
throw new Exception("SQLQuery::aggregate() doesn't work with groupby or limit, yet");
|
||||
}
|
||||
@ -844,7 +833,7 @@ class SQLQuery {
|
||||
$clone->limit = null;
|
||||
$clone->orderby = null;
|
||||
$clone->groupby = null;
|
||||
$clone->select = $columns;
|
||||
$clone->select($column);
|
||||
|
||||
return $clone;
|
||||
}
|
||||
|
@ -91,7 +91,7 @@ class DbDatetimeTest extends FunctionalTest {
|
||||
$this->matchesRoughly($result, date('Y-m-d H:i:s', strtotime('+1 Day', $this->getDbNow())), 'tomorrow');
|
||||
|
||||
$query = new SQLQuery();
|
||||
$query->select($this->adapter->datetimeIntervalClause('"Created"', '-15 Minutes') . ' AS "test"')
|
||||
$query->select(array("test" => $this->adapter->datetimeIntervalClause('"Created"', '-15 Minutes')))
|
||||
->from('"DbDateTimeTest_Team"')
|
||||
->limit(1);
|
||||
$result = $query->execute()->value();
|
||||
@ -116,7 +116,7 @@ class DbDatetimeTest extends FunctionalTest {
|
||||
$this->matchesRoughly($result, -45 * 60, 'now - 45 minutes ahead');
|
||||
|
||||
$query = new SQLQuery();
|
||||
$query->select($this->adapter->datetimeDifferenceClause('"LastEdited"', '"Created"') . ' AS "test"')
|
||||
$query->select(array("test" => $this->adapter->datetimeDifferenceClause('"LastEdited"', '"Created"')))
|
||||
->from('"DbDateTimeTest_Team"')
|
||||
->limit(1);
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user