QueryBuilder.php 30.3 KB
Newer Older
w  
Qiang Xue committed
1 2 3 4 5 6 7 8 9 10 11 12
<?php
/**
 * This file contains the Command class.
 *
 * @author Qiang Xue <qiang.xue@gmail.com>
 * @link http://www.yiiframework.com/
 * @copyright Copyright &copy; 2008-2012 Yii Software LLC
 * @license http://www.yiiframework.com/license/
 */

namespace yii\db\dao;

w  
Qiang Xue committed
13 14
use yii\db\Exception;

w  
Qiang Xue committed
15 16 17
/**
 * QueryBuilder builds a SQL statement based on the specification given as a [[Query]] object.
 *
Qiang Xue committed
18 19 20
 * QueryBuilder is often used behind the scenes by [[Query]] to build a DBMS-dependent SQL statement
 * from a [[Query]] object.
 *
w  
Qiang Xue committed
21 22 23
 * @author Qiang Xue <qiang.xue@gmail.com>
 * @since 2.0
 */
Qiang Xue committed
24
class QueryBuilder extends \yii\base\Object
w  
Qiang Xue committed
25 26
{
	/**
w  
Qiang Xue committed
27
	 * @var array the abstract column types mapped to physical column types.
Qiang Xue committed
28 29
	 * This is mainly used to support creating/modifying tables using DB-independent data type specifications.
	 * Child classes should override this property to declare supported type mappings.
w  
Qiang Xue committed
30
	 */
Qiang Xue committed
31
	public $typeMap = array();
Qiang Xue committed
32 33 34
	/**
	 * @var Connection the database connection.
	 */
w  
Qiang Xue committed
35
	public $connection;
Qiang Xue committed
36
	/**
Qiang Xue committed
37
	 * @var Driver the database driver used for this query builder.
Qiang Xue committed
38
	 */
Qiang Xue committed
39
	public $driver;
Qiang Xue committed
40 41 42 43 44
	/**
	 * @var string the separator between different fragments of a SQL statement.
	 * Defaults to an empty space. This is mainly used by [[build()]] when generating a SQL statement.
	 */
	public $separator = " ";
Qiang Xue committed
45
	/**
Qiang Xue committed
46
	 * @var Query the Query object that is currently processed by the query builder to generate a SQL statement.
Qiang Xue committed
47
	 * This property will be set null upon completion of [[build()]].
Qiang Xue committed
48
	 */
Qiang Xue committed
49
	public $query;
Qiang Xue committed
50 51 52 53
	/**
	 * @var boolean whether to automatically quote table and column names when generating SQL statements.
	 */
	public $autoQuote = true;
w  
Qiang Xue committed
54

Qiang Xue committed
55 56
	/**
	 * Constructor.
Qiang Xue committed
57
	 * @param Connection $connection the database connection.
Qiang Xue committed
58
	 */
Qiang Xue committed
59
	public function __construct($connection)
w  
Qiang Xue committed
60
	{
Qiang Xue committed
61
		$this->connection = $connection;
Qiang Xue committed
62
		$this->driver = $connection->getDriver();
w  
Qiang Xue committed
63 64
	}

Qiang Xue committed
65
	/**
Qiang Xue committed
66 67 68 69 70
	 * Generates a SQL statement from a [[Query]] object.
	 * Note that when generating SQL statements for INSERT and UPDATE queries,
	 * the query object's [[Query::params]] property may be appended with new parameters.
	 * @param Query $query the [[Query]] object from which the SQL statement will be generated
	 * @return string the generated SQL statement
Qiang Xue committed
71
	 */
w  
Qiang Xue committed
72 73
	public function build($query)
	{
Qiang Xue committed
74
		$this->query = $query;
Qiang Xue committed
75
		if ($query->operation !== null) {
Qiang Xue committed
76
			// non-SELECT query
Qiang Xue committed
77 78
			$method = array_shift($query->operation);
			$sql = call_user_func_array(array($this, $method), $query->operation);
Qiang Xue committed
79 80 81
		} else {
			// SELECT query
			$clauses = array(
Qiang Xue committed
82 83 84 85 86 87 88 89 90
				$this->buildSelect(),
				$this->buildFrom(),
				$this->buildJoin(),
				$this->buildWhere(),
				$this->buildGroupBy(),
				$this->buildHaving(),
				$this->buildUnion(),
				$this->buildOrderBy(),
				$this->buildLimit(),
Qiang Xue committed
91
			);
Qiang Xue committed
92
			$sql = implode($this->separator, array_filter($clauses));
Qiang Xue committed
93
		}
Qiang Xue committed
94 95
		$this->query = null;
		return $sql;
w  
Qiang Xue committed
96 97 98 99 100
	}

	/**
	 * Creates and executes an INSERT SQL statement.
	 * The method will properly escape the column names, and bind the values to be inserted.
Qiang Xue committed
101 102 103 104 105
	 * For example,
	 *
	 * ~~~
	 * $params = array();
	 * $sql = $queryBuilder->insert('tbl_user', array(
Qiang Xue committed
106 107
	 *	 'name' => 'Sam',
	 *	 'age' => 30,
Qiang Xue committed
108 109 110
	 * ), $params);
	 * ~~~
	 *
w  
Qiang Xue committed
111 112
	 * @param string $table the table that new rows will be inserted into.
	 * @param array $columns the column data (name=>value) to be inserted into the table.
Qiang Xue committed
113 114
	 * @param array $params the parameters to be bound to the query. This method will modify
	 * this parameter by appending new parameters to be bound to the query.
w  
Qiang Xue committed
115 116 117 118 119 120 121 122
	 * @return integer number of rows affected by the execution.
	 */
	public function insert($table, $columns, &$params = array())
	{
		$names = array();
		$placeholders = array();
		$count = 0;
		foreach ($columns as $name => $value) {
Qiang Xue committed
123
			$names[] = $this->quoteColumnName($name);
w  
Qiang Xue committed
124 125 126 127 128
			if ($value instanceof Expression) {
				$placeholders[] = $value->expression;
				foreach ($value->params as $n => $v) {
					$params[$n] = $v;
				}
Qiang Xue committed
129
			} else {
w  
Qiang Xue committed
130 131 132 133 134
				$placeholders[] = ':p' . $count;
				$params[':p' . $count] = $value;
				$count++;
			}
		}
Qiang Xue committed
135 136
		if ($this->query instanceof Query) {
			$this->query->addParams($params);
Qiang Xue committed
137
		}
w  
Qiang Xue committed
138

Qiang Xue committed
139
		return 'INSERT INTO ' . $this->quoteTableName($table)
w  
Qiang Xue committed
140 141 142 143 144 145 146
			. ' (' . implode(', ', $names) . ') VALUES ('
			. implode(', ', $placeholders) . ')';
	}

	/**
	 * Creates and executes an UPDATE SQL statement.
	 * The method will properly escape the column names and bind the values to be updated.
Qiang Xue committed
147 148 149 150 151
	 * For example,
	 *
	 * ~~~
	 * $params = array();
	 * $sql = $queryBuilder->update('tbl_user', array(
Qiang Xue committed
152
	 *	 'status' => 1,
Qiang Xue committed
153 154 155
	 * ), 'age > 30', $params);
	 * ~~~
	 *
w  
Qiang Xue committed
156 157
	 * @param string $table the table to be updated.
	 * @param array $columns the column data (name=>value) to be updated.
Qiang Xue committed
158 159
	 * @param mixed $condition the condition that will be put in the WHERE part. Please
	 * refer to [[Query::where()]] on how to specify condition.
Qiang Xue committed
160 161
	 * @param array $params the parameters to be bound to the query. This method will modify
	 * this parameter by appending new parameters to be bound to the query.
w  
Qiang Xue committed
162 163
	 * @return integer number of rows affected by the execution.
	 */
Qiang Xue committed
164
	public function update($table, $columns, $condition = '', &$params = array())
w  
Qiang Xue committed
165 166 167 168 169
	{
		$lines = array();
		$count = 0;
		foreach ($columns as $name => $value) {
			if ($value instanceof Expression) {
Qiang Xue committed
170
				$lines[] = $this->quoteColumnName($name, true) . '=' . $value->expression;
w  
Qiang Xue committed
171 172 173
				foreach ($value->params as $n => $v) {
					$params[$n] = $v;
				}
Qiang Xue committed
174
			} else {
Qiang Xue committed
175
				$lines[] = $this->quoteColumnName($name, true) . '=:p' . $count;
w  
Qiang Xue committed
176 177 178 179
				$params[':p' . $count] = $value;
				$count++;
			}
		}
Qiang Xue committed
180 181
		if ($this->query instanceof Query) {
			$this->query->addParams($params);
Qiang Xue committed
182
		}
Qiang Xue committed
183
		$sql = 'UPDATE ' . $this->quoteTableName($table) . ' SET ' . implode(', ', $lines);
Qiang Xue committed
184
		if (($where = $this->buildCondition($condition)) != '') {
w  
Qiang Xue committed
185 186
			$sql .= ' WHERE ' . $where;
		}
w  
Qiang Xue committed
187

w  
Qiang Xue committed
188 189 190 191 192
		return $sql;
	}

	/**
	 * Creates and executes a DELETE SQL statement.
Qiang Xue committed
193 194 195 196 197 198
	 * For example,
	 *
	 * ~~~
	 * $sql = $queryBuilder->delete('tbl_user', 'status = 0');
	 * ~~~
	 *
w  
Qiang Xue committed
199
	 * @param string $table the table where the data will be deleted from.
Qiang Xue committed
200 201
	 * @param mixed $condition the condition that will be put in the WHERE part. Please
	 * refer to [[Query::where()]] on how to specify condition.
w  
Qiang Xue committed
202 203
	 * @return integer number of rows affected by the execution.
	 */
Qiang Xue committed
204
	public function delete($table, $condition = '')
w  
Qiang Xue committed
205
	{
Qiang Xue committed
206
		$sql = 'DELETE FROM ' . $this->quoteTableName($table);
Qiang Xue committed
207
		if (($where = $this->buildCondition($condition)) != '') {
w  
Qiang Xue committed
208 209 210
			$sql .= ' WHERE ' . $where;
		}
		return $sql;
w  
Qiang Xue committed
211 212
	}

w  
Qiang Xue committed
213 214 215 216 217 218
	/**
	 * Builds a SQL statement for creating a new DB table.
	 *
	 * The columns in the new  table should be specified as name-definition pairs (e.g. 'name'=>'string'),
	 * where name stands for a column name which will be properly quoted by the method, and definition
	 * stands for the column type which can contain an abstract DB type.
Qiang Xue committed
219
	 * The [[getColumnType()]] method will be invoked to convert any abstract type into a physical one.
w  
Qiang Xue committed
220 221 222 223
	 *
	 * If a column is specified with definition only (e.g. 'PRIMARY KEY (name, type)'), it will be directly
	 * inserted into the generated SQL.
	 *
Qiang Xue committed
224 225 226 227
	 * For example,
	 *
	 * ~~~
	 * $sql = $queryBuilder->createTable('tbl_user', array(
Qiang Xue committed
228 229 230
	 *	 'id' => 'pk',
	 *	 'name' => 'string',
	 *	 'age' => 'integer',
Qiang Xue committed
231 232 233
	 * ));
	 * ~~~
	 *
w  
Qiang Xue committed
234 235 236 237 238 239 240 241
	 * @param string $table the name of the table to be created. The name will be properly quoted by the method.
	 * @param array $columns the columns (name=>definition) in the new table.
	 * @param string $options additional SQL fragment that will be appended to the generated SQL.
	 * @return string the SQL statement for creating a new DB table.
	 */
	public function createTable($table, $columns, $options = null)
	{
		$cols = array();
w  
Qiang Xue committed
242 243
		foreach ($columns as $name => $type) {
			if (is_string($name)) {
Qiang Xue committed
244 245
				$cols[] = "\t" . $this->quoteColumnName($name) . ' ' . $this->getColumnType($type);
			} else {
w  
Qiang Xue committed
246
				$cols[] = "\t" . $type;
Qiang Xue committed
247
			}
w  
Qiang Xue committed
248
		}
Qiang Xue committed
249
		$sql = "CREATE TABLE " . $this->quoteTableName($table) . " (\n" . implode(",\n", $cols) . "\n)";
w  
Qiang Xue committed
250 251 252 253 254
		return $options === null ? $sql : $sql . ' ' . $options;
	}

	/**
	 * Builds a SQL statement for renaming a DB table.
Qiang Xue committed
255
	 * @param string $oldName the table to be renamed. The name will be properly quoted by the method.
w  
Qiang Xue committed
256 257 258
	 * @param string $newName the new table name. The name will be properly quoted by the method.
	 * @return string the SQL statement for renaming a DB table.
	 */
Qiang Xue committed
259
	public function renameTable($oldName, $newName)
w  
Qiang Xue committed
260
	{
Qiang Xue committed
261
		return 'RENAME TABLE ' . $this->quoteTableName($oldName) . ' TO ' . $this->quoteTableName($newName);
w  
Qiang Xue committed
262 263 264 265 266 267 268 269 270
	}

	/**
	 * Builds a SQL statement for dropping a DB table.
	 * @param string $table the table to be dropped. The name will be properly quoted by the method.
	 * @return string the SQL statement for dropping a DB table.
	 */
	public function dropTable($table)
	{
Qiang Xue committed
271
		return "DROP TABLE " . $this->quoteTableName($table);
w  
Qiang Xue committed
272 273 274 275 276 277 278 279
	}

	/**
	 * Builds a SQL statement for truncating a DB table.
	 * @param string $table the table to be truncated. The name will be properly quoted by the method.
	 * @return string the SQL statement for truncating a DB table.
	 */
	public function truncateTable($table)
w  
Qiang Xue committed
280
	{
Qiang Xue committed
281
		return "TRUNCATE TABLE " . $this->quoteTableName($table);
w  
Qiang Xue committed
282 283 284 285 286 287
	}

	/**
	 * Builds a SQL statement for adding a new DB column.
	 * @param string $table the table that the new column will be added to. The table name will be properly quoted by the method.
	 * @param string $column the name of the new column. The name will be properly quoted by the method.
Qiang Xue committed
288
	 * @param string $type the column type. The [[getColumnType()]] method will be invoked to convert abstract column type (if any)
w  
Qiang Xue committed
289 290 291 292 293 294
	 * into the physical one. Anything that is not recognized as abstract type will be kept in the generated SQL.
	 * For example, 'string' will be turned into 'varchar(255)', while 'string not null' will become 'varchar(255) not null'.
	 * @return string the SQL statement for adding a new column.
	 */
	public function addColumn($table, $column, $type)
	{
Qiang Xue committed
295 296
		return 'ALTER TABLE ' . $this->quoteTableName($table)
			. ' ADD ' . $this->quoteColumnName($column) . ' '
w  
Qiang Xue committed
297 298
			. $this->getColumnType($type);
	}
w  
Qiang Xue committed
299

w  
Qiang Xue committed
300 301 302 303 304 305 306 307
	/**
	 * Builds a SQL statement for dropping a DB column.
	 * @param string $table the table whose column is to be dropped. The name will be properly quoted by the method.
	 * @param string $column the name of the column to be dropped. The name will be properly quoted by the method.
	 * @return string the SQL statement for dropping a DB column.
	 */
	public function dropColumn($table, $column)
	{
Qiang Xue committed
308 309
		return "ALTER TABLE " . $this->quoteTableName($table)
			. " DROP COLUMN " . $this->quoteColumnName($column, true);
w  
Qiang Xue committed
310 311 312 313 314
	}

	/**
	 * Builds a SQL statement for renaming a column.
	 * @param string $table the table whose column is to be renamed. The name will be properly quoted by the method.
Qiang Xue committed
315
	 * @param string $oldName the old name of the column. The name will be properly quoted by the method.
w  
Qiang Xue committed
316 317 318
	 * @param string $newName the new name of the column. The name will be properly quoted by the method.
	 * @return string the SQL statement for renaming a DB column.
	 */
Qiang Xue committed
319
	public function renameColumn($table, $oldName, $newName)
w  
Qiang Xue committed
320
	{
Qiang Xue committed
321 322 323
		return "ALTER TABLE " . $this->quoteTableName($table)
			. " RENAME COLUMN " . $this->quoteColumnName($oldName, true)
			. " TO " . $this->quoteColumnName($newName, true);
w  
Qiang Xue committed
324 325 326 327 328 329
	}

	/**
	 * Builds a SQL statement for changing the definition of a column.
	 * @param string $table the table whose column is to be changed. The table name will be properly quoted by the method.
	 * @param string $column the name of the column to be changed. The name will be properly quoted by the method.
Qiang Xue committed
330 331 332 333
	 * @param string $type the new column type. The [[getColumnType()]] method will be invoked to convert abstract
	 * column type (if any) into the physical one. Anything that is not recognized as abstract type will be kept
	 * in the generated SQL. For example, 'string' will be turned into 'varchar(255)', while 'string not null'
	 * will become 'varchar(255) not null'.
w  
Qiang Xue committed
334 335 336 337
	 * @return string the SQL statement for changing the definition of a column.
	 */
	public function alterColumn($table, $column, $type)
	{
Qiang Xue committed
338 339 340
		return 'ALTER TABLE ' . $this->quoteTableName($table) . ' CHANGE '
			. $this->quoteColumnName($column, true) . ' '
			. $this->quoteColumnName($column, true) . ' '
w  
Qiang Xue committed
341 342 343 344 345 346 347 348
			. $this->getColumnType($type);
	}

	/**
	 * Builds a SQL statement for adding a foreign key constraint to an existing table.
	 * The method will properly quote the table and column names.
	 * @param string $name the name of the foreign key constraint.
	 * @param string $table the table that the foreign key constraint will be added to.
Qiang Xue committed
349 350
	 * @param string|array $columns the name of the column to that the constraint will be added on.
	 * If there are multiple columns, separate them with commas or use an array to represent them.
w  
Qiang Xue committed
351
	 * @param string $refTable the table that the foreign key references to.
Qiang Xue committed
352 353
	 * @param string|array $refColumns the name of the column that the foreign key references to.
	 * If there are multiple columns, separate them with commas or use an array to represent them.
w  
Qiang Xue committed
354 355 356 357 358 359
	 * @param string $delete the ON DELETE option. Most DBMS support these options: RESTRICT, CASCADE, NO ACTION, SET DEFAULT, SET NULL
	 * @param string $update the ON UPDATE option. Most DBMS support these options: RESTRICT, CASCADE, NO ACTION, SET DEFAULT, SET NULL
	 * @return string the SQL statement for adding a foreign key constraint to an existing table.
	 */
	public function addForeignKey($name, $table, $columns, $refTable, $refColumns, $delete = null, $update = null)
	{
Qiang Xue committed
360 361 362 363 364
		$sql = 'ALTER TABLE ' . $this->quoteTableName($table)
			. ' ADD CONSTRAINT ' . $this->quoteColumnName($name)
			. ' FOREIGN KEY (' . $this->buildColumns($columns) . ')'
			. ' REFERENCES ' . $this->quoteTableName($refTable)
			. ' (' . $this->buildColumns($refColumns) . ')';
Qiang Xue committed
365
		if ($delete !== null) {
w  
Qiang Xue committed
366
			$sql .= ' ON DELETE ' . $delete;
Qiang Xue committed
367 368
		}
		if ($update !== null) {
w  
Qiang Xue committed
369
			$sql .= ' ON UPDATE ' . $update;
Qiang Xue committed
370
		}
w  
Qiang Xue committed
371 372 373 374 375 376 377 378 379 380 381
		return $sql;
	}

	/**
	 * Builds a SQL statement for dropping a foreign key constraint.
	 * @param string $name the name of the foreign key constraint to be dropped. The name will be properly quoted by the method.
	 * @param string $table the table whose foreign is to be dropped. The name will be properly quoted by the method.
	 * @return string the SQL statement for dropping a foreign key constraint.
	 */
	public function dropForeignKey($name, $table)
	{
Qiang Xue committed
382 383
		return 'ALTER TABLE ' . $this->quoteTableName($table)
			. ' DROP CONSTRAINT ' . $this->quoteColumnName($name);
w  
Qiang Xue committed
384 385 386 387 388 389
	}

	/**
	 * Builds a SQL statement for creating a new index.
	 * @param string $name the name of the index. The name will be properly quoted by the method.
	 * @param string $table the table that the new index will be created for. The table name will be properly quoted by the method.
Qiang Xue committed
390 391 392
	 * @param string|array $columns the column(s) that should be included in the index. If there are multiple columns,
	 * separate them with commas or use an array to represent them. Each column name will be properly quoted
	 * by the method, unless a parenthesis is found in the name.
w  
Qiang Xue committed
393 394 395
	 * @param boolean $unique whether to add UNIQUE constraint on the created index.
	 * @return string the SQL statement for creating a new index.
	 */
Qiang Xue committed
396
	public function createIndex($name, $table, $columns, $unique = false)
w  
Qiang Xue committed
397 398
	{
		return ($unique ? 'CREATE UNIQUE INDEX ' : 'CREATE INDEX ')
Qiang Xue committed
399 400 401
			. $this->quoteTableName($name) . ' ON '
			. $this->quoteTableName($table)
			. ' (' . $this->buildColumns($columns) . ')';
w  
Qiang Xue committed
402 403 404 405 406 407 408 409 410 411
	}

	/**
	 * Builds a SQL statement for dropping an index.
	 * @param string $name the name of the index to be dropped. The name will be properly quoted by the method.
	 * @param string $table the table whose index is to be dropped. The name will be properly quoted by the method.
	 * @return string the SQL statement for dropping an index.
	 */
	public function dropIndex($name, $table)
	{
Qiang Xue committed
412
		return 'DROP INDEX ' . $this->quoteTableName($name) . ' ON ' . $this->quoteTableName($table);
w  
Qiang Xue committed
413 414
	}

w  
Qiang Xue committed
415 416 417 418
	/**
	 * Resets the sequence value of a table's primary key.
	 * The sequence will be reset such that the primary key of the next new row inserted
	 * will have the specified value or 1.
Qiang Xue committed
419
	 * @param string $table the table schema whose primary key sequence will be reset
w  
Qiang Xue committed
420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437
	 * @param mixed $value the value for the primary key of the next new row inserted. If this is not set,
	 * the next new row's primary key will have a value 1.
	 */
	public function resetSequence($table, $value = null)
	{
	}

	/**
	 * Enables or disables integrity check.
	 * @param boolean $check whether to turn on or off the integrity check.
	 * @param string $schema the schema of the tables. Defaults to empty string, meaning the current or default schema.
	 */
	public function checkIntegrity($check = true, $schema = '')
	{
	}

	/**
	 * Converts an abstract column type into a physical column type.
Qiang Xue committed
438
	 * The conversion is done using the type map specified in [[typeMap]].
Qiang Xue committed
439
	 * The following abstract column types are supported (using MySQL as an example to explain the corresponding
w  
Qiang Xue committed
440
	 * physical types):
Qiang Xue committed
441
	 *
Qiang Xue committed
442 443 444 445 446 447 448 449 450 451 452 453 454 455 456
	 * - `pk`: an auto-incremental primary key type, will be converted into "int(11) NOT NULL AUTO_INCREMENT PRIMARY KEY"
	 * - `string`: string type, will be converted into "varchar(255)"
	 * - `text`: a long string type, will be converted into "text"
	 * - `smallint`: a small integer type, will be converted into "smallint(6)"
	 * - `integer`: integer type, will be converted into "int(11)"
	 * - `bigint`: a big integer type, will be converted into "bigint(20)"
	 * - `boolean`: boolean type, will be converted into "tinyint(1)"
	 * - `float``: float number type, will be converted into "float"
	 * - `decimal`: decimal number type, will be converted into "decimal"
	 * - `datetime`: datetime type, will be converted into "datetime"
	 * - `timestamp`: timestamp type, will be converted into "timestamp"
	 * - `time`: time type, will be converted into "time"
	 * - `date`: date type, will be converted into "date"
	 * - `money`: money type, will be converted into "decimal(19,4)"
	 * - `binary`: binary data type, will be converted into "blob"
w  
Qiang Xue committed
457 458
	 *
	 * If the abstract type contains two or more parts separated by spaces (e.g. "string NOT NULL"), then only
Qiang Xue committed
459
	 * the first part will be converted, and the rest of the parts will be appended to the converted result.
w  
Qiang Xue committed
460
	 * For example, 'string NOT NULL' is converted to 'varchar(255) NOT NULL'.
Qiang Xue committed
461 462
	 *
	 * If a type cannot be found in [[typeMap]], it will be returned without any change.
w  
Qiang Xue committed
463 464 465
	 * @param string $type abstract column type
	 * @return string physical column type.
	 */
Qiang Xue committed
466 467
	public function getColumnType($type)
	{
w  
Qiang Xue committed
468 469
		if (isset($this->typeMap[$type])) {
			return $this->typeMap[$type];
Qiang Xue committed
470 471 472 473 474 475 476 477 478
		} elseif (preg_match('/^(\w+)\s+/', $type, $matches)) {
			if (isset($this->typeMap[$matches[0]])) {
				return preg_replace('/^\w+/', $this->typeMap[$matches[0]], $type);
			}
		}
		return $type;
	}

	/**
Qiang Xue committed
479 480 481 482 483
	 * Parses the condition specification and generates the corresponding SQL expression.
	 * @param string|array $condition the condition specification. Please refer to [[Query::where()]]
	 * on how to specify a condition.
	 * @return string the generated SQL expression
	 * @throws \yii\db\Exception if the condition is in bad format
Qiang Xue committed
484 485 486
	 */
	public function buildCondition($condition)
	{
Qiang Xue committed
487 488 489 490 491 492 493 494 495 496 497 498 499
		static $builders = array(
			'and' => 'buildAndCondition',
			'or' => 'buildAndCondition',
			'between' => 'buildBetweenCondition',
			'not between' => 'buildBetweenCondition',
			'in' => 'buildInCondition',
			'not in' => 'buildInCondition',
			'like' => 'buildLikeCondition',
			'not like' => 'buildLikeCondition',
			'or like' => 'buildLikeCondition',
			'or not like' => 'buildLikeCondition',
		);

Qiang Xue committed
500 501 502 503 504
		if (!is_array($condition)) {
			return $condition;
		} elseif ($condition === array()) {
			return '';
		}
Qiang Xue committed
505 506 507 508 509 510 511 512 513 514 515 516 517
		if (isset($condition[0])) { // operator format: operator, operand 1, operand 2, ...
			$operator = $condition[0];
			if (isset($builders[$operator])) {
				$method = $builders[$operator];
				array_shift($condition);
				return $this->$method($operator, $condition);
			} else {
				throw new Exception('Found unknown operator in query: ' . $operator);
			}
		} else { // hash format: 'column1'=>'value1', 'column2'=>'value2', ...
			return $this->buildHashCondition($condition);
		}
	}
Qiang Xue committed
518

Qiang Xue committed
519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534
	private function buildHashCondition($condition)
	{
		$parts = array();
		foreach ($condition as $column => $value) {
			if (is_array($value)) { // IN condition
				$parts[] = $this->buildInCondition('in', array($column, $value));
			} else {
				if (strpos($column, '(') === false) {
					$column = $this->quoteColumnName($column);
				}
				if ($value === null) {
					$parts[] = "$column IS NULL";
				} elseif (is_string($value)) {
					$parts[] = "$column=" . $this->connection->quoteValue($value);
				} else {
					$parts[] = "$column=$value";
Qiang Xue committed
535 536 537
				}
			}
		}
Qiang Xue committed
538 539
		return '(' . implode(') AND (', $parts) . ')';
	}
Qiang Xue committed
540

Qiang Xue committed
541 542 543 544 545 546 547 548 549 550
	private function buildAndCondition($operator, $operands)
	{
		$parts = array();
		foreach ($operands as $operand) {
			if (is_array($operand)) {
				$operand = $this->buildCondition($operand);
			}
			if ($operand !== '') {
				$parts[] = $operand;
			}
Qiang Xue committed
551
		}
Qiang Xue committed
552 553 554 555 556 557 558 559 560 561 562 563 564 565
		if ($parts !== array()) {
			return '(' . implode(") $operator (", $parts) . ')';
		} else {
			return '';
		}
	}

	private function buildBetweenCondition($operator, $operands)
	{
		if (!isset($operands[0], $operands[1], $operands[2])) {
			throw new Exception("Operator '$operator' requires three operands.");
		}

		list($column, $value1, $value2) = $operands;
Qiang Xue committed
566 567

		if (strpos($column, '(') === false) {
Qiang Xue committed
568
			$column = $this->quoteColumnName($column);
Qiang Xue committed
569
		}
Qiang Xue committed
570 571
		$value1 = is_string($value1) ? $this->connection->quoteValue($value1) : (string)$value1;
		$value2 = is_string($value2) ? $this->connection->quoteValue($value2) : (string)$value2;
Qiang Xue committed
572

Qiang Xue committed
573 574 575 576 577 578 579
		return "$column $operator $value1 AND $value2";
	}

	private function buildInCondition($operator, $operands)
	{
		if (!isset($operands[0], $operands[1])) {
			throw new Exception("Operator '$operator' requires two operands.");
Qiang Xue committed
580 581
		}

Qiang Xue committed
582 583
		list($column, $values) = $operands;

Qiang Xue committed
584 585 586 587
		if (!is_array($values)) {
			$values = array($values);
		}

Qiang Xue committed
588 589
		if ($values === array()) {
			return $operator === 'in' ? '0=1' : '';
Qiang Xue committed
590 591
		}

Qiang Xue committed
592 593 594
		foreach ($values as $i => $value) {
			$values[$i] = is_string($value) ? $this->connection->quoteValue($value) : (string)$value;
		}
Qiang Xue committed
595

Qiang Xue committed
596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632
		if (strpos($column, '(') === false) {
			$column = $this->quoteColumnName($column);
		}

		return "$column $operator (" . implode(', ', $values) . ')';
	}

	private function buildLikeCondition($operator, $operands)
	{
		if (!isset($operands[0], $operands[1])) {
			throw new Exception("Operator '$operator' requires two operands.");
		}

		list($column, $values) = $operands;

		if (!is_array($values)) {
			$values = array($values);
		}

		if ($values === array()) {
			return $operator === 'like' || $operator === 'or like' ? '0=1' : '';
		}

		if ($operator === 'like' || $operator === 'not like') {
			$andor = ' and ';
		} else {
			$andor = ' or ';
			$operator = $operator === 'or like' ? 'like' : 'not like';
		}

		if (strpos($column, '(') === false) {
			$column = $this->quoteColumnName($column);
		}

		$parts = array();
		foreach ($values as $value) {
			$parts[] = "$column $operator " . $this->connection->quoteValue($value);
w  
Qiang Xue committed
633
		}
Qiang Xue committed
634

Qiang Xue committed
635
		return implode($andor, $parts);
w  
Qiang Xue committed
636 637
	}

Qiang Xue committed
638 639 640 641
	/**
	 * @return string the SELECT clause built from [[query]].
	 */
	protected function buildSelect()
w  
Qiang Xue committed
642
	{
Qiang Xue committed
643 644 645
		$select = $this->query->distinct ? 'SELECT DISTINCT' : 'SELECT';
		if ($this->query->selectOption != '') {
			$select .= ' ' . $this->query->selectOption;
w  
Qiang Xue committed
646
		}
w  
Qiang Xue committed
647

Qiang Xue committed
648
		$columns = $this->query->select;
w  
Qiang Xue committed
649 650 651 652
		if (empty($columns)) {
			return $select . ' *';
		}

Qiang Xue committed
653 654 655 656
		if ($this->autoQuote) {
			if (!is_array($columns)) {
				if (strpos($columns, '(') !== false) {
					return $select . ' ' . $columns;
Qiang Xue committed
657
				} else {
Qiang Xue committed
658 659 660 661 662 663 664 665 666 667 668 669
					$columns = preg_split('/\s*,\s*/', trim($columns), -1, PREG_SPLIT_NO_EMPTY);
				}
			}
			foreach ($columns as $i => $column) {
				if (is_object($column)) {
					$columns[$i] = (string)$column;
				} elseif (strpos($column, '(') === false) {
					if (preg_match('/^(.*?)(?i:\s+as\s+|\s+)([\w\-\.])$/', $column, $matches)) {
						$columns[$i] = $this->driver->quoteColumnName($matches[1]) . ' AS ' . $this->driver->quoteSimpleColumnName($matches[2]);
					} else {
						$columns[$i] = $this->driver->quoteColumnName($column);
					}
w  
Qiang Xue committed
670 671 672 673
				}
			}
		}

Qiang Xue committed
674 675 676 677 678
		if (is_array($columns)) {
			$columns = implode(', ', $columns);
		}

		return $select . ' ' . $columns;
w  
Qiang Xue committed
679 680
	}

Qiang Xue committed
681 682 683 684
	/**
	 * @return string the FROM clause built from [[query]].
	 */
	protected function buildFrom()
w  
Qiang Xue committed
685
	{
Qiang Xue committed
686
		if (empty($this->query->from)) {
Qiang Xue committed
687 688 689
			return '';
		}

Qiang Xue committed
690
		$tables = $this->query->from;
w  
Qiang Xue committed
691

Qiang Xue committed
692 693 694 695
		if ($this->autoQuote) {
			if (!is_array($tables)) {
				if (strpos($tables, '(') !== false) {
					return 'FROM ' . $tables;
Qiang Xue committed
696
				} else {
Qiang Xue committed
697
					$tables = preg_split('/\s*,\s*/', trim($tables), -1, PREG_SPLIT_NO_EMPTY);
w  
Qiang Xue committed
698 699
				}
			}
Qiang Xue committed
700 701 702 703 704 705 706 707 708 709 710 711 712
			foreach ($tables as $i => $table) {
				if (strpos($table, '(') === false) {
					if (preg_match('/^(.*?)(?i:\s+as\s+|\s+)(.*)$/i', $table, $matches)) { // with alias
						$tables[$i] = $this->driver->quoteTableName($matches[1]) . ' ' . $this->driver->quoteTableName($matches[2]);
					} else {
						$tables[$i] = $this->driver->quoteTableName($table);
					}
				}
			}
		}

		if (is_array($tables)) {
			$tables = implode(', ', $tables);
w  
Qiang Xue committed
713 714
		}

Qiang Xue committed
715
		return 'FROM ' . $tables;
w  
Qiang Xue committed
716
	}
w  
Qiang Xue committed
717

Qiang Xue committed
718 719 720 721
	/**
	 * @return string the JOIN clause built from [[query]].
	 */
	protected function buildJoin()
w  
Qiang Xue committed
722
	{
Qiang Xue committed
723
		$joins = $this->query->join;
w  
Qiang Xue committed
724 725 726 727 728 729
		if (empty($joins)) {
			return '';
		}
		if (is_string($joins)) {
			return $joins;
		}
w  
Qiang Xue committed
730

w  
Qiang Xue committed
731
		foreach ($joins as $i => $join) {
Qiang Xue committed
732
			if (is_array($join)) { // join type, table name, on-condition
w  
Qiang Xue committed
733 734
				if (isset($join[0], $join[1])) {
					$table = $join[1];
Qiang Xue committed
735
					if ($this->autoQuote && strpos($table, '(') === false) {
Qiang Xue committed
736
						if (preg_match('/^(.*?)(?i:\s+as\s+|\s+)(.*)$/', $table, $matches)) { // with alias
Qiang Xue committed
737
							$table = $this->driver->quoteTableName($matches[1]) . ' ' . $this->driver->quoteTableName($matches[2]);
Qiang Xue committed
738
						} else {
Qiang Xue committed
739
							$table = $this->driver->quoteTableName($table);
w  
Qiang Xue committed
740 741 742
						}
					}
					$joins[$i] = strtoupper($join[0]) . ' ' . $table;
Qiang Xue committed
743
					if (isset($join[2])) { // join condition
Qiang Xue committed
744
						$joins[$i] .= ' ON ' . $this->buildCondition($join[2]);
w  
Qiang Xue committed
745
					}
Qiang Xue committed
746
				} else {
Qiang Xue committed
747
					throw new Exception('A join clause must be specified as an array of at least two elements.');
w  
Qiang Xue committed
748 749 750
				}
			}
		}
w  
Qiang Xue committed
751

w  
Qiang Xue committed
752
		return implode("\n", $joins);
w  
Qiang Xue committed
753 754
	}

Qiang Xue committed
755 756 757 758
	/**
	 * @return string the WHERE clause built from [[query]].
	 */
	protected function buildWhere()
w  
Qiang Xue committed
759
	{
Qiang Xue committed
760
		$where = $this->buildCondition($this->query->where);
w  
Qiang Xue committed
761
		return empty($where) ? '' : 'WHERE ' . $where;
w  
Qiang Xue committed
762 763
	}

Qiang Xue committed
764 765 766 767
	/**
	 * @return string the GROUP BY clause built from [[query]].
	 */
	protected function buildGroupBy()
w  
Qiang Xue committed
768
	{
Qiang Xue committed
769
		if (empty($this->query->groupBy)) {
w  
Qiang Xue committed
770
			return '';
Qiang Xue committed
771 772
		} else {
			return 'GROUP BY ' . $this->buildColumns($this->query->groupBy);
w  
Qiang Xue committed
773
		}
w  
Qiang Xue committed
774 775
	}

Qiang Xue committed
776 777 778 779
	/**
	 * @return string the HAVING clause built from [[query]].
	 */
	protected function buildHaving()
w  
Qiang Xue committed
780
	{
Qiang Xue committed
781
		$having = $this->buildCondition($this->query->having);
w  
Qiang Xue committed
782
		return empty($having) ? '' : 'HAVING ' . $having;
w  
Qiang Xue committed
783 784
	}

Qiang Xue committed
785 786 787 788
	/**
	 * @return string the ORDER BY clause built from [[query]].
	 */
	protected function buildOrderBy()
w  
Qiang Xue committed
789
	{
Qiang Xue committed
790
		if (empty($this->query->orderBy)) {
w  
Qiang Xue committed
791 792
			return '';
		}
Qiang Xue committed
793 794 795 796 797
		$columns = $this->query->orderBy;
		if ($this->autoQuote) {
			if (!is_array($columns)) {
				if (strpos($columns, '(') !== false) {
					return 'ORDER BY ' . $columns;
Qiang Xue committed
798
				} else {
Qiang Xue committed
799
					$columns = preg_split('/\s*,\s*/', trim($columns), -1, PREG_SPLIT_NO_EMPTY);
w  
Qiang Xue committed
800 801
				}
			}
Qiang Xue committed
802 803 804 805 806 807 808 809 810 811 812 813 814 815
			foreach ($columns as $i => $column) {
				if (is_object($column)) {
					$columns[$i] = (string)$column;
				} elseif (strpos($column, '(') === false) {
					if (preg_match('/^(.*?)\s+(asc|desc)$/i', $column, $matches)) {
						$columns[$i] = $this->driver->quoteColumnName($matches[1]) . ' ' . $matches[2];
					} else {
						$columns[$i] = $this->driver->quoteColumnName($column);
					}
				}
			}
		}
		if (is_array($columns)) {
			$columns = implode(', ', $columns);
w  
Qiang Xue committed
816
		}
Qiang Xue committed
817
		return 'ORDER BY ' . $columns;
w  
Qiang Xue committed
818 819
	}

Qiang Xue committed
820 821 822 823
	/**
	 * @return string the LIMIT and OFFSET clauses built from [[query]].
	 */
	protected function buildLimit()
w  
Qiang Xue committed
824
	{
w  
Qiang Xue committed
825
		$sql = '';
Qiang Xue committed
826 827
		if ($this->query->limit !== null && $this->query->limit >= 0) {
			$sql = 'LIMIT ' . (int)$this->query->limit;
w  
Qiang Xue committed
828
		}
Qiang Xue committed
829 830
		if ($this->query->offset > 0) {
			$sql .= ' OFFSET ' . (int)$this->query->offset;
w  
Qiang Xue committed
831 832
		}
		return ltrim($sql);
w  
Qiang Xue committed
833 834
	}

Qiang Xue committed
835 836 837 838
	/**
	 * @return string the UNION clause built from [[query]].
	 */
	protected function buildUnion()
w  
Qiang Xue committed
839
	{
Qiang Xue committed
840
		$unions = $this->query->union;
w  
Qiang Xue committed
841 842 843 844 845 846 847 848
		if (empty($unions)) {
			return '';
		}
		if (!is_array($unions)) {
			$unions = array($unions);
		}
		foreach ($unions as $i => $union) {
			if ($union instanceof Query) {
w  
Qiang Xue committed
849
				$unions[$i] = $union->getSql($this->connection);
w  
Qiang Xue committed
850 851 852
			}
		}
		return "UNION (\n" . implode("\n) UNION (\n", $unions) . "\n)";
w  
Qiang Xue committed
853
	}
Qiang Xue committed
854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913

	/**
	 * Processes columns and properly quote them if necessary.
	 * This method will quote columns if [[autoQuote]] is true.
	 * It will join all columns into a string with comma as separators.
	 * @param string|array $columns the columns to be processed
	 * @return string the processing result
	 */
	protected function buildColumns($columns)
	{
		if ($this->autoQuote) {
			if (!is_array($columns)) {
				if (strpos($columns, '(') !== false) {
					return $columns;
				} else {
					$columns = preg_split('/\s*,\s*/', $columns, -1, PREG_SPLIT_NO_EMPTY);
				}
			}
			foreach ($columns as $i => $column) {
				if (is_object($column)) {
					$columns[$i] = (string)$column;
				} elseif (strpos($column, '(') === false) {
					$columns[$i] = $this->driver->quoteColumnName($column);
				}
			}
		}
		return is_array($columns) ? implode(', ', $columns) : $columns;
	}

	/**
	 * Quotes a table name for use in a query.
	 * This method will perform name quoting only when [[autoQuote]] is true.
	 * @param string $name table name
	 * @param boolean $simple whether the name should be treated as a simple table name without any prefix.
	 * @return string the properly quoted table name
	 */
	protected function quoteTableName($name, $simple = false)
	{
		if ($this->autoQuote) {
			return $simple ? $this->driver->quoteSimpleTableName($name) : $this->driver->quoteTableName($name);
		} else {
			return $name;
		}
	}

	/**
	 * Quotes a column name for use in a query.
	 * This method will perform name quoting only when [[autoQuote]] is true.
	 * @param string $name column name
	 * @param boolean $simple whether the name should be treated as a simple column name without any prefix.
	 * @return string the properly quoted column name
	 */
	protected function quoteColumnName($name, $simple = false)
	{
		if ($this->autoQuote) {
			return $simple ? $this->driver->quoteSimpleColumnName($name) : $this->driver->quoteColumnName($name);
		} else {
			return $name;
		}
	}
w  
Qiang Xue committed
914
}