2017-03-02 18:42:19 -05:00
|
|
|
<?php
|
|
|
|
declare(strict_types=1);
|
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-03-02 18:42:19 -05:00
|
|
|
|
|
|
|
abstract class AbstractStatement implements Statement {
|
2017-05-04 19:12:33 -04:00
|
|
|
protected $types = [];
|
|
|
|
protected $isNullable = [];
|
2017-03-02 18:42:19 -05:00
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
abstract public function runArray(array $values = []): Result;
|
2017-03-02 18:42:19 -05:00
|
|
|
|
|
|
|
public function run(...$values): Result {
|
|
|
|
return $this->runArray($values);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function rebind(...$bindings): bool {
|
|
|
|
return $this->rebindArray($bindings);
|
|
|
|
}
|
|
|
|
|
2017-04-06 11:02:47 -04:00
|
|
|
public function rebindArray(array $bindings, bool $append = false): bool {
|
2017-08-29 10:50:31 -04:00
|
|
|
if (!$append) {
|
2017-07-20 22:40:09 -04:00
|
|
|
$this->types = [];
|
|
|
|
}
|
2017-08-29 10:50:31 -04:00
|
|
|
foreach ($bindings as $binding) {
|
|
|
|
if (is_array($binding)) {
|
2017-04-06 11:02:47 -04:00
|
|
|
// recursively flatten any arrays, which may be provided for SET or IN() clauses
|
|
|
|
$this->rebindArray($binding, true);
|
|
|
|
} else {
|
|
|
|
$binding = trim(strtolower($binding));
|
2017-08-29 10:50:31 -04:00
|
|
|
if (strpos($binding, "strict ")===0) {
|
2017-05-04 19:12:33 -04:00
|
|
|
// "strict" types' values may never be null; null values will later be cast to the type specified
|
|
|
|
$this->isNullable[] = false;
|
|
|
|
$binding = substr($binding, 7);
|
|
|
|
} else {
|
|
|
|
$this->isNullable[] = true;
|
|
|
|
}
|
2017-08-29 10:50:31 -04:00
|
|
|
if (!array_key_exists($binding, self::TYPES)) {
|
2017-07-22 15:29:12 -04:00
|
|
|
throw new Exception("paramTypeInvalid", $binding); // @codeCoverageIgnore
|
2017-07-20 22:40:09 -04:00
|
|
|
}
|
2017-04-06 11:02:47 -04:00
|
|
|
$this->types[] = self::TYPES[$binding];
|
|
|
|
}
|
2017-03-02 18:42:19 -05:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-05-04 19:12:33 -04:00
|
|
|
protected function cast($v, string $t, bool $nullable) {
|
2017-08-29 10:50:31 -04:00
|
|
|
switch ($t) {
|
2017-04-06 21:41:21 -04:00
|
|
|
case "date":
|
2017-08-29 10:50:31 -04:00
|
|
|
if (is_null($v) && !$nullable) {
|
2017-07-20 22:40:09 -04:00
|
|
|
$v = 0;
|
|
|
|
}
|
2017-07-17 07:47:57 -04:00
|
|
|
return Date::transform($v, "date");
|
2017-04-06 21:41:21 -04:00
|
|
|
case "time":
|
2017-08-29 10:50:31 -04:00
|
|
|
if (is_null($v) && !$nullable) {
|
2017-07-20 22:40:09 -04:00
|
|
|
$v = 0;
|
|
|
|
}
|
2017-07-17 07:47:57 -04:00
|
|
|
return Date::transform($v, "time");
|
2017-04-06 21:41:21 -04:00
|
|
|
case "datetime":
|
2017-08-29 10:50:31 -04:00
|
|
|
if (is_null($v) && !$nullable) {
|
2017-07-20 22:40:09 -04:00
|
|
|
$v = 0;
|
|
|
|
}
|
2017-07-17 07:47:57 -04:00
|
|
|
return Date::transform($v, "sql");
|
2017-04-06 21:41:21 -04:00
|
|
|
case "null":
|
|
|
|
case "integer":
|
|
|
|
case "float":
|
|
|
|
case "binary":
|
|
|
|
case "string":
|
|
|
|
case "boolean":
|
2017-08-29 10:50:31 -04:00
|
|
|
if ($t=="binary") {
|
2017-07-20 22:40:09 -04:00
|
|
|
$t = "string";
|
|
|
|
}
|
2017-08-29 10:50:31 -04:00
|
|
|
if ($v instanceof \DateTimeInterface) {
|
|
|
|
if ($t=="string") {
|
2017-07-17 07:47:57 -04:00
|
|
|
return Date::transform($v, "sql");
|
2017-04-06 21:41:21 -04:00
|
|
|
} else {
|
2017-07-07 21:06:38 -04:00
|
|
|
$v = $v->getTimestamp();
|
2017-08-29 10:50:31 -04:00
|
|
|
settype($v, $t);
|
2017-04-06 21:41:21 -04:00
|
|
|
}
|
2017-07-07 21:06:38 -04:00
|
|
|
} else {
|
|
|
|
settype($v, $t);
|
2017-04-06 21:41:21 -04:00
|
|
|
}
|
2017-07-07 21:06:38 -04:00
|
|
|
return $v;
|
2017-04-06 21:41:21 -04:00
|
|
|
default:
|
2017-07-22 15:29:12 -04:00
|
|
|
throw new Exception("paramTypeUnknown", $type); // @codeCoverageIgnore
|
2017-04-06 21:41:21 -04:00
|
|
|
}
|
|
|
|
}
|
2017-08-29 10:50:31 -04:00
|
|
|
}
|