2017-03-02 18:42:19 -05: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-03-02 18:42:19 -05:00
|
|
|
declare(strict_types=1);
|
2021-04-14 11:17:01 -04:00
|
|
|
|
2017-03-27 23:12:12 -05:00
|
|
|
namespace JKingWeb\Arsse\Db;
|
2017-08-29 10:50:31 -04:00
|
|
|
|
2017-07-17 07:47:57 -04:00
|
|
|
use JKingWeb\Arsse\Misc\Date;
|
2017-12-30 18:50:56 -05:00
|
|
|
use JKingWeb\Arsse\Misc\ValueInfo;
|
2017-03-02 18:42:19 -05:00
|
|
|
|
|
|
|
abstract class AbstractStatement implements Statement {
|
2019-01-10 19:01:32 -05:00
|
|
|
use SQLState;
|
|
|
|
|
2020-03-01 18:32:01 -05:00
|
|
|
public const TYPE_NORM_MAP = [
|
2020-03-01 15:16:50 -05:00
|
|
|
self::T_INTEGER => ValueInfo::M_NULL | ValueInfo::T_INT,
|
|
|
|
self::T_STRING => ValueInfo::M_NULL | ValueInfo::T_STRING,
|
|
|
|
self::T_BOOLEAN => ValueInfo::M_NULL | ValueInfo::T_BOOL,
|
|
|
|
self::T_DATETIME => ValueInfo::M_NULL | ValueInfo::T_DATE,
|
|
|
|
self::T_FLOAT => ValueInfo::M_NULL | ValueInfo::T_FLOAT,
|
|
|
|
self::T_BINARY => ValueInfo::M_NULL | ValueInfo::T_STRING,
|
2019-03-01 12:17:33 -05:00
|
|
|
self::T_NOT_NULL + self::T_INTEGER => ValueInfo::T_INT,
|
|
|
|
self::T_NOT_NULL + self::T_STRING => ValueInfo::T_STRING,
|
|
|
|
self::T_NOT_NULL + self::T_BOOLEAN => ValueInfo::T_BOOL,
|
|
|
|
self::T_NOT_NULL + self::T_DATETIME => ValueInfo::T_DATE,
|
|
|
|
self::T_NOT_NULL + self::T_FLOAT => ValueInfo::T_FLOAT,
|
|
|
|
self::T_NOT_NULL + self::T_BINARY => ValueInfo::T_STRING,
|
|
|
|
];
|
|
|
|
|
2017-05-04 19:12:33 -04:00
|
|
|
protected $types = [];
|
2017-03-02 18:42:19 -05:00
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
abstract public function runArray(array $values = []): Result;
|
2019-03-01 12:17:33 -05:00
|
|
|
abstract protected function bindValue($value, int $type, int $position): bool;
|
2018-12-21 12:35:10 -05:00
|
|
|
abstract protected function prepare(string $query): bool;
|
2019-01-10 19:01:32 -05:00
|
|
|
abstract protected static function buildEngineException($code, string $msg): array;
|
2017-03-02 18:42:19 -05:00
|
|
|
|
|
|
|
public function run(...$values): Result {
|
|
|
|
return $this->runArray($values);
|
|
|
|
}
|
|
|
|
|
2017-12-30 17:04:21 -05:00
|
|
|
public function retype(...$bindings): bool {
|
|
|
|
return $this->retypeArray($bindings);
|
2017-03-02 18:42:19 -05:00
|
|
|
}
|
|
|
|
|
2018-12-21 12:35:10 -05:00
|
|
|
public static function mungeQuery(string $query, array $types, ...$extraData): string {
|
|
|
|
return $query;
|
|
|
|
}
|
|
|
|
|
2019-10-18 13:10:03 -04:00
|
|
|
public function retypeArray(array $bindings): bool {
|
|
|
|
$this->types = [];
|
|
|
|
foreach (ValueInfo::flatten($bindings) as $binding) { // recursively flatten any arrays, which may be provided for SET or IN() clauses
|
|
|
|
$bindId = self::TYPES[trim(strtolower($binding))] ?? 0;
|
|
|
|
assert($bindId, new Exception("paramTypeInvalid", $binding));
|
|
|
|
$this->types[] = $bindId;
|
2018-12-21 12:35:10 -05:00
|
|
|
}
|
2019-10-18 13:10:03 -04:00
|
|
|
$this->prepare(static::mungeQuery($this->query, $this->types));
|
2017-03-02 18:42:19 -05:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-03-01 12:17:33 -05:00
|
|
|
protected function cast($v, int $t) {
|
2017-08-29 10:50:31 -04:00
|
|
|
switch ($t) {
|
2019-03-01 12:17:33 -05:00
|
|
|
case self::T_DATETIME:
|
|
|
|
return Date::transform($v, "sql");
|
|
|
|
case self::T_DATETIME + self::T_NOT_NULL:
|
2017-12-30 18:50:56 -05:00
|
|
|
$v = Date::transform($v, "sql");
|
2019-03-01 12:17:33 -05:00
|
|
|
return $v ? $v : "0001-01-01 00:00:00";
|
2017-04-06 21:41:21 -04:00
|
|
|
default:
|
2019-03-01 12:17:33 -05:00
|
|
|
$v = ValueInfo::normalize($v, self::TYPE_NORM_MAP[$t], null, "sql");
|
|
|
|
return is_bool($v) ? (int) $v : $v;
|
2017-04-06 21:41:21 -04:00
|
|
|
}
|
|
|
|
}
|
2017-12-30 18:50:56 -05:00
|
|
|
|
2019-10-18 13:10:03 -04:00
|
|
|
protected function bindValues(array $values): bool {
|
2019-10-25 15:16:35 -04:00
|
|
|
// recursively flatten any arrays, which may be provided for SET or IN() clauses
|
2019-10-18 13:10:03 -04:00
|
|
|
$values = ValueInfo::flatten($values);
|
|
|
|
foreach ($values as $a => $value) {
|
|
|
|
if (array_key_exists($a, $this->types)) {
|
2019-03-01 12:17:33 -05:00
|
|
|
$value = $this->cast($value, $this->types[$a]);
|
|
|
|
$this->bindValue($value, $this->types[$a] % self::T_NOT_NULL, ++$a);
|
2017-12-30 18:50:56 -05:00
|
|
|
} else {
|
2020-03-01 15:16:50 -05:00
|
|
|
throw new Exception("paramTypeMissing", $a + 1);
|
2017-12-30 18:50:56 -05:00
|
|
|
}
|
|
|
|
}
|
2019-10-18 13:10:03 -04:00
|
|
|
// once all values are bound, check that all parameters have been supplied values and bind null for any missing ones
|
2018-11-20 15:45:20 -05:00
|
|
|
// SQLite will happily substitute null for a missing value, but other engines (viz. PostgreSQL) produce an error
|
2019-10-18 13:10:03 -04:00
|
|
|
for ($a = sizeof($values); $a < sizeof($this->types); $a++) {
|
|
|
|
$this->bindValue(null, $this->types[$a] % self::T_NOT_NULL, $a + 1);
|
2018-11-20 15:45:20 -05:00
|
|
|
}
|
2019-10-18 13:10:03 -04:00
|
|
|
return true;
|
2017-12-30 18:50:56 -05:00
|
|
|
}
|
2017-08-29 10:50:31 -04:00
|
|
|
}
|