2017-03-08 22:16:35 -05:00
|
|
|
<?php
|
|
|
|
declare(strict_types=1);
|
2017-03-27 23:12:12 -05:00
|
|
|
namespace JKingWeb\Arsse;
|
2017-03-08 22:16:35 -05:00
|
|
|
|
|
|
|
|
|
|
|
class TestDbDriverSQLite3 extends \PHPUnit\Framework\TestCase {
|
|
|
|
use Test\Tools;
|
|
|
|
|
2017-03-09 21:39:42 -05:00
|
|
|
protected $data;
|
|
|
|
protected $drv;
|
2017-03-08 22:16:35 -05:00
|
|
|
|
|
|
|
function setUp() {
|
2017-03-28 18:50:00 -04:00
|
|
|
$this->clearData();
|
2017-03-08 22:16:35 -05:00
|
|
|
$conf = new Conf();
|
|
|
|
$conf->dbDriver = Db\SQLite3\Driver::class;
|
|
|
|
$conf->dbSQLite3File = tempnam(sys_get_temp_dir(), 'ook');
|
2017-03-28 18:50:00 -04:00
|
|
|
Data::$conf = $conf;
|
|
|
|
$this->drv = new Db\SQLite3\Driver(true);
|
2017-03-08 22:16:35 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
function tearDown() {
|
|
|
|
unset($this->drv);
|
2017-03-28 18:50:00 -04:00
|
|
|
unlink(Data::$conf->dbSQLite3File);
|
|
|
|
$this->clearData();
|
2017-03-08 22:16:35 -05:00
|
|
|
}
|
|
|
|
|
2017-03-09 17:25:50 -05:00
|
|
|
function testFetchDriverName() {
|
2017-03-28 18:50:00 -04:00
|
|
|
$class = Data::$conf->dbDriver;
|
2017-03-09 17:25:50 -05:00
|
|
|
$this->assertTrue(strlen($class::driverName()) > 0);
|
|
|
|
}
|
|
|
|
|
2017-03-08 22:16:35 -05:00
|
|
|
function testExecAValidStatement() {
|
|
|
|
$this->assertTrue($this->drv->exec("CREATE TABLE test(id integer primary key)"));
|
|
|
|
}
|
|
|
|
|
|
|
|
function testExecAnInvalidStatement() {
|
|
|
|
$this->assertException("engineErrorGeneral", "Db");
|
|
|
|
$this->drv->exec("And the meek shall inherit the earth...");
|
|
|
|
}
|
|
|
|
|
|
|
|
function testExecMultipleStatements() {
|
|
|
|
$this->assertTrue($this->drv->exec("CREATE TABLE test(id integer primary key); INSERT INTO test(id) values(2112)"));
|
2017-03-28 18:50:00 -04:00
|
|
|
$ch = new \SQLite3(Data::$conf->dbSQLite3File);
|
2017-03-08 22:16:35 -05:00
|
|
|
$this->assertEquals(2112, $ch->querySingle("SELECT id from test"));
|
|
|
|
}
|
|
|
|
|
|
|
|
function testExecTimeout() {
|
2017-03-28 18:50:00 -04:00
|
|
|
$ch = new \SQLite3(Data::$conf->dbSQLite3File);
|
2017-03-08 22:16:35 -05:00
|
|
|
$ch->exec("BEGIN EXCLUSIVE TRANSACTION");
|
|
|
|
$this->assertException("general", "Db", "ExceptionTimeout");
|
|
|
|
$this->drv->exec("CREATE TABLE test(id integer primary key)");
|
|
|
|
}
|
|
|
|
|
|
|
|
function testExecConstraintViolation() {
|
|
|
|
$this->drv->exec("CREATE TABLE test(id integer not null)");
|
|
|
|
$this->assertException("constraintViolation", "Db", "ExceptionInput");
|
|
|
|
$this->drv->exec("INSERT INTO test(id) values(null)");
|
|
|
|
}
|
|
|
|
|
2017-03-09 09:44:50 -05:00
|
|
|
function testExecTypeViolation() {
|
|
|
|
$this->drv->exec("CREATE TABLE test(id integer primary key)");
|
|
|
|
$this->assertException("typeViolation", "Db", "ExceptionInput");
|
|
|
|
$this->drv->exec("INSERT INTO test(id) values('ook')");
|
|
|
|
}
|
|
|
|
|
2017-03-09 14:48:42 -05:00
|
|
|
function testMakeAValidQuery() {
|
2017-03-08 22:16:35 -05:00
|
|
|
$this->assertInstanceOf(Db\SQLite3\Result::class, $this->drv->query("SELECT 1"));
|
|
|
|
}
|
|
|
|
|
2017-03-09 14:48:42 -05:00
|
|
|
function testMakeAnInvalidQuery() {
|
2017-03-08 22:16:35 -05:00
|
|
|
$this->assertException("engineErrorGeneral", "Db");
|
|
|
|
$this->drv->query("Apollo was astonished; Dionysus thought me mad");
|
|
|
|
}
|
|
|
|
|
|
|
|
function testQueryTimeout() {
|
2017-03-28 18:50:00 -04:00
|
|
|
$ch = new \SQLite3(Data::$conf->dbSQLite3File);
|
2017-03-08 22:16:35 -05:00
|
|
|
$ch->exec("BEGIN EXCLUSIVE TRANSACTION");
|
|
|
|
$this->assertException("general", "Db", "ExceptionTimeout");
|
|
|
|
$this->drv->query("CREATE TABLE test(id integer primary key)");
|
|
|
|
}
|
|
|
|
|
|
|
|
function testQueryConstraintViolation() {
|
|
|
|
$this->drv->exec("CREATE TABLE test(id integer not null)");
|
|
|
|
$this->assertException("constraintViolation", "Db", "ExceptionInput");
|
|
|
|
$this->drv->query("INSERT INTO test(id) values(null)");
|
|
|
|
}
|
2017-03-09 09:44:50 -05:00
|
|
|
|
|
|
|
function testQueryTypeViolation() {
|
|
|
|
$this->drv->exec("CREATE TABLE test(id integer primary key)");
|
|
|
|
$this->assertException("typeViolation", "Db", "ExceptionInput");
|
|
|
|
$this->drv->query("INSERT INTO test(id) values('ook')");
|
|
|
|
}
|
2017-03-09 14:48:42 -05:00
|
|
|
|
|
|
|
function testPrepareAValidQuery() {
|
|
|
|
$s = $this->drv->prepare("SELECT ?, ?", "int", "int");
|
|
|
|
$this->assertInstanceOf(Db\SQLite3\Statement::class, $s);
|
|
|
|
}
|
|
|
|
|
|
|
|
function testPrepareAnInvalidQuery() {
|
|
|
|
$this->assertException("engineErrorGeneral", "Db");
|
|
|
|
$s = $this->drv->prepare("This is an invalid query", "int", "int");
|
|
|
|
}
|
2017-03-09 16:36:33 -05:00
|
|
|
|
|
|
|
function testBeginTransaction() {
|
|
|
|
$select = "SELECT count(*) FROM test";
|
|
|
|
$insert = "INSERT INTO test(id) values(null)";
|
2017-03-28 18:50:00 -04:00
|
|
|
$ch = new \SQLite3(Data::$conf->dbSQLite3File);
|
2017-03-09 16:36:33 -05:00
|
|
|
$this->drv->exec("CREATE TABLE test(id integer primary key)");
|
|
|
|
$this->drv->begin();
|
|
|
|
$this->drv->query($insert);
|
|
|
|
$this->assertEquals(1, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
$this->drv->query($insert);
|
|
|
|
$this->assertEquals(2, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
}
|
|
|
|
|
|
|
|
function testCommitTransaction() {
|
|
|
|
$select = "SELECT count(*) FROM test";
|
|
|
|
$insert = "INSERT INTO test(id) values(null)";
|
2017-03-28 18:50:00 -04:00
|
|
|
$ch = new \SQLite3(Data::$conf->dbSQLite3File);
|
2017-03-09 16:36:33 -05:00
|
|
|
$this->drv->exec("CREATE TABLE test(id integer primary key)");
|
|
|
|
$this->drv->begin();
|
|
|
|
$this->drv->query($insert);
|
|
|
|
$this->assertEquals(1, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
$this->drv->commit();
|
|
|
|
$this->assertEquals(1, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(1, $ch->querySingle($select));
|
|
|
|
}
|
|
|
|
|
|
|
|
function testRollbackTransaction() {
|
|
|
|
$select = "SELECT count(*) FROM test";
|
|
|
|
$insert = "INSERT INTO test(id) values(null)";
|
2017-03-28 18:50:00 -04:00
|
|
|
$ch = new \SQLite3(Data::$conf->dbSQLite3File);
|
2017-03-09 16:36:33 -05:00
|
|
|
$this->drv->exec("CREATE TABLE test(id integer primary key)");
|
|
|
|
$this->drv->begin();
|
|
|
|
$this->drv->query($insert);
|
|
|
|
$this->assertEquals(1, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
$this->drv->rollback();
|
|
|
|
$this->assertEquals(0, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
}
|
|
|
|
|
|
|
|
function testBeginChainedTransactions() {
|
|
|
|
$select = "SELECT count(*) FROM test";
|
|
|
|
$insert = "INSERT INTO test(id) values(null)";
|
2017-03-28 18:50:00 -04:00
|
|
|
$ch = new \SQLite3(Data::$conf->dbSQLite3File);
|
2017-03-09 16:36:33 -05:00
|
|
|
$this->drv->exec("CREATE TABLE test(id integer primary key)");
|
|
|
|
$this->drv->begin();
|
|
|
|
$this->drv->query($insert);
|
|
|
|
$this->assertEquals(1, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
$this->drv->begin();
|
|
|
|
$this->drv->query($insert);
|
|
|
|
$this->assertEquals(2, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
}
|
|
|
|
|
|
|
|
function testCommitChainedTransactions() {
|
|
|
|
$select = "SELECT count(*) FROM test";
|
|
|
|
$insert = "INSERT INTO test(id) values(null)";
|
2017-03-28 18:50:00 -04:00
|
|
|
$ch = new \SQLite3(Data::$conf->dbSQLite3File);
|
2017-03-09 16:36:33 -05:00
|
|
|
$this->drv->exec("CREATE TABLE test(id integer primary key)");
|
|
|
|
$this->drv->begin();
|
|
|
|
$this->drv->query($insert);
|
|
|
|
$this->assertEquals(1, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
$this->drv->begin();
|
|
|
|
$this->drv->query($insert);
|
|
|
|
$this->assertEquals(2, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
$this->drv->commit();
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
$this->drv->commit();
|
|
|
|
$this->assertEquals(2, $ch->querySingle($select));
|
|
|
|
}
|
|
|
|
|
|
|
|
function testRollbackChainedTransactions() {
|
|
|
|
$select = "SELECT count(*) FROM test";
|
|
|
|
$insert = "INSERT INTO test(id) values(null)";
|
2017-03-28 18:50:00 -04:00
|
|
|
$ch = new \SQLite3(Data::$conf->dbSQLite3File);
|
2017-03-09 16:36:33 -05:00
|
|
|
$this->drv->exec("CREATE TABLE test(id integer primary key)");
|
|
|
|
$this->drv->begin();
|
|
|
|
$this->drv->query($insert);
|
|
|
|
$this->assertEquals(1, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
$this->drv->begin();
|
|
|
|
$this->drv->query($insert);
|
|
|
|
$this->assertEquals(2, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
$this->drv->rollback();
|
|
|
|
$this->assertEquals(1, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
$this->drv->rollback();
|
|
|
|
$this->assertEquals(0, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
}
|
|
|
|
|
|
|
|
function testPartiallyRollbackChainedTransactions() {
|
|
|
|
$select = "SELECT count(*) FROM test";
|
|
|
|
$insert = "INSERT INTO test(id) values(null)";
|
2017-03-28 18:50:00 -04:00
|
|
|
$ch = new \SQLite3(Data::$conf->dbSQLite3File);
|
2017-03-09 16:36:33 -05:00
|
|
|
$this->drv->exec("CREATE TABLE test(id integer primary key)");
|
|
|
|
$this->drv->begin();
|
|
|
|
$this->drv->query($insert);
|
|
|
|
$this->assertEquals(1, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
$this->drv->begin();
|
|
|
|
$this->drv->query($insert);
|
|
|
|
$this->assertEquals(2, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
$this->drv->rollback();
|
|
|
|
$this->assertEquals(1, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
$this->drv->commit();
|
|
|
|
$this->assertEquals(1, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(1, $ch->querySingle($select));
|
|
|
|
}
|
|
|
|
|
|
|
|
function testFullyRollbackChainedTransactions() {
|
|
|
|
$select = "SELECT count(*) FROM test";
|
|
|
|
$insert = "INSERT INTO test(id) values(null)";
|
2017-03-28 18:50:00 -04:00
|
|
|
$ch = new \SQLite3(Data::$conf->dbSQLite3File);
|
2017-03-09 16:36:33 -05:00
|
|
|
$this->drv->exec("CREATE TABLE test(id integer primary key)");
|
|
|
|
$this->drv->begin();
|
|
|
|
$this->drv->query($insert);
|
|
|
|
$this->assertEquals(1, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
$this->drv->begin();
|
|
|
|
$this->drv->query($insert);
|
|
|
|
$this->assertEquals(2, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
$this->drv->rollback(true);
|
|
|
|
$this->assertEquals(0, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
}
|
|
|
|
|
|
|
|
function testFullyCommitChainedTransactions() {
|
|
|
|
$select = "SELECT count(*) FROM test";
|
|
|
|
$insert = "INSERT INTO test(id) values(null)";
|
2017-03-28 18:50:00 -04:00
|
|
|
$ch = new \SQLite3(Data::$conf->dbSQLite3File);
|
2017-03-09 16:36:33 -05:00
|
|
|
$this->drv->exec("CREATE TABLE test(id integer primary key)");
|
|
|
|
$this->drv->begin();
|
|
|
|
$this->drv->query($insert);
|
|
|
|
$this->assertEquals(1, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
$this->drv->begin();
|
|
|
|
$this->drv->query($insert);
|
|
|
|
$this->assertEquals(2, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(0, $ch->querySingle($select));
|
|
|
|
$this->drv->commit(true);
|
|
|
|
$this->assertEquals(2, $this->drv->query($select)->getValue());
|
|
|
|
$this->assertEquals(2, $ch->querySingle($select));
|
|
|
|
}
|
2017-03-09 17:14:26 -05:00
|
|
|
|
2017-03-09 17:25:50 -05:00
|
|
|
function testFetchSchemaVersion() {
|
|
|
|
$this->assertSame(0, $this->drv->schemaVersion());
|
|
|
|
$this->drv->exec("PRAGMA user_version=1");
|
|
|
|
$this->assertSame(1, $this->drv->schemaVersion());
|
|
|
|
$this->drv->exec("PRAGMA user_version=2");
|
|
|
|
$this->assertSame(2, $this->drv->schemaVersion());
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2017-03-09 17:14:26 -05:00
|
|
|
function testManipulateAdvisoryLock() {
|
2017-03-09 21:39:42 -05:00
|
|
|
$this->assertTrue($this->drv->unlock());
|
2017-03-09 17:14:26 -05:00
|
|
|
$this->assertFalse($this->drv->isLocked());
|
|
|
|
$this->assertTrue($this->drv->lock());
|
|
|
|
$this->assertFalse($this->drv->isLocked());
|
2017-03-27 23:12:12 -05:00
|
|
|
$this->drv->exec("CREATE TABLE arsse_settings(key primary key, value, type) without rowid; PRAGMA user_version=1");
|
2017-03-09 17:14:26 -05:00
|
|
|
$this->assertTrue($this->drv->lock());
|
|
|
|
$this->assertTrue($this->drv->isLocked());
|
|
|
|
$this->assertFalse($this->drv->lock());
|
|
|
|
$this->drv->exec("PRAGMA user_version=0");
|
|
|
|
$this->assertFalse($this->drv->isLocked());
|
|
|
|
$this->assertTrue($this->drv->lock());
|
|
|
|
$this->assertFalse($this->drv->isLocked());
|
|
|
|
$this->drv->exec("PRAGMA user_version=1");
|
|
|
|
$this->assertTrue($this->drv->isLocked());
|
|
|
|
$this->assertTrue($this->drv->unlock());
|
|
|
|
$this->assertFalse($this->drv->isLocked());
|
|
|
|
}
|
2017-03-08 22:16:35 -05:00
|
|
|
}
|