2017-03-01 22:47:51 -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-01 22:47:51 -05:00
|
|
|
declare(strict_types=1);
|
2017-12-21 22:47:19 -05:00
|
|
|
namespace JKingWeb\Arsse\TestCase\Db\SQLite3;
|
2017-03-01 22:47:51 -05:00
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
use JKingWeb\Arsse\Db\Statement;
|
2017-03-01 22:47:51 -05:00
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
/**
|
2017-07-20 18:36:03 -04:00
|
|
|
* @covers \JKingWeb\Arsse\Db\SQLite3\Statement<extended>
|
|
|
|
* @covers \JKingWeb\Arsse\Db\SQLite3\ExceptionBuilder */
|
2017-12-21 22:47:19 -05:00
|
|
|
class TestStatement extends \JKingWeb\Arsse\Test\AbstractTest {
|
|
|
|
use \JKingWeb\Arsse\Test\Db\BindingTests;
|
2017-03-01 22:47:51 -05:00
|
|
|
|
|
|
|
protected $c;
|
2017-12-21 22:47:19 -05:00
|
|
|
protected static $imp = \JKingWeb\Arsse\Db\SQLite3\Statement::class;
|
2017-03-01 22:47:51 -05:00
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function setUp() {
|
2017-07-20 18:36:03 -04:00
|
|
|
$this->clearData();
|
2017-08-29 10:50:31 -04:00
|
|
|
if (!extension_loaded("sqlite3")) {
|
2017-05-22 10:02:36 -04:00
|
|
|
$this->markTestSkipped("SQLite extension not loaded");
|
|
|
|
}
|
2017-03-01 22:47:51 -05:00
|
|
|
$c = new \SQLite3(":memory:");
|
|
|
|
$c->enableExceptions(true);
|
|
|
|
$this->c = $c;
|
|
|
|
}
|
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function tearDown() {
|
2017-03-01 22:47:51 -05:00
|
|
|
$this->c->close();
|
|
|
|
unset($this->c);
|
|
|
|
}
|
|
|
|
|
2017-05-04 19:38:54 -04:00
|
|
|
protected function checkBinding($input, array $expectations, bool $strict = false) {
|
2017-04-06 21:41:21 -04:00
|
|
|
$nativeStatement = $this->c->prepare("SELECT ? as value");
|
|
|
|
$s = new self::$imp($this->c, $nativeStatement);
|
|
|
|
$types = array_unique(Statement::TYPES);
|
2017-08-29 10:50:31 -04:00
|
|
|
foreach ($types as $type) {
|
2017-05-04 19:38:54 -04:00
|
|
|
$s->rebindArray([$strict ? "strict $type" : $type]);
|
2017-04-06 21:41:21 -04:00
|
|
|
$val = $s->runArray([$input])->getRow()['value'];
|
2017-07-07 21:06:38 -04:00
|
|
|
$this->assertSame($expectations[$type], $val, "Binding from type $type failed comparison.");
|
2017-07-22 15:29:12 -04:00
|
|
|
$s->rebind(...[$strict ? "strict $type" : $type]);
|
|
|
|
$val = $s->run(...[$input])->getRow()['value'];
|
|
|
|
$this->assertSame($expectations[$type], $val, "Binding from type $type failed comparison.");
|
2017-04-06 21:41:21 -04:00
|
|
|
}
|
|
|
|
}
|
2017-03-03 13:20:26 -05:00
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function testConstructStatement() {
|
2017-03-03 13:20:26 -05:00
|
|
|
$nativeStatement = $this->c->prepare("SELECT ? as value");
|
2017-12-21 22:47:19 -05:00
|
|
|
$this->assertInstanceOf(Statement::class, new \JKingWeb\Arsse\Db\SQLite3\Statement($this->c, $nativeStatement));
|
2017-04-06 21:41:21 -04:00
|
|
|
}
|
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function testBindMissingValue() {
|
2017-04-06 21:41:21 -04:00
|
|
|
$nativeStatement = $this->c->prepare("SELECT ? as value");
|
|
|
|
$s = new self::$imp($this->c, $nativeStatement);
|
|
|
|
$val = $s->runArray()->getRow()['value'];
|
|
|
|
$this->assertSame(null, $val);
|
|
|
|
}
|
2017-03-03 13:20:26 -05:00
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function testBindMultipleValues() {
|
2017-03-03 13:20:26 -05:00
|
|
|
$exp = [
|
|
|
|
'one' => 1,
|
|
|
|
'two' => 2,
|
|
|
|
];
|
2017-04-06 21:41:21 -04:00
|
|
|
$nativeStatement = $this->c->prepare("SELECT ? as one, ? as two");
|
|
|
|
$s = new self::$imp($this->c, $nativeStatement, ["int", "int"]);
|
|
|
|
$val = $s->runArray([1,2])->getRow();
|
|
|
|
$this->assertSame($exp, $val);
|
2017-03-03 13:20:26 -05:00
|
|
|
}
|
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function testBindRecursively() {
|
2017-04-06 21:24:26 -04:00
|
|
|
$exp = [
|
|
|
|
'one' => 1,
|
|
|
|
'two' => 2,
|
|
|
|
'three' => 3,
|
2017-04-06 21:41:21 -04:00
|
|
|
'four' => 4,
|
2017-04-06 21:24:26 -04:00
|
|
|
];
|
2017-04-06 21:41:21 -04:00
|
|
|
$nativeStatement = $this->c->prepare("SELECT ? as one, ? as two, ? as three, ? as four");
|
|
|
|
$s = new self::$imp($this->c, $nativeStatement, ["int", ["int", "int"], "int"]);
|
|
|
|
$val = $s->runArray([1, [2, 3], 4])->getRow();
|
|
|
|
$this->assertSame($exp, $val);
|
|
|
|
}
|
2017-04-06 21:24:26 -04:00
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function testBindWithoutType() {
|
2017-03-08 13:59:20 -05:00
|
|
|
$nativeStatement = $this->c->prepare("SELECT ? as value");
|
2017-04-06 21:41:21 -04:00
|
|
|
$this->assertException("paramTypeMissing", "Db");
|
|
|
|
$s = new self::$imp($this->c, $nativeStatement, []);
|
|
|
|
$s->runArray([1]);
|
2017-03-03 13:20:26 -05:00
|
|
|
}
|
2017-03-08 13:59:20 -05:00
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function testViolateConstraint() {
|
2017-04-06 21:41:21 -04:00
|
|
|
$this->c->exec("CREATE TABLE test(id integer not null)");
|
|
|
|
$nativeStatement = $this->c->prepare("INSERT INTO test(id) values(?)");
|
|
|
|
$s = new self::$imp($this->c, $nativeStatement, ["int"]);
|
|
|
|
$this->assertException("constraintViolation", "Db", "ExceptionInput");
|
|
|
|
$s->runArray([null]);
|
|
|
|
}
|
2017-03-09 09:44:50 -05:00
|
|
|
|
2017-08-29 10:50:31 -04:00
|
|
|
public function testMismatchTypes() {
|
2017-04-06 21:41:21 -04:00
|
|
|
$this->c->exec("CREATE TABLE test(id integer primary key)");
|
|
|
|
$nativeStatement = $this->c->prepare("INSERT INTO test(id) values(?)");
|
|
|
|
$s = new self::$imp($this->c, $nativeStatement, ["str"]);
|
|
|
|
$this->assertException("typeViolation", "Db", "ExceptionInput");
|
|
|
|
$s->runArray(['ook']);
|
|
|
|
}
|
2017-08-29 10:50:31 -04:00
|
|
|
}
|