QueryTrait.php 14KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391
  1. <?php
  2. /**
  3. * @link http://www.yiiframework.com/
  4. * @copyright Copyright (c) 2008 Yii Software LLC
  5. * @license http://www.yiiframework.com/license/
  6. */
  7. namespace yii\db;
  8. use yii\base\NotSupportedException;
  9. /**
  10. * The BaseQuery trait represents the minimum method set of a database Query.
  11. *
  12. * It is supposed to be used in a class that implements the [[QueryInterface]].
  13. *
  14. * @author Qiang Xue <qiang.xue@gmail.com>
  15. * @author Carsten Brandt <mail@cebe.cc>
  16. * @since 2.0
  17. */
  18. trait QueryTrait
  19. {
  20. /**
  21. * @var string|array query condition. This refers to the WHERE clause in a SQL statement.
  22. * For example, `['age' => 31, 'team' => 1]`.
  23. * @see where() for valid syntax on specifying this value.
  24. */
  25. public $where;
  26. /**
  27. * @var integer maximum number of records to be returned. If not set or less than 0, it means no limit.
  28. */
  29. public $limit;
  30. /**
  31. * @var integer zero-based offset from where the records are to be returned. If not set or
  32. * less than 0, it means starting from the beginning.
  33. */
  34. public $offset;
  35. /**
  36. * @var array how to sort the query results. This is used to construct the ORDER BY clause in a SQL statement.
  37. * The array keys are the columns to be sorted by, and the array values are the corresponding sort directions which
  38. * can be either [SORT_ASC](http://php.net/manual/en/array.constants.php#constant.sort-asc)
  39. * or [SORT_DESC](http://php.net/manual/en/array.constants.php#constant.sort-desc).
  40. * The array may also contain [[Expression]] objects. If that is the case, the expressions
  41. * will be converted into strings without any change.
  42. */
  43. public $orderBy;
  44. /**
  45. * @var string|callable $column the name of the column by which the query results should be indexed by.
  46. * This can also be a callable (e.g. anonymous function) that returns the index value based on the given
  47. * row data. For more details, see [[indexBy()]]. This property is only used by [[QueryInterface::all()|all()]].
  48. */
  49. public $indexBy;
  50. /**
  51. * Sets the [[indexBy]] property.
  52. * @param string|callable $column the name of the column by which the query results should be indexed by.
  53. * This can also be a callable (e.g. anonymous function) that returns the index value based on the given
  54. * row data. The signature of the callable should be:
  55. *
  56. * ```php
  57. * function ($row)
  58. * {
  59. * // return the index value corresponding to $row
  60. * }
  61. * ```
  62. *
  63. * @return $this the query object itself
  64. */
  65. public function indexBy($column)
  66. {
  67. $this->indexBy = $column;
  68. return $this;
  69. }
  70. /**
  71. * Sets the WHERE part of the query.
  72. *
  73. * See [[QueryInterface::where()]] for detailed documentation.
  74. *
  75. * @param string|array $condition the conditions that should be put in the WHERE part.
  76. * @return $this the query object itself
  77. * @see andWhere()
  78. * @see orWhere()
  79. */
  80. public function where($condition)
  81. {
  82. $this->where = $condition;
  83. return $this;
  84. }
  85. /**
  86. * Adds an additional WHERE condition to the existing one.
  87. * The new condition and the existing one will be joined using the 'AND' operator.
  88. * @param string|array $condition the new WHERE condition. Please refer to [[where()]]
  89. * on how to specify this parameter.
  90. * @return $this the query object itself
  91. * @see where()
  92. * @see orWhere()
  93. */
  94. public function andWhere($condition)
  95. {
  96. if ($this->where === null) {
  97. $this->where = $condition;
  98. } else {
  99. $this->where = ['and', $this->where, $condition];
  100. }
  101. return $this;
  102. }
  103. /**
  104. * Adds an additional WHERE condition to the existing one.
  105. * The new condition and the existing one will be joined using the 'OR' operator.
  106. * @param string|array $condition the new WHERE condition. Please refer to [[where()]]
  107. * on how to specify this parameter.
  108. * @return $this the query object itself
  109. * @see where()
  110. * @see andWhere()
  111. */
  112. public function orWhere($condition)
  113. {
  114. if ($this->where === null) {
  115. $this->where = $condition;
  116. } else {
  117. $this->where = ['or', $this->where, $condition];
  118. }
  119. return $this;
  120. }
  121. /**
  122. * Sets the WHERE part of the query but ignores [[isEmpty()|empty operands]].
  123. *
  124. * This method is similar to [[where()]]. The main difference is that this method will
  125. * remove [[isEmpty()|empty query operands]]. As a result, this method is best suited
  126. * for building query conditions based on filter values entered by users.
  127. *
  128. * The following code shows the difference between this method and [[where()]]:
  129. *
  130. * ```php
  131. * // WHERE `age`=:age
  132. * $query->filterWhere(['name' => null, 'age' => 20]);
  133. * // WHERE `age`=:age
  134. * $query->where(['age' => 20]);
  135. * // WHERE `name` IS NULL AND `age`=:age
  136. * $query->where(['name' => null, 'age' => 20]);
  137. * ```
  138. *
  139. * Note that unlike [[where()]], you cannot pass binding parameters to this method.
  140. *
  141. * @param array $condition the conditions that should be put in the WHERE part.
  142. * See [[where()]] on how to specify this parameter.
  143. * @return $this the query object itself
  144. * @see where()
  145. * @see andFilterWhere()
  146. * @see orFilterWhere()
  147. */
  148. public function filterWhere(array $condition)
  149. {
  150. $condition = $this->filterCondition($condition);
  151. if ($condition !== []) {
  152. $this->where($condition);
  153. }
  154. return $this;
  155. }
  156. /**
  157. * Adds an additional WHERE condition to the existing one but ignores [[isEmpty()|empty operands]].
  158. * The new condition and the existing one will be joined using the 'AND' operator.
  159. *
  160. * This method is similar to [[andWhere()]]. The main difference is that this method will
  161. * remove [[isEmpty()|empty query operands]]. As a result, this method is best suited
  162. * for building query conditions based on filter values entered by users.
  163. *
  164. * @param array $condition the new WHERE condition. Please refer to [[where()]]
  165. * on how to specify this parameter.
  166. * @return $this the query object itself
  167. * @see filterWhere()
  168. * @see orFilterWhere()
  169. */
  170. public function andFilterWhere(array $condition)
  171. {
  172. $condition = $this->filterCondition($condition);
  173. if ($condition !== []) {
  174. $this->andWhere($condition);
  175. }
  176. return $this;
  177. }
  178. /**
  179. * Adds an additional WHERE condition to the existing one but ignores [[isEmpty()|empty operands]].
  180. * The new condition and the existing one will be joined using the 'OR' operator.
  181. *
  182. * This method is similar to [[orWhere()]]. The main difference is that this method will
  183. * remove [[isEmpty()|empty query operands]]. As a result, this method is best suited
  184. * for building query conditions based on filter values entered by users.
  185. *
  186. * @param array $condition the new WHERE condition. Please refer to [[where()]]
  187. * on how to specify this parameter.
  188. * @return $this the query object itself
  189. * @see filterWhere()
  190. * @see andFilterWhere()
  191. */
  192. public function orFilterWhere(array $condition)
  193. {
  194. $condition = $this->filterCondition($condition);
  195. if ($condition !== []) {
  196. $this->orWhere($condition);
  197. }
  198. return $this;
  199. }
  200. /**
  201. * Removes [[isEmpty()|empty operands]] from the given query condition.
  202. *
  203. * @param array $condition the original condition
  204. * @return array the condition with [[isEmpty()|empty operands]] removed.
  205. * @throws NotSupportedException if the condition operator is not supported
  206. */
  207. protected function filterCondition($condition)
  208. {
  209. if (!is_array($condition)) {
  210. return $condition;
  211. }
  212. if (!isset($condition[0])) {
  213. // hash format: 'column1' => 'value1', 'column2' => 'value2', ...
  214. foreach ($condition as $name => $value) {
  215. if ($this->isEmpty($value)) {
  216. unset($condition[$name]);
  217. }
  218. }
  219. return $condition;
  220. }
  221. // operator format: operator, operand 1, operand 2, ...
  222. $operator = array_shift($condition);
  223. switch (strtoupper($operator)) {
  224. case 'NOT':
  225. case 'AND':
  226. case 'OR':
  227. foreach ($condition as $i => $operand) {
  228. $subCondition = $this->filterCondition($operand);
  229. if ($this->isEmpty($subCondition)) {
  230. unset($condition[$i]);
  231. } else {
  232. $condition[$i] = $subCondition;
  233. }
  234. }
  235. if (empty($condition)) {
  236. return [];
  237. }
  238. break;
  239. case 'BETWEEN':
  240. case 'NOT BETWEEN':
  241. if (array_key_exists(1, $condition) && array_key_exists(2, $condition)) {
  242. if ($this->isEmpty($condition[1]) || $this->isEmpty($condition[2])) {
  243. return [];
  244. }
  245. }
  246. break;
  247. default:
  248. if (array_key_exists(1, $condition) && $this->isEmpty($condition[1])) {
  249. return [];
  250. }
  251. }
  252. array_unshift($condition, $operator);
  253. return $condition;
  254. }
  255. /**
  256. * Returns a value indicating whether the give value is "empty".
  257. *
  258. * The value is considered "empty", if one of the following conditions is satisfied:
  259. *
  260. * - it is `null`,
  261. * - an empty string (`''`),
  262. * - a string containing only whitespace characters,
  263. * - or an empty array.
  264. *
  265. * @param mixed $value
  266. * @return boolean if the value is empty
  267. */
  268. protected function isEmpty($value)
  269. {
  270. return $value === '' || $value === [] || $value === null || is_string($value) && trim($value) === '';
  271. }
  272. /**
  273. * Sets the ORDER BY part of the query.
  274. * @param string|array|Expression $columns the columns (and the directions) to be ordered by.
  275. * Columns can be specified in either a string (e.g. `"id ASC, name DESC"`) or an array
  276. * (e.g. `['id' => SORT_ASC, 'name' => SORT_DESC]`).
  277. *
  278. * The method will automatically quote the column names unless a column contains some parenthesis
  279. * (which means the column contains a DB expression).
  280. *
  281. * Note that if your order-by is an expression containing commas, you should always use an array
  282. * to represent the order-by information. Otherwise, the method will not be able to correctly determine
  283. * the order-by columns.
  284. *
  285. * Since version 2.0.7, an [[Expression]] object can be passed to specify the ORDER BY part explicitly in plain SQL.
  286. * @return $this the query object itself
  287. * @see addOrderBy()
  288. */
  289. public function orderBy($columns)
  290. {
  291. $this->orderBy = $this->normalizeOrderBy($columns);
  292. return $this;
  293. }
  294. /**
  295. * Adds additional ORDER BY columns to the query.
  296. * @param string|array|Expression $columns the columns (and the directions) to be ordered by.
  297. * Columns can be specified in either a string (e.g. "id ASC, name DESC") or an array
  298. * (e.g. `['id' => SORT_ASC, 'name' => SORT_DESC]`).
  299. *
  300. * The method will automatically quote the column names unless a column contains some parenthesis
  301. * (which means the column contains a DB expression).
  302. *
  303. * Note that if your order-by is an expression containing commas, you should always use an array
  304. * to represent the order-by information. Otherwise, the method will not be able to correctly determine
  305. * the order-by columns.
  306. *
  307. * Since version 2.0.7, an [[Expression]] object can be passed to specify the ORDER BY part explicitly in plain SQL.
  308. * @return $this the query object itself
  309. * @see orderBy()
  310. */
  311. public function addOrderBy($columns)
  312. {
  313. $columns = $this->normalizeOrderBy($columns);
  314. if ($this->orderBy === null) {
  315. $this->orderBy = $columns;
  316. } else {
  317. $this->orderBy = array_merge($this->orderBy, $columns);
  318. }
  319. return $this;
  320. }
  321. /**
  322. * Normalizes format of ORDER BY data
  323. *
  324. * @param array|string|Expression $columns the columns value to normalize. See [[orderBy]] and [[addOrderBy]].
  325. * @return array
  326. */
  327. protected function normalizeOrderBy($columns)
  328. {
  329. if ($columns instanceof Expression) {
  330. return [$columns];
  331. } elseif (is_array($columns)) {
  332. return $columns;
  333. } else {
  334. $columns = preg_split('/\s*,\s*/', trim($columns), -1, PREG_SPLIT_NO_EMPTY);
  335. $result = [];
  336. foreach ($columns as $column) {
  337. if (preg_match('/^(.*?)\s+(asc|desc)$/i', $column, $matches)) {
  338. $result[$matches[1]] = strcasecmp($matches[2], 'desc') ? SORT_ASC : SORT_DESC;
  339. } else {
  340. $result[$column] = SORT_ASC;
  341. }
  342. }
  343. return $result;
  344. }
  345. }
  346. /**
  347. * Sets the LIMIT part of the query.
  348. * @param integer $limit the limit. Use null or negative value to disable limit.
  349. * @return $this the query object itself
  350. */
  351. public function limit($limit)
  352. {
  353. $this->limit = $limit;
  354. return $this;
  355. }
  356. /**
  357. * Sets the OFFSET part of the query.
  358. * @param integer $offset the offset. Use null or negative value to disable offset.
  359. * @return $this the query object itself
  360. */
  361. public function offset($offset)
  362. {
  363. $this->offset = $offset;
  364. return $this;
  365. }
  366. }