"bigint", "float" => "decimal", "datetime" => "timestamp", "binary" => "bytea", "string" => "text", "boolean" => "smallint", // FIXME: using boolean leads to incompatibilities with versions of SQLite bundled prior to PHP 7.3 ]; protected $db; protected $st; protected $qOriginal; protected $qMunged; protected $bindings; public function __construct(\PDO $db, string $query, array $bindings = []) { $this->db = $db; // both db and st are the same object due to the logic of the PDOError handler $this->qOriginal = $query; $this->retypeArray($bindings); } public function __destruct() { unset($this->db, $this->st); } public function retypeArray(array $bindings, bool $append = false): bool { if ($append) { return parent::retypeArray($bindings, $append); } else { $this->bindings = $bindings; parent::retypeArray($bindings, $append); $this->qMunged = self::mungeQuery($this->qOriginal, $this->types, false); try { // statement creation with PostgreSQL should never fail (it is not evaluated at creation time) $s = $this->db->prepare($this->qMunged); } catch (\PDOException $e) { // @codeCoverageIgnore list($excClass, $excMsg, $excData) = $this->exceptionBuild(true); // @codeCoverageIgnore throw new $excClass($excMsg, $excData); // @codeCoverageIgnore } $this->st = new \JKingWeb\Arsse\Db\PDOStatement($this->db, $s, $this->bindings); } return true; } public static function mungeQuery(string $q, array $types, bool $mungeParamMarkers = true): string { $q = explode("?", $q); $out = ""; for ($b = 1; $b < sizeof($q); $b++) { $a = $b - 1; $mark = $mungeParamMarkers ? "\$$b" : "?"; $type = isset($types[$a]) ? "::".self::BINDINGS[$types[$a]] : ""; $out .= $q[$a].$mark.$type; } $out .= array_pop($q); return $out; } public function runArray(array $values = []): \JKingWeb\Arsse\Db\Result { return $this->st->runArray($values); } /** @codeCoverageIgnore */ protected function bindValue($value, string $type, int $position): bool { // stub required by abstract parent, but never used return $value; } }