2017-06-04 18:00:18 -04:00
|
|
|
<?php
|
2017-11-16 20:23:18 -05:00
|
|
|
/** @license MIT
|
|
|
|
* Copyright 2017 J. King, Dustin Wilson et al.
|
|
|
|
* See LICENSE and AUTHORS files for details */
|
|
|
|
|
2017-06-04 18:00:18 -04:00
|
|
|
declare(strict_types=1);
|
2017-06-18 10:23:37 -04:00
|
|
|
namespace JKingWeb\Arsse\Misc;
|
2017-06-04 18:00:18 -04:00
|
|
|
|
|
|
|
class Query {
|
2017-07-07 11:49:54 -04:00
|
|
|
protected $qBody = ""; // main query body
|
|
|
|
protected $tBody = []; // main query parameter types
|
|
|
|
protected $vBody = []; // main query parameter values
|
2017-06-04 18:00:18 -04:00
|
|
|
protected $qCTE = []; // Common table expression query components
|
|
|
|
protected $tCTE = []; // Common table expression type bindings
|
|
|
|
protected $vCTE = []; // Common table expression binding values
|
|
|
|
protected $jCTE = []; // Common Table Expression joins
|
2017-10-06 20:26:22 -04:00
|
|
|
protected $qJoin = []; // JOIN clause components
|
|
|
|
protected $tJoin = []; // JOIN clause type bindings
|
|
|
|
protected $vJoin = []; // JOIN clause binding values
|
2017-06-04 18:00:18 -04:00
|
|
|
protected $qWhere = []; // WHERE clause components
|
|
|
|
protected $tWhere = []; // WHERE clause type bindings
|
|
|
|
protected $vWhere = []; // WHERE clause binding values
|
2019-02-25 22:41:12 -05:00
|
|
|
protected $qWhereNot = []; // WHERE NOT clause components
|
|
|
|
protected $tWhereNot = []; // WHERE NOT clause type bindings
|
|
|
|
protected $vWhereNot = []; // WHERE NOT clause binding values
|
2018-12-04 20:41:21 -05:00
|
|
|
protected $group = []; // GROUP BY clause components
|
2017-06-04 18:00:18 -04:00
|
|
|
protected $order = []; // ORDER BY clause components
|
|
|
|
protected $limit = 0;
|
|
|
|
protected $offset = 0;
|
|
|
|
|
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function __construct(string $body = "", $types = null, $values = null) {
|
2017-07-07 11:49:54 -04:00
|
|
|
$this->setBody($body, $types, $values);
|
2017-06-04 18:00:18 -04:00
|
|
|
}
|
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function setBody(string $body = "", $types = null, $values = null): bool {
|
2017-07-07 11:49:54 -04:00
|
|
|
$this->qBody = $body;
|
2017-08-29 10:50:31 -04:00
|
|
|
if (!is_null($types)) {
|
2017-07-07 11:49:54 -04:00
|
|
|
$this->tBody[] = $types;
|
|
|
|
$this->vBody[] = $values;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function setCTE(string $tableSpec, string $body, $types = null, $values = null, string $join = ''): bool {
|
2017-07-07 11:49:54 -04:00
|
|
|
$this->qCTE[] = "$tableSpec as ($body)";
|
2017-08-29 10:50:31 -04:00
|
|
|
if (!is_null($types)) {
|
2017-06-04 18:00:18 -04:00
|
|
|
$this->tCTE[] = $types;
|
|
|
|
$this->vCTE[] = $values;
|
|
|
|
}
|
2017-08-29 10:50:31 -04:00
|
|
|
if (strlen($join)) { // the CTE might only participate in subqueries rather than a join on the main query
|
2017-07-20 22:40:09 -04:00
|
|
|
$this->jCTE[] = $join;
|
|
|
|
}
|
2017-06-04 18:00:18 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-10-06 20:26:22 -04:00
|
|
|
public function setJoin(string $join, $types = null, $values = null): bool {
|
|
|
|
$this->qJoin[] = $join;
|
|
|
|
if (!is_null($types)) {
|
|
|
|
$this->tJoin[] = $types;
|
|
|
|
$this->vJoin[] = $values;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function setWhere(string $where, $types = null, $values = null): bool {
|
2017-06-04 18:00:18 -04:00
|
|
|
$this->qWhere[] = $where;
|
2017-08-29 10:50:31 -04:00
|
|
|
if (!is_null($types)) {
|
2017-06-04 18:00:18 -04:00
|
|
|
$this->tWhere[] = $types;
|
|
|
|
$this->vWhere[] = $values;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-02-25 22:41:12 -05:00
|
|
|
public function setWhereNot(string $where, $types = null, $values = null): bool {
|
|
|
|
$this->qWhereNot[] = $where;
|
|
|
|
if (!is_null($types)) {
|
|
|
|
$this->tWhereNot[] = $types;
|
|
|
|
$this->vWhereNot[] = $values;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-12-04 20:41:21 -05:00
|
|
|
public function setGroup(string ...$column): bool {
|
|
|
|
foreach ($column as $col) {
|
|
|
|
$this->group[] = $col;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function setOrder(string $order, bool $prepend = false): bool {
|
|
|
|
if ($prepend) {
|
2017-06-04 18:00:18 -04:00
|
|
|
array_unshift($this->order, $order);
|
|
|
|
} else {
|
|
|
|
$this->order[] = $order;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function setLimit(int $limit, int $offset = 0): bool {
|
2017-07-07 11:49:54 -04:00
|
|
|
$this->limit = $limit;
|
|
|
|
$this->offset = $offset;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function pushCTE(string $tableSpec, string $join = ''): bool {
|
2017-07-07 11:49:54 -04:00
|
|
|
// this function takes the query body and converts it to a common table expression, putting it at the bottom of the existing CTE stack
|
|
|
|
// all WHERE, ORDER BY, and LIMIT parts belong to the new CTE and are removed from the main query
|
2019-02-25 22:41:12 -05:00
|
|
|
$this->setCTE($tableSpec, $this->buildQueryBody(), [$this->tBody, $this->tWhere, $this->tWhereNot], [$this->vBody, $this->vWhere, $this->vWhereNot]);
|
2017-07-07 11:49:54 -04:00
|
|
|
$this->jCTE = [];
|
|
|
|
$this->tBody = [];
|
|
|
|
$this->vBody = [];
|
|
|
|
$this->qWhere = [];
|
|
|
|
$this->tWhere = [];
|
|
|
|
$this->vWhere = [];
|
2019-02-26 11:11:42 -05:00
|
|
|
$this->qWhereNot = [];
|
|
|
|
$this->tWhereNot = [];
|
|
|
|
$this->vWhereNot = [];
|
2017-10-06 20:26:22 -04:00
|
|
|
$this->qJoin = [];
|
|
|
|
$this->tJoin = [];
|
|
|
|
$this->vJoin = [];
|
2017-07-07 11:49:54 -04:00
|
|
|
$this->order = [];
|
2018-12-04 20:41:21 -05:00
|
|
|
$this->group = [];
|
2017-08-29 10:50:31 -04:00
|
|
|
$this->setLimit(0, 0);
|
|
|
|
if (strlen($join)) {
|
2017-07-20 22:40:09 -04:00
|
|
|
$this->jCTE[] = $join;
|
|
|
|
}
|
2017-07-07 11:49:54 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function __toString(): string {
|
2017-06-04 18:00:18 -04:00
|
|
|
$out = "";
|
2017-08-29 10:50:31 -04:00
|
|
|
if (sizeof($this->qCTE)) {
|
2017-06-04 18:00:18 -04:00
|
|
|
// start with common table expressions
|
|
|
|
$out .= "WITH RECURSIVE ".implode(", ", $this->qCTE)." ";
|
|
|
|
}
|
|
|
|
// add the body
|
2017-06-18 10:23:37 -04:00
|
|
|
$out .= $this->buildQueryBody();
|
|
|
|
return $out;
|
|
|
|
}
|
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function getQuery(): string {
|
2017-07-07 11:49:54 -04:00
|
|
|
return $this->__toString();
|
|
|
|
}
|
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function getTypes(): array {
|
2019-02-25 22:41:12 -05:00
|
|
|
return [$this->tCTE, $this->tBody, $this->tJoin, $this->tWhere, $this->tWhereNot];
|
2017-07-07 11:49:54 -04:00
|
|
|
}
|
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function getValues(): array {
|
2019-02-25 22:41:12 -05:00
|
|
|
return [$this->vCTE, $this->vBody, $this->vJoin, $this->vWhere, $this->vWhereNot];
|
2017-10-06 20:26:22 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
public function getJoinTypes(): array {
|
|
|
|
return $this->tJoin;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getJoinValues(): array {
|
|
|
|
return $this->vJoin;
|
2017-06-18 10:23:37 -04:00
|
|
|
}
|
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function getWhereTypes(): array {
|
2017-06-18 10:23:37 -04:00
|
|
|
return $this->tWhere;
|
|
|
|
}
|
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function getWhereValues(): array {
|
2017-06-18 10:23:37 -04:00
|
|
|
return $this->vWhere;
|
|
|
|
}
|
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function getCTETypes(): array {
|
2017-06-18 10:23:37 -04:00
|
|
|
return $this->tCTE;
|
|
|
|
}
|
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function getCTEValues(): array {
|
2017-06-18 10:23:37 -04:00
|
|
|
return $this->vCTE;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function buildQueryBody(): string {
|
|
|
|
$out = "";
|
|
|
|
// add the body
|
2017-07-07 11:49:54 -04:00
|
|
|
$out .= $this->qBody;
|
2017-08-29 10:50:31 -04:00
|
|
|
if (sizeof($this->qCTE)) {
|
2017-06-04 18:00:18 -04:00
|
|
|
// add any joins against CTEs
|
|
|
|
$out .= " ".implode(" ", $this->jCTE);
|
|
|
|
}
|
2017-10-06 20:26:22 -04:00
|
|
|
// add any JOINs
|
|
|
|
if (sizeof($this->qJoin)) {
|
|
|
|
$out .= " ".implode(" ", $this->qJoin);
|
|
|
|
}
|
2017-06-04 18:00:18 -04:00
|
|
|
// add any WHERE terms
|
2019-02-25 22:41:12 -05:00
|
|
|
if (sizeof($this->qWhere) || sizeof($this->qWhereNot)) {
|
|
|
|
$where = implode(" AND ", $this->qWhere);
|
|
|
|
$whereNot = implode(" OR ", $this->qWhereNot);
|
|
|
|
$whereNot = strlen($whereNot) ? "NOT ($whereNot)" : "";
|
|
|
|
$where = implode(" AND ", array_filter([$where, $whereNot]));
|
|
|
|
$out .= " WHERE $where";
|
2017-06-04 18:00:18 -04:00
|
|
|
}
|
2018-12-04 20:41:21 -05:00
|
|
|
// add any GROUP BY terms
|
|
|
|
if (sizeof($this->group)) {
|
|
|
|
$out .= " GROUP BY ".implode(", ", $this->group);
|
|
|
|
}
|
2017-06-04 18:00:18 -04:00
|
|
|
// add any ORDER BY terms
|
2017-08-29 10:50:31 -04:00
|
|
|
if (sizeof($this->order)) {
|
2017-06-04 18:00:18 -04:00
|
|
|
$out .= " ORDER BY ".implode(", ", $this->order);
|
|
|
|
}
|
|
|
|
// add LIMIT and OFFSET if the former is specified
|
2017-08-29 10:50:31 -04:00
|
|
|
if ($this->limit > 0) {
|
2017-06-04 18:00:18 -04:00
|
|
|
$out .= " LIMIT ".$this->limit;
|
2017-08-29 10:50:31 -04:00
|
|
|
if ($this->offset > 0) {
|
2017-06-04 18:00:18 -04:00
|
|
|
$out .= " OFFSET ".$this->offset;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $out;
|
|
|
|
}
|
2017-08-29 10:50:31 -04:00
|
|
|
}
|