query.php 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850
  1. <?php namespace Laravel\Database;
  2. use Closure;
  3. use Laravel\Database;
  4. use Laravel\Paginator;
  5. use Laravel\Database\Query\Grammars\Grammar;
  6. use Laravel\Database\Query\Grammars\SQLServer;
  7. class Query {
  8. /**
  9. * The database connection.
  10. *
  11. * @var Connection
  12. */
  13. public $connection;
  14. /**
  15. * The query grammar instance.
  16. *
  17. * @var Query\Grammars\Grammar
  18. */
  19. public $grammar;
  20. /**
  21. * The SELECT clause.
  22. *
  23. * @var array
  24. */
  25. public $selects;
  26. /**
  27. * The aggregating column and function.
  28. *
  29. * @var array
  30. */
  31. public $aggregate;
  32. /**
  33. * Indicates if the query should return distinct results.
  34. *
  35. * @var bool
  36. */
  37. public $distinct = false;
  38. /**
  39. * The table name.
  40. *
  41. * @var string
  42. */
  43. public $from;
  44. /**
  45. * The table joins.
  46. *
  47. * @var array
  48. */
  49. public $joins;
  50. /**
  51. * The WHERE clauses.
  52. *
  53. * @var array
  54. */
  55. public $wheres;
  56. /**
  57. * The GROUP BY clauses.
  58. *
  59. * @var array
  60. */
  61. public $groupings;
  62. /**
  63. * The ORDER BY clauses.
  64. *
  65. * @var array
  66. */
  67. public $orderings;
  68. /**
  69. * The LIMIT value.
  70. *
  71. * @var int
  72. */
  73. public $limit;
  74. /**
  75. * The OFFSET value.
  76. *
  77. * @var int
  78. */
  79. public $offset;
  80. /**
  81. * The query value bindings.
  82. *
  83. * @var array
  84. */
  85. public $bindings = array();
  86. /**
  87. * Create a new query instance.
  88. *
  89. * @param Connection $connection
  90. * @param Grammar $grammar
  91. * @param string $table
  92. * @return void
  93. */
  94. public function __construct(Connection $connection, Grammar $grammar, $table)
  95. {
  96. $this->from = $table;
  97. $this->grammar = $grammar;
  98. $this->connection = $connection;
  99. }
  100. /**
  101. * Force the query to return distinct results.
  102. *
  103. * @return Query
  104. */
  105. public function distinct()
  106. {
  107. $this->distinct = true;
  108. return $this;
  109. }
  110. /**
  111. * Add an array of columns to the SELECT clause.
  112. *
  113. * @param array $columns
  114. * @return Query
  115. */
  116. public function select($columns = array('*'))
  117. {
  118. $this->selects = (array) $columns;
  119. return $this;
  120. }
  121. /**
  122. * Add a join clause to the query.
  123. *
  124. * @param string $table
  125. * @param string $column1
  126. * @param string $operator
  127. * @param string $column2
  128. * @param string $type
  129. * @return Query
  130. */
  131. public function join($table, $column1, $operator = null, $column2 = null, $type = 'INNER')
  132. {
  133. // If the "column" is really an instance of a Closure, the developer is
  134. // trying to create a join with a complex "ON" clause. So, we will add
  135. // the join, and then call the Closure with the join.
  136. if ($column1 instanceof Closure)
  137. {
  138. $this->joins[] = new Query\Join($type, $table);
  139. call_user_func($column1, end($this->joins));
  140. }
  141. // If the column is just a string, we can assume that the join just
  142. // has a simple on clause, and we'll create the join instance and
  143. // add the clause automatically for the develoepr.
  144. else
  145. {
  146. $join = new Query\Join($type, $table);
  147. $join->on($column1, $operator, $column2);
  148. $this->joins[] = $join;
  149. }
  150. return $this;
  151. }
  152. /**
  153. * Add a left join to the query.
  154. *
  155. * @param string $table
  156. * @param string $column1
  157. * @param string $operator
  158. * @param string $column2
  159. * @return Query
  160. */
  161. public function left_join($table, $column1, $operator, $column2)
  162. {
  163. return $this->join($table, $column1, $operator, $column2, 'LEFT');
  164. }
  165. /**
  166. * Reset the where clause to its initial state.
  167. *
  168. * @return void
  169. */
  170. public function reset_where()
  171. {
  172. list($this->wheres, $this->bindings) = array(array(), array());
  173. }
  174. /**
  175. * Add a raw where condition to the query.
  176. *
  177. * @param string $where
  178. * @param array $bindings
  179. * @param string $connector
  180. * @return Query
  181. */
  182. public function raw_where($where, $bindings = array(), $connector = 'AND')
  183. {
  184. $this->wheres[] = array('type' => 'where_raw', 'connector' => $connector, 'sql' => $where);
  185. $this->bindings = array_merge($this->bindings, $bindings);
  186. return $this;
  187. }
  188. /**
  189. * Add a raw or where condition to the query.
  190. *
  191. * @param string $where
  192. * @param array $bindings
  193. * @return Query
  194. */
  195. public function raw_or_where($where, $bindings = array())
  196. {
  197. return $this->raw_where($where, $bindings, 'OR');
  198. }
  199. /**
  200. * Add a where condition to the query.
  201. *
  202. * @param string $column
  203. * @param string $operator
  204. * @param mixed $value
  205. * @param string $connector
  206. * @return Query
  207. */
  208. public function where($column, $operator = null, $value = null, $connector = 'AND')
  209. {
  210. // If a CLosure is passed into the method, it means a nested where
  211. // clause is being initiated, so we will take a different course
  212. // of action than when the statement is just a simple where.
  213. if ($column instanceof Closure)
  214. {
  215. return $this->where_nested($column, $connector);
  216. }
  217. $type = 'where';
  218. $this->wheres[] = compact('type', 'column', 'operator', 'value', 'connector');
  219. $this->bindings[] = $value;
  220. return $this;
  221. }
  222. /**
  223. * Add an or where condition to the query.
  224. *
  225. * @param string $column
  226. * @param string $operator
  227. * @param mixed $value
  228. * @return Query
  229. */
  230. public function or_where($column, $operator = null, $value = null)
  231. {
  232. return $this->where($column, $operator, $value, 'OR');
  233. }
  234. /**
  235. * Add an or where condition for the primary key to the query.
  236. *
  237. * @param mixed $value
  238. * @return Query
  239. */
  240. public function or_where_id($value)
  241. {
  242. return $this->or_where('id', '=', $value);
  243. }
  244. /**
  245. * Add a where in condition to the query.
  246. *
  247. * @param string $column
  248. * @param array $values
  249. * @param string $connector
  250. * @param bool $not
  251. * @return Query
  252. */
  253. public function where_in($column, $values, $connector = 'AND', $not = false)
  254. {
  255. $type = ($not) ? 'where_not_in' : 'where_in';
  256. $this->wheres[] = compact('type', 'column', 'values', 'connector');
  257. $this->bindings = array_merge($this->bindings, $values);
  258. return $this;
  259. }
  260. /**
  261. * Add an or where in condition to the query.
  262. *
  263. * @param string $column
  264. * @param array $values
  265. * @return Query
  266. */
  267. public function or_where_in($column, $values)
  268. {
  269. return $this->where_in($column, $values, 'OR');
  270. }
  271. /**
  272. * Add a where not in condition to the query.
  273. *
  274. * @param string $column
  275. * @param array $values
  276. * @param string $connector
  277. * @return Query
  278. */
  279. public function where_not_in($column, $values, $connector = 'AND')
  280. {
  281. return $this->where_in($column, $values, $connector, true);
  282. }
  283. /**
  284. * Add an or where not in condition to the query.
  285. *
  286. * @param string $column
  287. * @param array $values
  288. * @return Query
  289. */
  290. public function or_where_not_in($column, $values)
  291. {
  292. return $this->where_not_in($column, $values, 'OR');
  293. }
  294. /**
  295. * Add a where null condition to the query.
  296. *
  297. * @param string $column
  298. * @param string $connector
  299. * @param bool $not
  300. * @return Query
  301. */
  302. public function where_null($column, $connector = 'AND', $not = false)
  303. {
  304. $type = ($not) ? 'where_not_null' : 'where_null';
  305. $this->wheres[] = compact('type', 'column', 'connector');
  306. return $this;
  307. }
  308. /**
  309. * Add an or where null condition to the query.
  310. *
  311. * @param string $column
  312. * @return Query
  313. */
  314. public function or_where_null($column)
  315. {
  316. return $this->where_null($column, 'OR');
  317. }
  318. /**
  319. * Add a where not null condition to the query.
  320. *
  321. * @param string $column
  322. * @param string $connector
  323. * @return Query
  324. */
  325. public function where_not_null($column, $connector = 'AND')
  326. {
  327. return $this->where_null($column, $connector, true);
  328. }
  329. /**
  330. * Add an or where not null condition to the query.
  331. *
  332. * @param string $column
  333. * @return Query
  334. */
  335. public function or_where_not_null($column)
  336. {
  337. return $this->where_not_null($column, 'OR');
  338. }
  339. /**
  340. * Add a nested where condition to the query.
  341. *
  342. * @param Closure $callback
  343. * @param string $connector
  344. * @return Query
  345. */
  346. protected function where_nested($callback, $connector)
  347. {
  348. $type = 'where_nested';
  349. // To handle a nested where statement, we will actually instantiate a
  350. // new Query instance and run the callback over that instance, which
  351. // will allow the developer to have a fresh query to work with.
  352. $query = new Query($this->connection, $this->grammar, $this->from);
  353. // Once the callback has been run on the query, we will store the
  354. // nested query instance on the where clause array so that it's
  355. // passed to the query grammar.
  356. call_user_func($callback, $query);
  357. $this->wheres[] = compact('type', 'query', 'connector');
  358. $this->bindings = array_merge($this->bindings, $query->bindings);
  359. return $this;
  360. }
  361. /**
  362. * Add dynamic where conditions to the query.
  363. *
  364. * @param string $method
  365. * @param array $parameters
  366. * @return Query
  367. */
  368. private function dynamic_where($method, $parameters)
  369. {
  370. $finder = substr($method, 6);
  371. $flags = PREG_SPLIT_DELIM_CAPTURE;
  372. $segments = preg_split('/(_and_|_or_)/i', $finder, -1, $flags);
  373. // The connector variable will determine which connector will be
  374. // used for the condition. We'll change it as we come across new
  375. // connectors in the dynamic method string.
  376. //
  377. // The index variable helps us get the correct parameter value
  378. // for the where condition. We increment it each time we add
  379. // a condition to the query.
  380. $connector = 'AND';
  381. $index = 0;
  382. foreach ($segments as $segment)
  383. {
  384. // If the segment is not a boolean connector, we can assume it
  385. // it is a column name, and we'll add it to the query as a new
  386. // where clause.
  387. //
  388. // Otherwise, we'll store the connector so that we know how to
  389. // connection the next where clause we find to the query, as
  390. // all connectors should precede a new where clause.
  391. if ($segment != '_and_' and $segment != '_or_')
  392. {
  393. $this->where($segment, '=', $parameters[$index], $connector);
  394. $index++;
  395. }
  396. else
  397. {
  398. $connector = trim(strtoupper($segment), '_');
  399. }
  400. }
  401. return $this;
  402. }
  403. /**
  404. * Add a grouping to the query.
  405. *
  406. * @param string $column
  407. * @return Query
  408. */
  409. public function group_by($column)
  410. {
  411. $this->groupings[] = $column;
  412. return $this;
  413. }
  414. /**
  415. * Add an ordering to the query.
  416. *
  417. * @param string $column
  418. * @param string $direction
  419. * @return Query
  420. */
  421. public function order_by($column, $direction = 'asc')
  422. {
  423. $this->orderings[] = compact('column', 'direction');
  424. return $this;
  425. }
  426. /**
  427. * Set the query offset.
  428. *
  429. * @param int $value
  430. * @return Query
  431. */
  432. public function skip($value)
  433. {
  434. $this->offset = $value;
  435. return $this;
  436. }
  437. /**
  438. * Set the query limit.
  439. *
  440. * @param int $value
  441. * @return Query
  442. */
  443. public function take($value)
  444. {
  445. $this->limit = $value;
  446. return $this;
  447. }
  448. /**
  449. * Set the query limit and offset for a given page.
  450. *
  451. * @param int $page
  452. * @param int $per_page
  453. * @return Query
  454. */
  455. public function for_page($page, $per_page)
  456. {
  457. return $this->skip(($page - 1) * $per_page)->take($per_page);
  458. }
  459. /**
  460. * Find a record by the primary key.
  461. *
  462. * @param int $id
  463. * @param array $columns
  464. * @return object
  465. */
  466. public function find($id, $columns = array('*'))
  467. {
  468. return $this->where('id', '=', $id)->first($columns);
  469. }
  470. /**
  471. * Execute the query as a SELECT statement and return a single column.
  472. *
  473. * @param string $column
  474. * @return mixed
  475. */
  476. public function only($column)
  477. {
  478. $sql = $this->grammar->select($this->select(array($column)));
  479. return $this->connection->only($sql, $this->bindings);
  480. }
  481. /**
  482. * Execute the query as a SELECT statement and return the first result.
  483. *
  484. * @param array $columns
  485. * @return mixed
  486. */
  487. public function first($columns = array('*'))
  488. {
  489. $columns = (array) $columns;
  490. // Since we only need the first result, we'll go ahead and set the
  491. // limit clause to 1, since this will be much faster than getting
  492. // all of the rows and then only returning the first.
  493. $results = $this->take(1)->get($columns);
  494. return (count($results) > 0) ? $results[0] : null;
  495. }
  496. /**
  497. * Get an array with the values of a given column.
  498. *
  499. * @param string $column
  500. * @param string $key
  501. * @return array
  502. */
  503. public function lists($column, $key = null)
  504. {
  505. $columns = (is_null($key)) ? array($column) : array($column, $key);
  506. $results = $this->get($columns);
  507. // First we will get the array of values for the requested column.
  508. // Of course, this array will simply have numeric keys. After we
  509. // have this array we will determine if we need to key the array
  510. // by another column from the result set.
  511. $values = array_map(function($row) use ($column)
  512. {
  513. return $row->$column;
  514. }, $results);
  515. // If a key was provided, we will extract an array of keys and
  516. // set the keys on the array of values using the array_combine
  517. // function provided by PHP, which should give us the proper
  518. // array form to return from the method.
  519. if ( ! is_null($key))
  520. {
  521. return array_combine(array_map(function($row) use ($key)
  522. {
  523. return $row->$key;
  524. }, $results), $values);
  525. }
  526. return $values;
  527. }
  528. /**
  529. * Execute the query as a SELECT statement.
  530. *
  531. * @param array $columns
  532. * @return array
  533. */
  534. public function get($columns = array('*'))
  535. {
  536. if (is_null($this->selects)) $this->select($columns);
  537. $sql = $this->grammar->select($this);
  538. $results = $this->connection->query($sql, $this->bindings);
  539. // If the query has an offset and we are using the SQL Server grammar,
  540. // we need to spin through the results and remove the "rownum" from
  541. // each of the objects. Unfortunately SQL Server does not have an
  542. // offset keyword, so we have to use row numbers in the query.
  543. if ($this->offset > 0 and $this->grammar instanceof SQLServer)
  544. {
  545. array_walk($results, function($result)
  546. {
  547. unset($result->rownum);
  548. });
  549. }
  550. // Reset the SELECT clause so more queries can be performed using
  551. // the same instance. This is helpful for getting aggregates and
  552. // then getting actual results from the query.
  553. $this->selects = null;
  554. return $results;
  555. }
  556. /**
  557. * Get an aggregate value.
  558. *
  559. * @param string $aggregator
  560. * @param array $columns
  561. * @return mixed
  562. */
  563. public function aggregate($aggregator, $columns)
  564. {
  565. $this->aggregate = compact('aggregator', 'columns');
  566. $sql = $this->grammar->select($this);
  567. $result = $this->connection->only($sql, $this->bindings);
  568. // Reset the aggregate so more queries can be performed using
  569. // the same instance. This is helpful for getting aggregates
  570. // and then getting actual results from the query.
  571. $this->aggregate = null;
  572. return $result;
  573. }
  574. /**
  575. * Get the paginated query results as a Paginator instance.
  576. *
  577. * @param int $per_page
  578. * @param array $columns
  579. * @return Paginator
  580. */
  581. public function paginate($per_page = 20, $columns = array('*'))
  582. {
  583. // Because some database engines may throw errors if we leave orderings
  584. // on the query when retrieving the total number of records, we'll drop
  585. // all of the ordreings and put them back on the query after we have
  586. // retrieved the count from the table.
  587. list($orderings, $this->orderings) = array($this->orderings, null);
  588. $total = $this->count(reset($columns));
  589. $page = Paginator::page($total, $per_page);
  590. $this->orderings = $orderings;
  591. // Now we're ready to get the actual pagination results from the table
  592. // using the for_page and get methods. The "for_page" method provides
  593. // a convenient way to set the limit and offset so we get the correct
  594. // span of results from the table.
  595. $results = $this->for_page($page, $per_page)->get($columns);
  596. return Paginator::make($results, $total, $per_page);
  597. }
  598. /**
  599. * Insert an array of values into the database table.
  600. *
  601. * @param array $values
  602. * @return bool
  603. */
  604. public function insert($values)
  605. {
  606. // Force every insert to be treated like a batch insert to make creating
  607. // the binding array simpler since we can just spin through the inserted
  608. // rows as if there/ was more than one every time.
  609. if ( ! is_array(reset($values))) $values = array($values);
  610. $bindings = array();
  611. // We need to merge the the insert values into the array of the query
  612. // bindings so that they will be bound to the PDO statement when it
  613. // is executed by the database connection.
  614. foreach ($values as $value)
  615. {
  616. $bindings = array_merge($bindings, array_values($value));
  617. }
  618. $sql = $this->grammar->insert($this, $values);
  619. return $this->connection->statement($sql, $bindings);
  620. }
  621. /**
  622. * Insert an array of values into the database table and return the ID.
  623. *
  624. * @param array $values
  625. * @param string $sequence
  626. * @return int
  627. */
  628. public function insert_get_id($values, $sequence = null)
  629. {
  630. $sql = $this->grammar->insert($this, $values);
  631. $this->connection->statement($sql, array_values($values));
  632. // Some database systems (Postgres) require a sequence name to be
  633. // given when retrieving the auto-incrementing ID, so we'll pass
  634. // the given sequence into the method just in case.
  635. return (int) $this->connection->pdo->lastInsertId($sequence);
  636. }
  637. /**
  638. * Increment the value of a column by a given amount.
  639. *
  640. * @param string $column
  641. * @param int $amount
  642. * @return int
  643. */
  644. public function increment($column, $amount = 1)
  645. {
  646. return $this->adjust($column, $amount, ' + ');
  647. }
  648. /**
  649. * Decrement the value of a column by a given amount.
  650. *
  651. * @param string $column
  652. * @param int $amount
  653. * @return int
  654. */
  655. public function decrement($column, $amount = 1)
  656. {
  657. return $this->adjust($column, $amount, ' - ');
  658. }
  659. /**
  660. * Adjust the value of a column up or down by a given amount.
  661. *
  662. * @param string $column
  663. * @param int $amount
  664. * @param string $operator
  665. * @return int
  666. */
  667. protected function adjust($column, $amount, $operator)
  668. {
  669. // To make the adjustment to the column, we'll wrap the expression
  670. // in an Expression instance, which forces the adjustment to be
  671. // injected into the query as a string instead of bound.
  672. $value = Database::raw($this->grammar->wrap($column).$operator.$amount);
  673. return $this->update(array($column => $value));
  674. }
  675. /**
  676. * Update an array of values in the database table.
  677. *
  678. * @param array $values
  679. * @return int
  680. */
  681. public function update($values)
  682. {
  683. // For update statements, we need to merge the bindings such that
  684. // the update values occur before the where bindings in the array
  685. // since the set statements will precede any of the where clauses
  686. // in the SQL syntax that is generated.
  687. $bindings = array_merge(array_values($values), $this->bindings);
  688. $sql = $this->grammar->update($this, $values);
  689. return $this->connection->update($sql, $bindings);
  690. }
  691. /**
  692. * Execute the query as a DELETE statement.
  693. *
  694. * Optionally, an ID may be passed to the method do delete a specific row.
  695. *
  696. * @param int $id
  697. * @return int
  698. */
  699. public function delete($id = null)
  700. {
  701. // If an ID is given to the method, we'll set the where clause
  702. // to match on the value of the ID. This allows the developer
  703. // to quickly delete a row by its primary key value.
  704. if ( ! is_null($id))
  705. {
  706. $this->where('id', '=', $id);
  707. }
  708. $sql = $this->grammar->delete($this);
  709. return $this->connection->delete($sql, $this->bindings);
  710. }
  711. /**
  712. * Magic Method for handling dynamic functions.
  713. *
  714. * This method handles calls to aggregates as well as dynamic where clauses.
  715. */
  716. public function __call($method, $parameters)
  717. {
  718. if (strpos($method, 'where_') === 0)
  719. {
  720. return $this->dynamic_where($method, $parameters, $this);
  721. }
  722. if (in_array($method, array('count', 'min', 'max', 'avg', 'sum')))
  723. {
  724. if (count($parameters) == 0) $parameters[0] = '*';
  725. return $this->aggregate(strtoupper($method), (array) $parameters[0]);
  726. }
  727. throw new \Exception("Method [$method] is not defined on the Query class.");
  728. }
  729. }