mirror of
https://code.mensbeam.com/MensBeam/Arsse.git
synced 2025-01-05 15:32:40 +00:00
8fc31cfc40
This involved changes to the driver interface as well as the database schemata. The most significantly altered queries were for article selection and marking, which relied upon unusual features of SQLite. Overall query efficiency should not be adversely affected (it may have even imprved) in the common case, while very rare cases (not presently triggered by any REST handlers) require more queries. One notable benefit of these changes is that functions which query articles can now have complete control over which columns are returned. This has not, however, been implemented yet: symbolic column groups are still used for now. Note that PostgreSQL still fails many tests, but the test suite runs to completion. Note also that one line of the Database class is not covered; later changes will eventually make it easier to cover the line in question.
187 lines
6.1 KiB
PHP
187 lines
6.1 KiB
PHP
<?php
|
|
/** @license MIT
|
|
* Copyright 2017 J. King, Dustin Wilson et al.
|
|
* See LICENSE and AUTHORS files for details */
|
|
|
|
declare(strict_types=1);
|
|
namespace JKingWeb\Arsse\Db\SQLite3;
|
|
|
|
use JKingWeb\Arsse\Arsse;
|
|
use JKingWeb\Arsse\Db\Exception;
|
|
use JKingWeb\Arsse\Db\ExceptionInput;
|
|
use JKingWeb\Arsse\Db\ExceptionTimeout;
|
|
|
|
class Driver extends \JKingWeb\Arsse\Db\AbstractDriver {
|
|
use ExceptionBuilder;
|
|
|
|
const SQLITE_BUSY = 5;
|
|
const SQLITE_CONSTRAINT = 19;
|
|
const SQLITE_MISMATCH = 20;
|
|
|
|
protected $db;
|
|
|
|
public function __construct(string $dbFile = null) {
|
|
// check to make sure required extension is loaded
|
|
if (!self::requirementsMet()) {
|
|
throw new Exception("extMissing", self::driverName()); // @codeCoverageIgnore
|
|
}
|
|
// if no database file is specified in the configuration, use a suitable default
|
|
$dbFile = $dbFile ?? Arsse::$conf->dbSQLite3File ?? \JKingWeb\Arsse\BASE."arsse.db";
|
|
try {
|
|
$this->makeConnection($dbFile, Arsse::$conf->dbSQLite3Key);
|
|
} catch (\Throwable $e) {
|
|
// if opening the database doesn't work, check various pre-conditions to find out what the problem might be
|
|
$files = [
|
|
$dbFile, // main database file
|
|
$dbFile."-wal", // write-ahead log journal
|
|
$dbFile."-shm", // shared memory index
|
|
];
|
|
foreach ($files as $file) {
|
|
if (!file_exists($file) && !is_writable(dirname($file))) {
|
|
throw new Exception("fileUncreatable", $file);
|
|
} elseif (!is_readable($file) && !is_writable($file)) {
|
|
throw new Exception("fileUnusable", $file);
|
|
} elseif (!is_readable($file)) {
|
|
throw new Exception("fileUnreadable", $file);
|
|
} elseif (!is_writable($file)) {
|
|
throw new Exception("fileUnwritable", $file);
|
|
}
|
|
}
|
|
// otherwise the database is probably corrupt
|
|
throw new Exception("fileCorrupt", $dbFile);
|
|
}
|
|
// set the timeout
|
|
$timeout = (int) ceil((Arsse::$conf->dbSQLite3Timeout ?? 0) * 1000);
|
|
$this->setTimeout($timeout);
|
|
// set other initial options
|
|
$this->exec("PRAGMA foreign_keys = yes");
|
|
}
|
|
|
|
public static function requirementsMet(): bool {
|
|
return class_exists("SQLite3");
|
|
}
|
|
|
|
protected function makeConnection(string $file, string $key) {
|
|
$this->db = new \SQLite3($file, \SQLITE3_OPEN_READWRITE | \SQLITE3_OPEN_CREATE, $key);
|
|
// enable exceptions
|
|
$this->db->enableExceptions(true);
|
|
}
|
|
|
|
protected function setTimeout(int $msec) {
|
|
$this->exec("PRAGMA busy_timeout = $msec");
|
|
}
|
|
|
|
public function __destruct() {
|
|
try {
|
|
$this->db->close();
|
|
} catch (\Exception $e) { // @codeCoverageIgnore
|
|
}
|
|
unset($this->db);
|
|
}
|
|
|
|
/** @codeCoverageIgnore */
|
|
public static function create(): \JKingWeb\Arsse\Db\Driver {
|
|
if (self::requirementsMet()) {
|
|
return new self;
|
|
} elseif (PDODriver::requirementsMet()) {
|
|
return new PDODriver;
|
|
} else {
|
|
throw new Exception("extMissing", self::driverName());
|
|
}
|
|
}
|
|
|
|
|
|
public static function driverName(): string {
|
|
return Arsse::$lang->msg("Driver.Db.SQLite3.Name");
|
|
}
|
|
|
|
public static function schemaID(): string {
|
|
return "SQLite3";
|
|
}
|
|
|
|
public function schemaVersion(): int {
|
|
return (int) $this->query("PRAGMA user_version")->getValue();
|
|
}
|
|
|
|
public function sqlToken(string $token): string {
|
|
switch(strtolower($token)) {
|
|
case "greatest":
|
|
return "max";
|
|
default:
|
|
return $token;
|
|
}
|
|
}
|
|
|
|
public function schemaUpdate(int $to, string $basePath = null): bool {
|
|
// turn off foreign keys
|
|
$this->exec("PRAGMA foreign_keys = no");
|
|
// run the generic updater
|
|
try {
|
|
parent::schemaUpdate($to, $basePath);
|
|
} catch (\Throwable $e) {
|
|
// turn foreign keys back on
|
|
$this->exec("PRAGMA foreign_keys = yes");
|
|
// pass the exception up
|
|
throw $e;
|
|
}
|
|
// turn foreign keys back on
|
|
$this->exec("PRAGMA foreign_keys = yes");
|
|
return true;
|
|
}
|
|
|
|
public function charsetAcceptable(): bool {
|
|
// SQLite 3 databases are UTF-8 internally, thus always acceptable
|
|
return true;
|
|
}
|
|
|
|
protected function getError(): string {
|
|
return $this->db->lastErrorMsg();
|
|
}
|
|
|
|
public function exec(string $query): bool {
|
|
try {
|
|
return (bool) $this->db->exec($query);
|
|
} catch (\Exception $e) {
|
|
list($excClass, $excMsg, $excData) = $this->exceptionBuild();
|
|
throw new $excClass($excMsg, $excData);
|
|
}
|
|
}
|
|
|
|
public function query(string $query): \JKingWeb\Arsse\Db\Result {
|
|
try {
|
|
$r = $this->db->query($query);
|
|
} catch (\Exception $e) {
|
|
list($excClass, $excMsg, $excData) = $this->exceptionBuild();
|
|
throw new $excClass($excMsg, $excData);
|
|
}
|
|
$changes = $this->db->changes();
|
|
$lastId = $this->db->lastInsertRowID();
|
|
return new Result($r, [$changes, $lastId]);
|
|
}
|
|
|
|
public function prepareArray(string $query, array $paramTypes): \JKingWeb\Arsse\Db\Statement {
|
|
try {
|
|
$s = $this->db->prepare($query);
|
|
} catch (\Exception $e) {
|
|
list($excClass, $excMsg, $excData) = $this->exceptionBuild();
|
|
throw new $excClass($excMsg, $excData);
|
|
}
|
|
return new Statement($this->db, $s, $paramTypes);
|
|
}
|
|
|
|
protected function lock(): bool {
|
|
$timeout = (int) $this->query("PRAGMA busy_timeout")->getValue();
|
|
$this->setTimeout(0);
|
|
try {
|
|
$this->exec("BEGIN EXCLUSIVE TRANSACTION");
|
|
} finally {
|
|
$this->setTimeout($timeout);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
protected function unlock(bool $rollback = false): bool {
|
|
$this->exec((!$rollback) ? "COMMIT" : "ROLLBACK");
|
|
return true;
|
|
}
|
|
}
|