1
1
Fork 0
mirror of https://code.mensbeam.com/MensBeam/Arsse.git synced 2025-01-08 17:02:41 +00:00

Tests for Database::meta*() functions

Improves #49
This commit is contained in:
J. King 2017-07-18 16:38:23 -04:00
parent feadf51096
commit 3a26c75044
8 changed files with 114 additions and 23 deletions

View file

@ -8,10 +8,6 @@ use JKingWeb\Arsse\Misc\Date;
class Database { class Database {
const SCHEMA_VERSION = 1; const SCHEMA_VERSION = 1;
const FORMAT_TS = "Y-m-d h:i:s";
const FORMAT_DATE = "Y-m-d";
const FORMAT_TIME = "h:i:s";
/** @var Db\Driver */ /** @var Db\Driver */
public $db; public $db;
@ -29,7 +25,7 @@ class Database {
return debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS, 3)[2]['function']; return debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS, 3)[2]['function'];
} }
static public function listDrivers(): array { static public function driverList(): array {
$sep = \DIRECTORY_SEPARATOR; $sep = \DIRECTORY_SEPARATOR;
$path = __DIR__.$sep."Db".$sep; $path = __DIR__.$sep."Db".$sep;
$classes = []; $classes = [];
@ -41,11 +37,11 @@ class Database {
return $classes; return $classes;
} }
public function schemaVersion(): int { public function driverSchemaVersion(): int {
return $this->db->schemaVersion(); return $this->db->schemaVersion();
} }
public function schemaUpdate(): bool { public function driverSchemaUpdate(): bool {
if($this->db->schemaVersion() < self::SCHEMA_VERSION) return $this->db->schemaUpdate(self::SCHEMA_VERSION); if($this->db->schemaVersion() < self::SCHEMA_VERSION) return $this->db->schemaUpdate(self::SCHEMA_VERSION);
return false; return false;
} }
@ -86,17 +82,16 @@ class Database {
return $this->db->prepare("SELECT value from arsse_meta where key is ?", "str")->run($key)->getValue(); return $this->db->prepare("SELECT value from arsse_meta where key is ?", "str")->run($key)->getValue();
} }
public function metaSet(string $key, string $value, string $type = "str"): bool { public function metaSet(string $key, $value, string $type = "str"): bool {
$out = !$this->db->prepare("UPDATE arsse_meta set value = ? where key is ?", $type, "str")->run($value, $key)->changes(); $out = $this->db->prepare("UPDATE arsse_meta set value = ? where key is ?", $type, "str")->run($value, $key)->changes();
if(!$out) { if(!$out) {
$out = $this->db->prepare("INSERT INTO arsse_meta(key,value)", "str", $type)->run($key, $value)->changes(); $out = $this->db->prepare("INSERT INTO arsse_meta(key,value) values(?,?)", "str", $type)->run($key, $value)->changes();
} }
return (bool) $out; return (bool) $out;
} }
public function metaRemove(string $key): bool { public function metaRemove(string $key): bool {
$this->db->prepare("DELETE from arsse_meta where key is ?", "str")->run($key); return (bool) $this->db->prepare("DELETE from arsse_meta where key is ?", "str")->run($key)->changes();
return true;
} }
public function userExists(string $user): bool { public function userExists(string $user): bool {
@ -776,7 +771,7 @@ class Database {
return (bool) $out; return (bool) $out;
} }
public function articleValidateId(string $user, int $id): array { protected function articleValidateId(string $user, int $id): array {
$out = $this->db->prepare( $out = $this->db->prepare(
"SELECT "SELECT
arsse_articles.id as article, arsse_articles.id as article,

View file

@ -52,7 +52,7 @@ class Feed {
$this->nextFetch = $this->computeNextFetch(); $this->nextFetch = $this->computeNextFetch();
} }
public function download(string $url, string $lastModified = '', string $etag = '', string $username = '', string $password = ''): bool { protected function download(string $url, string $lastModified = '', string $etag = '', string $username = '', string $password = ''): bool {
try { try {
$this->reader = new Reader($this->config); $this->reader = new Reader($this->config);
$this->resource = $this->reader->download($url, $lastModified, $etag, $username, $password); $this->resource = $this->reader->download($url, $lastModified, $etag, $username, $password);
@ -62,7 +62,7 @@ class Feed {
return true; return true;
} }
public function parse(): bool { protected function parse(): bool {
try { try {
$this->parser = $this->reader->getParser( $this->parser = $this->reader->getParser(
$this->resource->getUrl(), $this->resource->getUrl(),
@ -192,7 +192,7 @@ class Feed {
return $out; return $out;
} }
public function matchToDatabase(int $feedID = null): bool { protected function matchToDatabase(int $feedID = null): bool {
// first perform deduplication on items // first perform deduplication on items
$items = $this->deduplicateItems($this->data->items); $items = $this->deduplicateItems($this->data->items);
// if we haven't been given a database feed ID to check against, all items are new // if we haven't been given a database feed ID to check against, all items are new
@ -221,7 +221,7 @@ class Feed {
return true; return true;
} }
public function matchItems(array $items, array $articles): array { protected function matchItems(array $items, array $articles): array {
$new = $edited = []; $new = $edited = [];
// iterate through the articles and for each determine whether it is existing, edited, or entirely new // iterate through the articles and for each determine whether it is existing, edited, or entirely new
foreach($items as $i) { foreach($items as $i) {
@ -260,7 +260,7 @@ class Feed {
return [$new, $edited]; return [$new, $edited];
} }
public function computeNextFetch(): \DateTime { protected function computeNextFetch(): \DateTime {
$now = Date::normalize(time()); $now = Date::normalize(time());
if(!$this->modified) { if(!$this->modified) {
$diff = $now->getTimestamp() - $this->lastModified->getTimestamp(); $diff = $now->getTimestamp() - $this->lastModified->getTimestamp();
@ -318,7 +318,7 @@ class Feed {
return $offset; return $offset;
} }
public function computeLastModified() { protected function computeLastModified() {
if(!$this->modified) return $this->lastModified; if(!$this->modified) return $this->lastModified;
$dates = $this->gatherDates(); $dates = $this->gatherDates();
if(sizeof($dates)) { if(sizeof($dates)) {

View file

@ -10,6 +10,18 @@ class Service {
/** @var \DateInterval */ /** @var \DateInterval */
protected $interval; protected $interval;
static public function driverList(): array {
$sep = \DIRECTORY_SEPARATOR;
$path = __DIR__.$sep."Service".$sep;
$classes = [];
foreach(glob($path."*".$sep."Driver.php") as $file) {
$name = basename(dirname($file));
$class = NS_BASE."User\\$name\\Driver";
$classes[$class] = $class::driverName();
}
return $classes;
}
protected static function interval(): \DateInterval { protected static function interval(): \DateInterval {
return new \DateInterval(Arsse::$conf->serviceFrequency); // FIXME: this needs to fall back in case of incorrect input return new \DateInterval(Arsse::$conf->serviceFrequency); // FIXME: this needs to fall back in case of incorrect input
} }
@ -32,6 +44,7 @@ class Service {
$this->drv->exec(); $this->drv->exec();
$this->drv->clean(); $this->drv->clean();
static::cleanupPost(); static::cleanupPost();
unset($list);
} }
$t->add($this->interval); $t->add($this->interval);
do { do {

View file

@ -19,7 +19,7 @@ class User {
protected $authzSupported = 0; protected $authzSupported = 0;
protected $actor = []; protected $actor = [];
static public function listDrivers(): array { static public function driverList(): array {
$sep = \DIRECTORY_SEPARATOR; $sep = \DIRECTORY_SEPARATOR;
$path = __DIR__.$sep."User".$sep; $path = __DIR__.$sep."User".$sep;
$classes = []; $classes = [];

View file

@ -0,0 +1,9 @@
<?php
declare(strict_types=1);
namespace JKingWeb\Arsse;
class TestDatabaseMetaSQLite3 extends Test\AbstractTest {
use Test\Database\Setup;
use Test\Database\DriverSQLite3;
use Test\Database\SeriesMeta;
}

View file

@ -0,0 +1,70 @@
<?php
declare(strict_types=1);
namespace JKingWeb\Arsse\Test\Database;
use JKingWeb\Arsse\Test\Database;
use JKingWeb\Arsse\Arsse;
trait SeriesMeta {
protected $dataBare = [
'arsse_meta' => [
'columns' => [
'key' => 'str',
'value' => 'str',
],
'rows' => [
//['schema_version', "".\JKingWeb\Arsse\Database::SCHEMA_VERSION],
['album',"A Farewell to Kings"],
],
],
];
function setUpSeries() {
// the schema_version key is a special case, and to avoid jumping through hoops for every test we deal with it now
$this->data = $this->dataBare;
// as far as tests are concerned the schema version is part of the expectations primed into the database
array_unshift($this->data['arsse_meta']['rows'], ['schema_version', "".Database::SCHEMA_VERSION]);
// but it's already been inserted by the driver, so we prime without it
$this->primeDatabase($this->dataBare);
}
function testAddANewValue() {
$this->assertTrue(Arsse::$db->metaSet("favourite", "Cygnus X-1"));
$state = $this->primeExpectations($this->data, ['arsse_meta' => ['key','value']]);
$state['arsse_meta']['rows'][] = ["favourite","Cygnus X-1"];
$this->compareExpectations($state);
}
function testAddANewTypedValue() {
$this->assertTrue(Arsse::$db->metaSet("answer", 42, "int"));
$this->assertTrue(Arsse::$db->metaSet("true", true, "bool"));
$this->assertTrue(Arsse::$db->metaSet("false", false, "bool"));
$this->assertTrue(Arsse::$db->metaSet("millennium", new \DateTime("2000-01-01T00:00:00Z"), "datetime"));
$state = $this->primeExpectations($this->data, ['arsse_meta' => ['key','value']]);
$state['arsse_meta']['rows'][] = ["answer","42"];
$state['arsse_meta']['rows'][] = ["true","1"];
$state['arsse_meta']['rows'][] = ["false","0"];
$state['arsse_meta']['rows'][] = ["millennium","2000-01-01 00:00:00"];
$this->compareExpectations($state);
}
function testChangeAnExistingValue() {
$this->assertTrue(Arsse::$db->metaSet("album", "Hemispheres"));
$state = $this->primeExpectations($this->data, ['arsse_meta' => ['key','value']]);
$state['arsse_meta']['rows'][1][1] = "Hemispheres";
$this->compareExpectations($state);
}
function testRemoveAValue() {
$this->assertTrue(Arsse::$db->metaRemove("album"));
$this->assertFalse(Arsse::$db->metaRemove("album"));
$state = $this->primeExpectations($this->data, ['arsse_meta' => ['key','value']]);
unset($state['arsse_meta']['rows'][1]);
$this->compareExpectations($state);
}
function testRetrieveAValue() {
$this->assertSame("".Database::SCHEMA_VERSION, Arsse::$db->metaGet("schema_version"));
$this->assertSame("A Farewell to Kings", Arsse::$db->metaGet("album"));
$this->assertSame(null, Arsse::$db->metaGet("this_key_does_not_exist"));
}
}

View file

@ -11,6 +11,7 @@ use Phake;
trait Setup { trait Setup {
protected $drv; protected $drv;
protected $primed = false;
function setUp() { function setUp() {
// establish a clean baseline // establish a clean baseline
@ -21,20 +22,21 @@ trait Setup {
$this->setUpDriver(); $this->setUpDriver();
// create the database interface with the suitable driver // create the database interface with the suitable driver
Arsse::$db = new Database($this->drv); Arsse::$db = new Database($this->drv);
Arsse::$db->schemaUpdate(); Arsse::$db->driverSchemaUpdate();
// create a mock user manager // create a mock user manager
Arsse::$user = Phake::mock(User::class); Arsse::$user = Phake::mock(User::class);
Phake::when(Arsse::$user)->authorize->thenReturn(true); Phake::when(Arsse::$user)->authorize->thenReturn(true);
// call the additional setup method if it exists // call the additional setup method if it exists
if(method_exists($this, "setUpSeries")) $this->setUpSeries(); if(method_exists($this, "setUpSeries")) $this->setUpSeries();
// prime the database with series data // prime the database with series data if it hasn't already been done
if(isset($this->data)) $this->primeDatabase($this->data); if(!$this->primed && isset($this->data)) $this->primeDatabase($this->data);
} }
function tearDown() { function tearDown() {
// call the additional teardiwn method if it exists // call the additional teardiwn method if it exists
if(method_exists($this, "tearDownSeries")) $this->tearDownSeries(); if(method_exists($this, "tearDownSeries")) $this->tearDownSeries();
// clean up // clean up
$this->primed = false;
$this->drv = null; $this->drv = null;
$this->clearData(); $this->clearData();
} }
@ -51,6 +53,7 @@ trait Setup {
} }
} }
$tr->commit(); $tr->commit();
$this->primed = true;
return true; return true;
} }

View file

@ -42,6 +42,7 @@
<file>Db/SQLite3/TestDbUpdateSQLite3.php</file> <file>Db/SQLite3/TestDbUpdateSQLite3.php</file>
</testsuite> </testsuite>
<testsuite name="Database functions"> <testsuite name="Database functions">
<file>Db/SQLite3/Database/TestDatabaseMetaSQLite3.php</file>
<file>Db/SQLite3/Database/TestDatabaseUserSQLite3.php</file> <file>Db/SQLite3/Database/TestDatabaseUserSQLite3.php</file>
<file>Db/SQLite3/Database/TestDatabaseFolderSQLite3.php</file> <file>Db/SQLite3/Database/TestDatabaseFolderSQLite3.php</file>
<file>Db/SQLite3/Database/TestDatabaseFeedSQLite3.php</file> <file>Db/SQLite3/Database/TestDatabaseFeedSQLite3.php</file>