grammar.php 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426
  1. <?php namespace Laravel\Database\Query\Grammars;
  2. use Laravel\Database\Query;
  3. use Laravel\Database\Expression;
  4. class Grammar extends \Laravel\Database\Grammar {
  5. /**
  6. * All of the query componenets in the order they should be built.
  7. *
  8. * @var array
  9. */
  10. protected $components = array(
  11. 'aggregate', 'selects', 'from', 'joins', 'wheres',
  12. 'groupings', 'orderings', 'limit', 'offset',
  13. );
  14. /**
  15. * Compile a SQL SELECT statement from a Query instance.
  16. *
  17. * @param Query $query
  18. * @return string
  19. */
  20. public function select(Query $query)
  21. {
  22. return $this->concatenate($this->components($query));
  23. }
  24. /**
  25. * Generate the SQL for every component of the query.
  26. *
  27. * @param Query $query
  28. * @return array
  29. */
  30. final protected function components($query)
  31. {
  32. // Each portion of the statement is compiled by a function corresponding
  33. // to an item in the components array. This lets us to keep the creation
  34. // of the query very granular and very flexible.
  35. foreach ($this->components as $component)
  36. {
  37. if ( ! is_null($query->$component))
  38. {
  39. $sql[$component] = call_user_func(array($this, $component), $query);
  40. }
  41. }
  42. return (array) $sql;
  43. }
  44. /**
  45. * Concatenate an array of SQL segments, removing those that are empty.
  46. *
  47. * @param array $components
  48. * @return string
  49. */
  50. final protected function concatenate($components)
  51. {
  52. return implode(' ', array_filter($components, function($value)
  53. {
  54. return (string) $value !== '';
  55. }));
  56. }
  57. /**
  58. * Compile the SELECT clause for a query.
  59. *
  60. * @param Query $query
  61. * @return string
  62. */
  63. protected function selects(Query $query)
  64. {
  65. if ( ! is_null($query->aggregate)) return;
  66. $select = ($query->distinct) ? 'SELECT DISTINCT ' : 'SELECT ';
  67. return $select.$this->columnize($query->selects);
  68. }
  69. /**
  70. * Compile an aggregating SELECT clause for a query.
  71. *
  72. * @param Query $query
  73. * @return string
  74. */
  75. protected function aggregate(Query $query)
  76. {
  77. $column = $this->columnize($query->aggregate['columns']);
  78. // If the "distinct" flag is set and we're not aggregating everything
  79. // we'll set the distinct clause on the query, since this is used
  80. // to count all of the distinct values in a column, etc.
  81. if ($query->distinct and $column !== '*')
  82. {
  83. $column = 'DISTINCT '.$column;
  84. }
  85. return 'SELECT '.$query->aggregate['aggregator'].'('.$column.') AS '.$this->wrap('aggregate');
  86. }
  87. /**
  88. * Compile the FROM clause for a query.
  89. *
  90. * @param Query $query
  91. * @return string
  92. */
  93. protected function from(Query $query)
  94. {
  95. return 'FROM '.$this->wrap_table($query->from);
  96. }
  97. /**
  98. * Compile the JOIN clauses for a query.
  99. *
  100. * @param Query $query
  101. * @return string
  102. */
  103. protected function joins(Query $query)
  104. {
  105. // We need to iterate through each JOIN clause that is attached to the
  106. // query an translate it into SQL. The table and the columns will be
  107. // wrapped in identifiers to avoid naming collisions.
  108. foreach ($query->joins as $join)
  109. {
  110. $table = $this->wrap_table($join->table);
  111. $clauses = array();
  112. // Each JOIN statement may have multiple clauses, so we will iterate
  113. // through each clause creating the conditions then we'll join all
  114. // of the together at the end to build the clause.
  115. foreach ($join->clauses as $clause)
  116. {
  117. extract($clause);
  118. $column1 = $this->wrap($column1);
  119. $column2 = $this->wrap($column2);
  120. $clauses[] = "{$connector} {$column1} {$operator} {$column2}";
  121. }
  122. // The first clause will have a connector on the front, but it is
  123. // not needed on the first condition, so we will strip it off of
  124. // the condition before adding it to the arrya of joins.
  125. $search = array('AND ', 'OR ');
  126. $clauses[0] = str_replace($search, '', $clauses[0]);
  127. $clauses = implode(' ', $clauses);
  128. $sql[] = "{$join->type} JOIN {$table} ON {$clauses}";
  129. }
  130. // Finally, we should have an array of JOIN clauses that we can
  131. // implode together and return as the complete SQL for the
  132. // join clause of the query under construction.
  133. return implode(' ', $sql);
  134. }
  135. /**
  136. * Compile the WHERE clause for a query.
  137. *
  138. * @param Query $query
  139. * @return string
  140. */
  141. final protected function wheres(Query $query)
  142. {
  143. if (is_null($query->wheres)) return '';
  144. // Each WHERE clause array has a "type" that is assigned by the query
  145. // builder, and each type has its own compiler function. We will call
  146. // the appropriate compiler for each where clause.
  147. foreach ($query->wheres as $where)
  148. {
  149. $sql[] = $where['connector'].' '.$this->{$where['type']}($where);
  150. }
  151. if (isset($sql))
  152. {
  153. // We attach the boolean connector to every where segment just
  154. // for convenience. Once we have built the entire clause we'll
  155. // remove the first instance of a connector.
  156. return 'WHERE '.preg_replace('/AND |OR /', '', implode(' ', $sql), 1);
  157. }
  158. }
  159. /**
  160. * Compile a nested WHERE clause.
  161. *
  162. * @param array $where
  163. * @return string
  164. */
  165. protected function where_nested($where)
  166. {
  167. return '('.substr($this->wheres($where['query']), 6).')';
  168. }
  169. /**
  170. * Compile a simple WHERE clause.
  171. *
  172. * @param array $where
  173. * @return string
  174. */
  175. protected function where($where)
  176. {
  177. $parameter = $this->parameter($where['value']);
  178. return $this->wrap($where['column']).' '.$where['operator'].' '.$parameter;
  179. }
  180. /**
  181. * Compile a WHERE IN clause.
  182. *
  183. * @param array $where
  184. * @return string
  185. */
  186. protected function where_in($where)
  187. {
  188. $parameters = $this->parameterize($where['values']);
  189. return $this->wrap($where['column']).' IN ('.$parameters.')';
  190. }
  191. /**
  192. * Compile a WHERE NOT IN clause.
  193. *
  194. * @param array $where
  195. * @return string
  196. */
  197. protected function where_not_in($where)
  198. {
  199. $parameters = $this->parameterize($where['values']);
  200. return $this->wrap($where['column']).' NOT IN ('.$parameters.')';
  201. }
  202. /**
  203. * Compile a WHERE NULL clause.
  204. *
  205. * @param array $where
  206. * @return string
  207. */
  208. protected function where_null($where)
  209. {
  210. return $this->wrap($where['column']).' IS NULL';
  211. }
  212. /**
  213. * Compile a WHERE NULL clause.
  214. *
  215. * @param array $where
  216. * @return string
  217. */
  218. protected function where_not_null($where)
  219. {
  220. return $this->wrap($where['column']).' IS NOT NULL';
  221. }
  222. /**
  223. * Compile a raw WHERE clause.
  224. *
  225. * @param array $where
  226. * @return string
  227. */
  228. final protected function where_raw($where)
  229. {
  230. return $where['sql'];
  231. }
  232. /**
  233. * Compile the GROUP BY clause for a query.
  234. *
  235. * @param Query $query
  236. * @return string
  237. */
  238. protected function groupings(Query $query)
  239. {
  240. return 'GROUP BY '.$this->columnize($query->groupings);
  241. }
  242. /**
  243. * Compile the ORDER BY clause for a query.
  244. *
  245. * @param Query $query
  246. * @return string
  247. */
  248. protected function orderings(Query $query)
  249. {
  250. foreach ($query->orderings as $ordering)
  251. {
  252. $sql[] = $this->wrap($ordering['column']).' '.strtoupper($ordering['direction']);
  253. }
  254. return 'ORDER BY '.implode(', ', $sql);
  255. }
  256. /**
  257. * Compile the LIMIT clause for a query.
  258. *
  259. * @param Query $query
  260. * @return string
  261. */
  262. protected function limit(Query $query)
  263. {
  264. return 'LIMIT '.$query->limit;
  265. }
  266. /**
  267. * Compile the OFFSET clause for a query.
  268. *
  269. * @param Query $query
  270. * @return string
  271. */
  272. protected function offset(Query $query)
  273. {
  274. return 'OFFSET '.$query->offset;
  275. }
  276. /**
  277. * Compile a SQL INSERT statment from a Query instance.
  278. *
  279. * This method handles the compilation of single row inserts and batch inserts.
  280. *
  281. * @param Query $query
  282. * @param array $values
  283. * @return string
  284. */
  285. public function insert(Query $query, $values)
  286. {
  287. $table = $this->wrap_table($query->from);
  288. // Force every insert to be treated like a batch insert. This simply makes
  289. // creating the SQL syntax a little easier on us since we can always treat
  290. // the values as if it contains multiple inserts.
  291. if ( ! is_array(reset($values))) $values = array($values);
  292. // Since we only care about the column names, we can pass any of the insert
  293. // arrays into the "columnize" method. The columns should be the same for
  294. // every record inserted into the table.
  295. $columns = $this->columnize(array_keys(reset($values)));
  296. // Build the list of parameter place-holders of values bound to the query.
  297. // Each insert should have the same number of bound paramters, so we can
  298. // just use the first array of values.
  299. $parameters = $this->parameterize(reset($values));
  300. $parameters = implode(', ', array_fill(0, count($values), "($parameters)"));
  301. return "INSERT INTO {$table} ({$columns}) VALUES {$parameters}";
  302. }
  303. /**
  304. * Compile a SQL UPDATE statment from a Query instance.
  305. *
  306. * @param Query $query
  307. * @param array $values
  308. * @return string
  309. */
  310. public function update(Query $query, $values)
  311. {
  312. $table = $this->wrap_table($query->from);
  313. // Each column in the UPDATE statement needs to be wrapped in the keyword
  314. // identifiers, and a place-holder needs to be created for each value in
  315. // the array of bindings, so we'll build the sets first.
  316. foreach ($values as $column => $value)
  317. {
  318. $columns[] = $this->wrap($column).' = '.$this->parameter($value);
  319. }
  320. $columns = implode(', ', $columns);
  321. // UPDATE statements may be constrained by a WHERE clause, so we'll run
  322. // the entire where compilation process for those contraints. This is
  323. // easily achieved by passing it to the "wheres" method.
  324. return trim("UPDATE {$table} SET {$columns} ".$this->wheres($query));
  325. }
  326. /**
  327. * Compile a SQL DELETE statment from a Query instance.
  328. *
  329. * @param Query $query
  330. * @return string
  331. */
  332. public function delete(Query $query)
  333. {
  334. $table = $this->wrap_table($query->from);
  335. return trim("DELETE FROM {$table} ".$this->wheres($query));
  336. }
  337. /**
  338. * Transform an SQL short-cuts into real SQL for PDO.
  339. *
  340. * @param string $sql
  341. * @param array $bindings
  342. * @return string
  343. */
  344. public function shortcut($sql, &$bindings)
  345. {
  346. // Laravel provides an easy short-cut notation for writing raw WHERE IN
  347. // statements. If (...) is in the query, it will be replaced with the
  348. // correct number of parameters based on the query bindings.
  349. if (strpos($sql, '(...)') !== false)
  350. {
  351. for ($i = 0; $i < count($bindings); $i++)
  352. {
  353. // If the binding is an array, we can just assume it's used to fill a
  354. // where in condition, so we'll just replace the next place-holder
  355. // in the query with the constraint and splice the bindings.
  356. if (is_array($bindings[$i]))
  357. {
  358. $parameters = $this->parameterize($bindings[$i]);
  359. array_splice($bindings, $i, 1, $bindings[$i]);
  360. $sql = preg_replace('~\(\.\.\.\)~', "({$parameters})", $sql, 1);
  361. }
  362. }
  363. }
  364. return trim($sql);
  365. }
  366. }