silverstripe-framework/docs/en/reference/sqlquery.md
2012-06-28 14:49:55 +02:00

6.6 KiB

SQL Query

Introduction

An object representing a SQL query. It is easier to deal with object-wrappers than string-parsing a raw SQL-query. This object is used by the SilverStripe ORM internally. Dealing with low-level SQL is not encouraged, since the ORM provides powerful abstraction APIs (see datamodel.

You'll run the risk of breaking various assumptions the ORM and code based on it have:

  • Custom getters/setters (object property can differ from database column)
  • DataObject hooks like onBeforeWrite() and onBeforeDelete()
  • Automatic casting
  • Default values set through objects
  • Database abstraction

We'll explain some ways to use SELECT with the full power of SQL, but still maintain a connection to the ORM where possible.

Usage

SELECT

:::php
$sqlQuery = new SQLQuery();
$sqlQuery->setFrom('Player');
$sqlQuery->selectField('FieldName', 'Name');
$sqlQuery->selectField('YEAR("Birthday")', 'BirthYear');
$sqlQuery->addLeftJoin(
	'Team',
  '"Player"."TeamID" = "Team"."ID"'
);
$sqlQuery->addWhere('YEAR("Birthday") = 1982');
// $sqlQuery->setOrderBy(...);
// $sqlQuery->setGroupBy(...);
// $sqlQuery->setHaving(...);
// $sqlQuery->setLimit(...);
// $sqlQuery->setDistinct(true);

// get the raw SQL
$rawSQL = $sqlQuery->sql();

// execute and return a Query-object
$result = $sqlQuery->execute();

DELETE

:::php
// ...
$sqlQuery->setDelete(true);

INSERT/UPDATE

(currently not supported -see below for alternative solutions)

Working with results

The result is an array lightly wrapped in a database-specific subclass of [api:Query]. This class implements the Iterator-interface defined in PHP5, and provides convenience-methods for accessing the data.

Iterating

:::php
foreach($result as $row) {
  echo $row['BirthYear'];
}

Quick value checking

Raw SQL is handy for performance-optimized calls.

:::php
class Team extends DataObject {
  public function getPlayerCount() {
    $sqlQuery = new SQLQuery();
    $sqlQuery->setFrom('Player');
    $sqlQuery->addSelect('COUNT("Player"."ID")');
    $sqlQuery->addLeftJoin('Team', '"Team"."ID" = "Player"."TeamID"');
    return $sqlQuery->execute()->value();
}

Way faster than dealing with [api:DataObject]s, but watch out for premature optimisation:

:::php
$players = $myTeam->Players();
echo $players->Count();

Mapping

Useful for creating dropdowns.

:::php
$sqlQuery = new SQLQuery();
$sqlQuery->setFrom('Player');
$sqlQuery->selectField('YEAR("Birthdate")', 'Birthdate');
$map = $sqlQuery->execute()->map();
$field = new DropdownField('Birthdates', 'Birthdates', $map);

"Raw" SQL with DB::query()

This is not recommended for most cases, but you can also use the SilverStripe database-layer to fire off a raw query:

:::php
DB::query('UPDATE "Player" SET "Status"=\'Active\'');

<<<<<<< Updated upstream

Transforming a result to [api:ArrayList]

======= One example for using a raw DB::query is when you are wanting to order twice in the database:

:::php
$records = DB::query('SELECT *, CASE WHEN "ThumbnailID" = 0 THEN 2 ELSE 1 END AS "HasThumbnail" FROM "TempDoc" ORDER BY "HasThumbnail", "Name" ASC');
$items = singleton('TempDoc')->buildDataObjectSet($records);

This CASE SQL creates a second field "HasThumbnail" depending if "ThumbnailID" exists in the database which you can then order by "HasThumbnail" to make sure the thumbnails are at the top of the list and then order by another field "Name" separately for both the items that have a thumbnail and then for those that don't have thumbnails.

"Semi-raw" SQL with buildSQL()

You can gain some ground on the datamodel-side when involving the selected class for querying. You don't necessarily need to call buildSQL from a specific object-instance, a singleton will do just fine.

:::php
$sqlQuery = singleton('Player')->buildSQL('YEAR("Birthdate") = 1982');

This form of building a query has the following advantages:

  • Respects DataObject::$default_sort
  • Automatically LEFT JOIN on all base-tables (see database-structure)
  • Selection of ID, ClassName, RecordClassName, which are necessary to use buildDataObjectSet later on
  • Filtering records for correct ClassName

Transforming a result to [api:DataObjectSet]

Stashed changes

This is a commonly used technique inside SilverStripe: Use raw SQL, but transfer the resulting rows back into [api:DataObject]s.

:::php
$sqlQuery = new SQLQuery();
$sqlQuery->setSelect(array(
  '"Firstname" AS "Name"',
  'YEAR("Birthday") AS "BirthYear"',
  // IMPORTANT: Needs to be set after other selects to avoid overlays
  '"Player"."ClassName" AS "ClassName"',
  '"Player"."ClassName" AS "RecordClassName"',
  '"Player"."ID" AS "ID"'
));
$sqlQuery->setFrom('Player');
$sqlQuery->addLeftJoin('Team', '"Player"."TeamID" = "Team"."ID"');
$sqlQuery->addWhere("YEAR("Player"."Birthday") = 1982");

$result = $sqlQuery->execute();
var_dump($result->first()); // array

// let Silverstripe work the magic
$myList = singleton('Player')->buildDataObjectSet($result);
var_dump($myDataObjectSet->First()); // DataObject

// this is where it gets tricky
$myFirstPlayer = $myDataObjectSet->First();
var_dump($myFirstPlayer->Name); // 'John'
var_dump($myFirstPlayer->Firstname); // undefined, as it was not part of the SELECT-clause;
var_dump($myFirstPlayer->Surname); // undefined, as it was not part of the SELECT-clause

// lets assume that class Player extends BasePlayer,
// and BasePlayer has a database-column "Status"
var_dump($myFirstPlayer->Status); // undefined, as we didn't LEFT JOIN the BasePlayer-table

CAUTION: Depending on the selected columns in your query, you might get into one of the following scenarios:

  • Not all object-properties accessible: You need to take care of selecting the right stuff yourself
  • Overlayed object-properties: If you LEFT JOIN a table which also has a column 'Birthdate' and do a global select on this table, you might not be able to access original object-properties.
  • You can't create [api:DataObject]s where no scalar record-data is available, e.g. when using GROUP BY
  • Naming conflicts with custom getters: A getter like Player->getName() will overlay the column-data selected in the above example

Be careful when saving back [api:DataObject]s created through buildDataObjectSet, you might get strange side-effects due to the issues noted above.

Using FormFields with custom SQL

Some subclasses of [api:FormField] for ways to create sophisticated report-tables based on SQL.

API Documentation

[api:SQLQuery]