From d1da6fbe5ef6c6e1da44b21e2162399563a0cbb1 Mon Sep 17 00:00:00 2001 From: "J. King" Date: Mon, 30 May 2022 17:29:34 -0400 Subject: [PATCH 01/10] Use cases rather than casting bools to int in SQL --- lib/Database.php | 14 +++++--------- lib/Db/MySQL/Driver.php | 2 -- tests/cases/Db/BaseDriver.php | 2 -- 3 files changed, 5 insertions(+), 13 deletions(-) diff --git a/lib/Database.php b/lib/Database.php index 3bfed979..107c6dff 100644 --- a/lib/Database.php +++ b/lib/Database.php @@ -814,7 +814,6 @@ class Database { // validate inputs $folder = $this->folderValidateId($user, $folder)['id']; // create a complex query - $integer = $this->db->sqlToken("integer"); $q = new Query( "WITH RECURSIVE topmost(f_id, top) as ( @@ -853,7 +852,7 @@ class Database { select subscription, sum(hidden) as hidden, - sum(cast((\"read\" = 1 and hidden = 0) as $integer)) as marked + sum(case when \"read\" = 1 and hidden = 0 then 1 else 0 end) as marked from arsse_marks group by subscription ) as mark_stats on mark_stats.subscription = s.id", ["str", "int"], @@ -2053,7 +2052,6 @@ class Database { /** Deletes from the database articles which are beyond the configured clean-up threshold */ public function articleCleanup(): bool { - $integer = $this->db->sqlToken("integer"); $query = $this->db->prepareArray( "WITH RECURSIVE exempt_articles as ( @@ -2079,8 +2077,8 @@ class Database { left join ( select article, - sum(cast((starred = 1 and hidden = 0) as $integer)) as starred, - sum(cast((\"read\" = 1 or hidden = 1) as $integer)) as \"read\", + sum(case when starred = 1 and hidden = 0 then 1 else 0 end) as starred, + sum(case when \"read\" = 1 or hidden = 1 then 1 else 0 end) as \"read\", max(arsse_marks.modified) as marked_date from arsse_marks group by article @@ -2211,7 +2209,6 @@ class Database { * @param boolean $includeEmpty Whether to include (true) or supress (false) labels which have no articles assigned to them */ public function labelList(string $user, bool $includeEmpty = true): Db\Result { - $integer = $this->db->sqlToken("integer"); return $this->db->prepareArray( "SELECT * FROM ( SELECT @@ -2227,7 +2224,7 @@ class Database { SELECT label, sum(hidden) as hidden, - sum(cast((\"read\" = 1 and hidden = 0) as $integer)) as marked + sum(case when \"read\" = 1 and hidden = 0 then 1 else 0 end) as marked from arsse_marks join arsse_subscriptions on arsse_subscriptions.id = arsse_marks.subscription join arsse_label_members on arsse_label_members.article = arsse_marks.article @@ -2277,7 +2274,6 @@ class Database { $this->labelValidateId($user, $id, $byName, false); $field = $byName ? "name" : "id"; $type = $byName ? "str" : "int"; - $integer = $this->db->sqlToken("integer"); $out = $this->db->prepareArray( "SELECT id, @@ -2292,7 +2288,7 @@ class Database { SELECT label, sum(hidden) as hidden, - sum(cast((\"read\" = 1 and hidden = 0) as $integer)) as marked + sum(case when \"read\" = 1 and hidden = 0 then 1 else 0 end) as marked from arsse_marks join arsse_subscriptions on arsse_subscriptions.id = arsse_marks.subscription join arsse_label_members on arsse_label_members.article = arsse_marks.article diff --git a/lib/Db/MySQL/Driver.php b/lib/Db/MySQL/Driver.php index c61762a1..de3bc3d6 100644 --- a/lib/Db/MySQL/Driver.php +++ b/lib/Db/MySQL/Driver.php @@ -81,8 +81,6 @@ class Driver extends \JKingWeb\Arsse\Db\AbstractDriver { switch (strtolower($token)) { case "nocase": return '"utf8mb4_unicode_ci"'; - case "integer": - return "signed integer"; case "asc": return ""; default: diff --git a/tests/cases/Db/BaseDriver.php b/tests/cases/Db/BaseDriver.php index fe7f344c..0999d3a7 100644 --- a/tests/cases/Db/BaseDriver.php +++ b/tests/cases/Db/BaseDriver.php @@ -384,7 +384,6 @@ abstract class BaseDriver extends \JKingWeb\Arsse\Test\AbstractTest { $greatest = $this->drv->sqlToken("GrEatESt"); $nocase = $this->drv->sqlToken("noCASE"); $like = $this->drv->sqlToken("liKe"); - $integer = $this->drv->sqlToken("InTEGer"); $asc = $this->drv->sqlToken("asc"); $desc = $this->drv->sqlToken("desc"); $least = $this->drv->sqlToken("leASt"); @@ -395,7 +394,6 @@ abstract class BaseDriver extends \JKingWeb\Arsse\Test\AbstractTest { $this->assertSame("Z", $this->drv->query("SELECT $greatest('Z', 'A')")->getValue()); $this->assertSame("Z", $this->drv->query("SELECT 'Z' collate $nocase")->getValue()); $this->assertSame("Z", $this->drv->query("SELECT 'Z' where 'Z' $like 'z'")->getValue()); - $this->assertEquals(1, $this->drv->query("SELECT CAST((1=1) as $integer)")->getValue()); $this->assertEquals([null, 1, 2], array_column($this->drv->query("SELECT 1 as t union select null as t union select 2 as t order by t $asc")->getAll(), "t")); $this->assertEquals([2, 1, null], array_column($this->drv->query("SELECT 1 as t union select null as t union select 2 as t order by t $desc")->getAll(), "t")); } From 2c19aa06b71ec56968e94a7b2a3629d72c175de9 Mon Sep 17 00:00:00 2001 From: "J. King" Date: Tue, 31 May 2022 23:08:05 -0400 Subject: [PATCH 02/10] Put column defs in one place in tests --- tests/cases/Database/TestDatabase.php | 146 ++++++++++++++++++++++++++ 1 file changed, 146 insertions(+) diff --git a/tests/cases/Database/TestDatabase.php b/tests/cases/Database/TestDatabase.php index 00838b3a..7408d7cd 100644 --- a/tests/cases/Database/TestDatabase.php +++ b/tests/cases/Database/TestDatabase.php @@ -10,6 +10,152 @@ use JKingWeb\Arsse\Database; /** @covers \JKingWeb\Arsse\Database */ class TestDatabase extends \JKingWeb\Arsse\Test\AbstractTest { + protected const COL_DEFS = [ + 'arsse_meta' => [ + 'key' => "strict str", + 'value' => "str", + ], + 'arsse_users' => [ + 'id' => "strict str", + 'password' => "str", + 'num' => "strict int", + 'admin' => "strict bool", + ], + 'arsse_user_meta' => [ + 'owner' => "strict str", + 'key' => "strict str", + 'modified' => "strict datetime", + 'value' => "str", + ], + 'arsse_sessions' => [ + 'id' => "strict str", + 'created' => "strict datetime", + 'expires' => "strict datetime", + 'user' => "strict str", + ], + 'arsse_tokens' => [ + 'id' => "strict str", + 'class' => "strict str", + 'user' => "strict str", + 'created' => "strict datetime", + 'expires' => "datetime", + 'data' => "str", + ], + 'arsse_feeds' => [ + 'id' => "int", + 'url' => "strict str", + 'title' => "str", + 'source' => "str", + 'updated' => "datetime", + 'modified' => "datetime", + 'next_fetch' => "datetime", + 'orphaned' => "datetime", + 'etag' => "strict str", + 'err_count' => "strict int", + 'err_msg' => "str", + 'username' => "strict str", + 'password' => "strict str", + 'size' => "strict int", + 'icon' => "int", + ], + 'arsse_icons' => [ + 'id' => "int", + 'url' => "strict str", + 'modified' => "datetime", + 'etag' => "strict str", + 'next_fetch' => "datetime", + 'orphaned' => "datetime", + 'type' => "str", + 'data' => "blob", + ], + 'arsse_articles' => [ + 'id' => "int", + 'feed' => "strict int", + 'url' => "str", + 'title' => "str", + 'author' => "str", + 'published' => "datetime", + 'edited' => "datetime", + 'modified' => "strict datetime", + 'guid' => "str", + 'url_title_hash' => "strict str", + 'url_content_hash' => "strict str", + 'title_content_hash' => "strict str", + 'content_scraped' => "str", + 'content' => "str", + ], + 'arsse_editions' => [ + 'id' => "int", + 'article' => "strict int", + 'modified' => "strict datetime", + ], + 'arsse_enclosures' => [ + 'article' => "strict int", + 'url' => "str", + 'type' => "str", + ], + 'arsse_categories' => [ + 'article' => "strict int", + 'name' => "str", + ], + 'arsse_marks' => [ + 'article' => "strict int", + 'subscription' => "strict int", + 'read' => "strict bool", + 'starred' => "strict bool", + 'modified' => "datetime", + 'note' => "strict str", + 'touched' => "strict bool", + 'hidden' => "strict bool", + ], + 'arsse_subscriptions' => [ + 'id' => "int", + 'owner' => "strict str", + 'feed' => "strict int", + 'added' => "strict datetime", + 'modified' => "strict datetime", + 'title' => "str", + 'order_type' => "strict int", + 'pinned' => "strict bool", + 'folder' => "int", + 'keep_rule' => "str", + 'block_rule' => "str", + 'scrape' => "strict bool", + ], + 'arsse_folders' => [ + 'id' => "int", + 'owner' => "strict str", + 'parent' => "int", + 'name' => "strict str", + 'modified' => "strict datetime", + ], + 'arsse_tags' => [ + 'id' => "int", + 'owner' => "strict str", + 'name' => "strict str", + 'modified' => "strict datetime", + ], + 'arsse_tag_members' => [ + 'tag' => "strict int", + 'subscription' => "strict int", + 'assigned' => "strict bool", + 'modified' => "strict datetime", + ], + 'arsse_labels' => [ + 'id' => "int", + 'owner' => "strict str", + 'name' => "strict str", + 'modified' => "strict datetime", + ], + 'arsse_label_members' => [ + 'label' => "strict int", + 'article' => "strict int", + 'subscription' => "strict int", + 'assigned' => "strict bool", + 'modified' => "strict datetime", + ], + ]; + protected $db = null; public function setUp(): void { From 4ed650fd87c63bf574ef62b6de04d08cd3134eb0 Mon Sep 17 00:00:00 2001 From: "J. King" Date: Tue, 31 May 2022 23:21:33 -0400 Subject: [PATCH 03/10] Style fixes --- .gitignore | 1 + .php_cs.dist => .php-cs-fixer.dist.php | 4 +- composer.lock | 56 +- lib/Context/ExclusionMembers.php | 1 - lib/Database.php | 40 +- lib/REST/TinyTinyRSS/Search.php | 2 +- tests/cases/Database/TestDatabase.php | 172 +++--- tests/cases/Misc/TestContext.php | 4 +- tests/cases/Misc/TestQuery.php | 6 +- tests/cases/REST/TinyTinyRSS/TestAPI.php | 10 +- vendor-bin/csfixer/composer.json | 2 +- vendor-bin/csfixer/composer.lock | 692 +++++++++-------------- vendor-bin/daux/composer.lock | 113 ++-- vendor-bin/phpstan/composer.lock | 2 +- vendor-bin/phpunit/composer.lock | 80 +-- vendor-bin/robo/composer.lock | 245 +++++--- 16 files changed, 663 insertions(+), 767 deletions(-) rename .php_cs.dist => .php-cs-fixer.dist.php (96%) diff --git a/.gitignore b/.gitignore index 42f0b63a..b367592e 100644 --- a/.gitignore +++ b/.gitignore @@ -11,6 +11,7 @@ /arsse.db* /config.php /.php_cs.cache +/.php-cs-fixer.cache /tests/.phpunit.result.cache # Dependencies diff --git a/.php_cs.dist b/.php-cs-fixer.dist.php similarity index 96% rename from .php_cs.dist rename to .php-cs-fixer.dist.php index c2761677..925d4d19 100644 --- a/.php_cs.dist +++ b/.php-cs-fixer.dist.php @@ -48,7 +48,7 @@ $rules = [ 'pow_to_exponentiation' => true, 'set_type_to_cast' => true, 'standardize_not_equals' => true, - 'trailing_comma_in_multiline_array' => true, + 'trailing_comma_in_multiline' => true, 'unary_operator_spaces' => true, 'yoda_style' => false, // PSR standard to apply @@ -82,4 +82,4 @@ foreach ($paths as $path) { $finder = $finder->in($path); } } -return \PhpCsFixer\Config::create()->setRiskyAllowed(true)->setRules($rules)->setFinder($finder); +return (new \PhpCsFixer\Config)->setRiskyAllowed(true)->setRules($rules)->setFinder($finder); diff --git a/composer.lock b/composer.lock index 57bae4a3..21480c42 100644 --- a/composer.lock +++ b/composer.lock @@ -58,16 +58,16 @@ }, { "name": "guzzlehttp/guzzle", - "version": "6.5.5", + "version": "6.5.6", "source": { "type": "git", "url": "https://github.com/guzzle/guzzle.git", - "reference": "9d4290de1cfd701f38099ef7e183b64b4b7b0c5e" + "reference": "f092dd734083473658de3ee4bef093ed77d2689c" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/guzzle/guzzle/zipball/9d4290de1cfd701f38099ef7e183b64b4b7b0c5e", - "reference": "9d4290de1cfd701f38099ef7e183b64b4b7b0c5e", + "url": "https://api.github.com/repos/guzzle/guzzle/zipball/f092dd734083473658de3ee4bef093ed77d2689c", + "reference": "f092dd734083473658de3ee4bef093ed77d2689c", "shasum": "" }, "require": { @@ -104,10 +104,40 @@ "MIT" ], "authors": [ + { + "name": "Graham Campbell", + "email": "hello@gjcampbell.co.uk", + "homepage": "https://github.com/GrahamCampbell" + }, { "name": "Michael Dowling", "email": "mtdowling@gmail.com", "homepage": "https://github.com/mtdowling" + }, + { + "name": "Jeremy Lindblom", + "email": "jeremeamia@gmail.com", + "homepage": "https://github.com/jeremeamia" + }, + { + "name": "George Mponos", + "email": "gmponos@gmail.com", + "homepage": "https://github.com/gmponos" + }, + { + "name": "Tobias Nyholm", + "email": "tobias.nyholm@gmail.com", + "homepage": "https://github.com/Nyholm" + }, + { + "name": "Márk Sági-Kazár", + "email": "mark.sagikazar@gmail.com", + "homepage": "https://github.com/sagikazarmark" + }, + { + "name": "Tobias Schultze", + "email": "webmaster@tubo-world.de", + "homepage": "https://github.com/Tobion" } ], "description": "Guzzle is a PHP HTTP client library", @@ -123,9 +153,23 @@ ], "support": { "issues": "https://github.com/guzzle/guzzle/issues", - "source": "https://github.com/guzzle/guzzle/tree/6.5" + "source": "https://github.com/guzzle/guzzle/tree/6.5.6" }, - "time": "2020-06-16T21:01:06+00:00" + "funding": [ + { + "url": "https://github.com/GrahamCampbell", + "type": "github" + }, + { + "url": "https://github.com/Nyholm", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/guzzlehttp/guzzle", + "type": "tidelift" + } + ], + "time": "2022-05-25T13:19:12+00:00" }, { "name": "guzzlehttp/promises", diff --git a/lib/Context/ExclusionMembers.php b/lib/Context/ExclusionMembers.php index b326f6d9..da8ca87c 100644 --- a/lib/Context/ExclusionMembers.php +++ b/lib/Context/ExclusionMembers.php @@ -243,7 +243,6 @@ trait ExclusionMembers { return $this->act(__FUNCTION__, func_num_args(), $spec); } - public function markedRange($start = null, $end = null) { if ($start === null && $end === null) { $spec = null; diff --git a/lib/Database.php b/lib/Database.php index 107c6dff..87a3c56f 100644 --- a/lib/Database.php +++ b/lib/Database.php @@ -280,7 +280,7 @@ class Database { } /** Renames a user - * + * * This does not have an effect on their numeric ID, but has a cascading effect on many tables */ public function userRename(string $user, string $name): bool { @@ -337,7 +337,7 @@ class Database { } /** Retrieves any metadata associated with a user - * + * * @param string $user The user whose metadata is to be retrieved * @param bool $includeLarge Whether to include values which can be arbitrarily large text */ @@ -855,8 +855,8 @@ class Database { sum(case when \"read\" = 1 and hidden = 0 then 1 else 0 end) as marked from arsse_marks group by subscription ) as mark_stats on mark_stats.subscription = s.id", - ["str", "int"], - [$user, $folder] + ["str", "int"], + [$user, $folder] ); $q->setWhere("s.owner = ?", ["str"], [$user]); $nocase = $this->db->sqlToken("nocase"); @@ -1613,26 +1613,26 @@ class Database { } // ensure any used array-type context options contain at least one member foreach ([ - "articles", + "articles", "editions", "subscriptions", - "folders", - "foldersShallow", - "labels", - "labelNames", - "tags", - "tagNames", - "searchTerms", - "titleTerms", - "authorTerms", + "folders", + "foldersShallow", + "labels", + "labelNames", + "tags", + "tagNames", + "searchTerms", + "titleTerms", + "authorTerms", "annotationTerms", "modifiedRanges", "markedRanges", ] as $m) { - if ($context->$m() && !$context->$m) { - throw new Db\ExceptionInput("tooShort", ['field' => $m, 'action' => $this->caller(), 'min' => 1]); - } + if ($context->$m() && !$context->$m) { + throw new Db\ExceptionInput("tooShort", ['field' => $m, 'action' => $this->caller(), 'min' => 1]); } + } // next compute the context, supplying the query to manipulate directly $this->articleFilter($context, $q); } @@ -1921,8 +1921,8 @@ class Database { touched = 1 where article in (select article from target_articles) - and subscription in (select distinct subscription from target_articles)", - [$subq->getTypes(), "bool"], + and subscription in (select distinct subscription from target_articles)", + [$subq->getTypes(), "bool"], [$subq->getValues(), $data['read']] ); $this->db->prepare($q->getQuery(), $q->getTypes())->run($q->getValues()); @@ -1952,7 +1952,7 @@ class Database { where article in (select article from target_articles) and subscription in (select distinct subscription from target_articles)", - [$subq->getTypes(), $setTypes], + [$subq->getTypes(), $setTypes], [$subq->getValues(), $setValues] ); $this->db->prepare($q->getQuery(), $q->getTypes())->run($q->getValues()); diff --git a/lib/REST/TinyTinyRSS/Search.php b/lib/REST/TinyTinyRSS/Search.php index 3ddd24ef..6fadf6bb 100644 --- a/lib/REST/TinyTinyRSS/Search.php +++ b/lib/REST/TinyTinyRSS/Search.php @@ -319,7 +319,7 @@ class Search { $start = $day."T00:00:00 $tz"; $end = $day."T23:59:59 $tz"; $cc = $neg ? $c->not : $c; - // NOTE: TTRSS treats multiple positive dates as contradictory; we instead treat them as complimentary instead, because it makes more sense + // NOTE: TTRSS treats multiple positive dates as contradictory; we instead treat them as complimentary instead, because it makes more sense return $cc->modifiedRanges(array_merge($cc->modifiedRanges, [[$start, $end]])); } diff --git a/tests/cases/Database/TestDatabase.php b/tests/cases/Database/TestDatabase.php index 7408d7cd..f39425d8 100644 --- a/tests/cases/Database/TestDatabase.php +++ b/tests/cases/Database/TestDatabase.php @@ -12,147 +12,147 @@ use JKingWeb\Arsse\Database; class TestDatabase extends \JKingWeb\Arsse\Test\AbstractTest { protected const COL_DEFS = [ 'arsse_meta' => [ - 'key' => "strict str", + 'key' => "strict str", 'value' => "str", ], 'arsse_users' => [ - 'id' => "strict str", + 'id' => "strict str", 'password' => "str", - 'num' => "strict int", - 'admin' => "strict bool", + 'num' => "strict int", + 'admin' => "strict bool", ], 'arsse_user_meta' => [ - 'owner' => "strict str", - 'key' => "strict str", + 'owner' => "strict str", + 'key' => "strict str", 'modified' => "strict datetime", - 'value' => "str", + 'value' => "str", ], 'arsse_sessions' => [ - 'id' => "strict str", + 'id' => "strict str", 'created' => "strict datetime", 'expires' => "strict datetime", - 'user' => "strict str", + 'user' => "strict str", ], 'arsse_tokens' => [ - 'id' => "strict str", - 'class' => "strict str", - 'user' => "strict str", + 'id' => "strict str", + 'class' => "strict str", + 'user' => "strict str", 'created' => "strict datetime", 'expires' => "datetime", - 'data' => "str", + 'data' => "str", ], 'arsse_feeds' => [ - 'id' => "int", - 'url' => "strict str", - 'title' => "str", - 'source' => "str", - 'updated' => "datetime", - 'modified' => "datetime", + 'id' => "int", + 'url' => "strict str", + 'title' => "str", + 'source' => "str", + 'updated' => "datetime", + 'modified' => "datetime", 'next_fetch' => "datetime", - 'orphaned' => "datetime", - 'etag' => "strict str", - 'err_count' => "strict int", - 'err_msg' => "str", - 'username' => "strict str", - 'password' => "strict str", - 'size' => "strict int", - 'icon' => "int", + 'orphaned' => "datetime", + 'etag' => "strict str", + 'err_count' => "strict int", + 'err_msg' => "str", + 'username' => "strict str", + 'password' => "strict str", + 'size' => "strict int", + 'icon' => "int", ], 'arsse_icons' => [ - 'id' => "int", - 'url' => "strict str", - 'modified' => "datetime", - 'etag' => "strict str", + 'id' => "int", + 'url' => "strict str", + 'modified' => "datetime", + 'etag' => "strict str", 'next_fetch' => "datetime", - 'orphaned' => "datetime", - 'type' => "str", - 'data' => "blob", + 'orphaned' => "datetime", + 'type' => "str", + 'data' => "blob", ], 'arsse_articles' => [ - 'id' => "int", - 'feed' => "strict int", - 'url' => "str", - 'title' => "str", - 'author' => "str", - 'published' => "datetime", - 'edited' => "datetime", - 'modified' => "strict datetime", - 'guid' => "str", - 'url_title_hash' => "strict str", - 'url_content_hash' => "strict str", + 'id' => "int", + 'feed' => "strict int", + 'url' => "str", + 'title' => "str", + 'author' => "str", + 'published' => "datetime", + 'edited' => "datetime", + 'modified' => "strict datetime", + 'guid' => "str", + 'url_title_hash' => "strict str", + 'url_content_hash' => "strict str", 'title_content_hash' => "strict str", - 'content_scraped' => "str", - 'content' => "str", + 'content_scraped' => "str", + 'content' => "str", ], 'arsse_editions' => [ - 'id' => "int", - 'article' => "strict int", + 'id' => "int", + 'article' => "strict int", 'modified' => "strict datetime", ], 'arsse_enclosures' => [ 'article' => "strict int", - 'url' => "str", - 'type' => "str", + 'url' => "str", + 'type' => "str", ], 'arsse_categories' => [ 'article' => "strict int", - 'name' => "str", + 'name' => "str", ], 'arsse_marks' => [ - 'article' => "strict int", + 'article' => "strict int", 'subscription' => "strict int", - 'read' => "strict bool", - 'starred' => "strict bool", - 'modified' => "datetime", - 'note' => "strict str", - 'touched' => "strict bool", - 'hidden' => "strict bool", + 'read' => "strict bool", + 'starred' => "strict bool", + 'modified' => "datetime", + 'note' => "strict str", + 'touched' => "strict bool", + 'hidden' => "strict bool", ], 'arsse_subscriptions' => [ - 'id' => "int", - 'owner' => "strict str", - 'feed' => "strict int", - 'added' => "strict datetime", - 'modified' => "strict datetime", - 'title' => "str", + 'id' => "int", + 'owner' => "strict str", + 'feed' => "strict int", + 'added' => "strict datetime", + 'modified' => "strict datetime", + 'title' => "str", 'order_type' => "strict int", - 'pinned' => "strict bool", - 'folder' => "int", - 'keep_rule' => "str", + 'pinned' => "strict bool", + 'folder' => "int", + 'keep_rule' => "str", 'block_rule' => "str", - 'scrape' => "strict bool", + 'scrape' => "strict bool", ], 'arsse_folders' => [ - 'id' => "int", - 'owner' => "strict str", - 'parent' => "int", - 'name' => "strict str", + 'id' => "int", + 'owner' => "strict str", + 'parent' => "int", + 'name' => "strict str", 'modified' => "strict datetime", ], 'arsse_tags' => [ - 'id' => "int", - 'owner' => "strict str", - 'name' => "strict str", + 'id' => "int", + 'owner' => "strict str", + 'name' => "strict str", 'modified' => "strict datetime", ], 'arsse_tag_members' => [ - 'tag' => "strict int", + 'tag' => "strict int", 'subscription' => "strict int", - 'assigned' => "strict bool", - 'modified' => "strict datetime", + 'assigned' => "strict bool", + 'modified' => "strict datetime", ], 'arsse_labels' => [ - 'id' => "int", - 'owner' => "strict str", - 'name' => "strict str", + 'id' => "int", + 'owner' => "strict str", + 'name' => "strict str", 'modified' => "strict datetime", ], 'arsse_label_members' => [ - 'label' => "strict int", - 'article' => "strict int", + 'label' => "strict int", + 'article' => "strict int", 'subscription' => "strict int", - 'assigned' => "strict bool", - 'modified' => "strict datetime", + 'assigned' => "strict bool", + 'modified' => "strict datetime", ], ]; diff --git a/tests/cases/Misc/TestContext.php b/tests/cases/Misc/TestContext.php index 008d4bf9..f038594c 100644 --- a/tests/cases/Misc/TestContext.php +++ b/tests/cases/Misc/TestContext.php @@ -93,7 +93,7 @@ class TestContext extends \JKingWeb\Arsse\Test\AbstractTest { 'articleRange' => [[1, 100], [1, 100]], 'editionRange' => [[1, 100], [1, 100]], ]; - foreach($tests as $k => $t) { + foreach ($tests as $k => $t) { yield $k => array_merge([$k], $t, [false]); if (method_exists(ExclusionContext::class, $k)) { yield "$k (not)" => array_merge([$k], $t, [true]); @@ -169,7 +169,7 @@ class TestContext extends \JKingWeb\Arsse\Test\AbstractTest { $this->assertTrue(isset($c1[2])); $c1[] = $c2; $act = []; - foreach($c1 as $k => $v) { + foreach ($c1 as $k => $v) { $act[$k] = $v; } $exp = [2 => $c3, $c2]; diff --git a/tests/cases/Misc/TestQuery.php b/tests/cases/Misc/TestQuery.php index e638d76a..7f2df742 100644 --- a/tests/cases/Misc/TestQuery.php +++ b/tests/cases/Misc/TestQuery.php @@ -9,9 +9,9 @@ namespace JKingWeb\Arsse\TestCase\Misc; use JKingWeb\Arsse\Misc\Query; use JKingWeb\Arsse\Misc\QueryFilter; -/** +/** * @covers \JKingWeb\Arsse\Misc\Query - * @covers \JKingWeb\Arsse\Misc\QueryFilter + * @covers \JKingWeb\Arsse\Misc\QueryFilter */ class TestQuery extends \JKingWeb\Arsse\Test\AbstractTest { public function testBasicQuery(): void { @@ -112,4 +112,4 @@ class TestQuery extends \JKingWeb\Arsse\Test\AbstractTest { $this->assertSame(["datetime", "str", "int", "str", "int"], $q->getTypes()); $this->assertSame([1, "ook", 42, "ook", 42], $q->getValues()); } -} \ No newline at end of file +} diff --git a/tests/cases/REST/TinyTinyRSS/TestAPI.php b/tests/cases/REST/TinyTinyRSS/TestAPI.php index 5220a69f..77497a61 100644 --- a/tests/cases/REST/TinyTinyRSS/TestAPI.php +++ b/tests/cases/REST/TinyTinyRSS/TestAPI.php @@ -1524,11 +1524,11 @@ LONG_STRING; 'content' => '

Article content 1

', ], [ - 'id' => "102", - 'guid' => "SHA256:5be8a5a46ecd52ed132191c8d27fb1af6b3d4edc00234c5d9f8f0e10562ed3b7", - 'title' => 'Article title 2', - 'link' => 'http://example.com/2', - 'labels' => [], + 'id' => "102", + 'guid' => "SHA256:5be8a5a46ecd52ed132191c8d27fb1af6b3d4edc00234c5d9f8f0e10562ed3b7", + 'title' => 'Article title 2', + 'link' => 'http://example.com/2', + 'labels' => [], 'unread' => false, 'marked' => false, 'published' => false, diff --git a/vendor-bin/csfixer/composer.json b/vendor-bin/csfixer/composer.json index c49c0755..d0935d26 100644 --- a/vendor-bin/csfixer/composer.json +++ b/vendor-bin/csfixer/composer.json @@ -1,5 +1,5 @@ { "require-dev": { - "friendsofphp/php-cs-fixer": "^2.8" + "friendsofphp/php-cs-fixer": "^3.0" } } diff --git a/vendor-bin/csfixer/composer.lock b/vendor-bin/csfixer/composer.lock index 4c89ef5b..f3e938b5 100644 --- a/vendor-bin/csfixer/composer.lock +++ b/vendor-bin/csfixer/composer.lock @@ -4,35 +4,35 @@ "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", "This file is @generated automatically" ], - "content-hash": "dec0bf343ea4650da0303df8dfb2ab40", + "content-hash": "afbb2dec878898133753608e8bcea93e", "packages": [], "packages-dev": [ { "name": "composer/pcre", - "version": "1.0.1", + "version": "3.0.0", "source": { "type": "git", "url": "https://github.com/composer/pcre.git", - "reference": "67a32d7d6f9f560b726ab25a061b38ff3a80c560" + "reference": "e300eb6c535192decd27a85bc72a9290f0d6b3bd" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/composer/pcre/zipball/67a32d7d6f9f560b726ab25a061b38ff3a80c560", - "reference": "67a32d7d6f9f560b726ab25a061b38ff3a80c560", + "url": "https://api.github.com/repos/composer/pcre/zipball/e300eb6c535192decd27a85bc72a9290f0d6b3bd", + "reference": "e300eb6c535192decd27a85bc72a9290f0d6b3bd", "shasum": "" }, "require": { - "php": "^5.3.2 || ^7.0 || ^8.0" + "php": "^7.4 || ^8.0" }, "require-dev": { "phpstan/phpstan": "^1.3", "phpstan/phpstan-strict-rules": "^1.1", - "symfony/phpunit-bridge": "^4.2 || ^5" + "symfony/phpunit-bridge": "^5" }, "type": "library", "extra": { "branch-alias": { - "dev-main": "1.x-dev" + "dev-main": "3.x-dev" } }, "autoload": { @@ -60,7 +60,7 @@ ], "support": { "issues": "https://github.com/composer/pcre/issues", - "source": "https://github.com/composer/pcre/tree/1.0.1" + "source": "https://github.com/composer/pcre/tree/3.0.0" }, "funding": [ { @@ -76,7 +76,7 @@ "type": "tidelift" } ], - "time": "2022-01-21T20:24:37+00:00" + "time": "2022-02-25T20:21:48+00:00" }, { "name": "composer/semver", @@ -161,27 +161,27 @@ }, { "name": "composer/xdebug-handler", - "version": "2.0.5", + "version": "3.0.3", "source": { "type": "git", "url": "https://github.com/composer/xdebug-handler.git", - "reference": "9e36aeed4616366d2b690bdce11f71e9178c579a" + "reference": "ced299686f41dce890debac69273b47ffe98a40c" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/composer/xdebug-handler/zipball/9e36aeed4616366d2b690bdce11f71e9178c579a", - "reference": "9e36aeed4616366d2b690bdce11f71e9178c579a", + "url": "https://api.github.com/repos/composer/xdebug-handler/zipball/ced299686f41dce890debac69273b47ffe98a40c", + "reference": "ced299686f41dce890debac69273b47ffe98a40c", "shasum": "" }, "require": { - "composer/pcre": "^1", - "php": "^5.3.2 || ^7.0 || ^8.0", + "composer/pcre": "^1 || ^2 || ^3", + "php": "^7.2.5 || ^8.0", "psr/log": "^1 || ^2 || ^3" }, "require-dev": { "phpstan/phpstan": "^1.0", "phpstan/phpstan-strict-rules": "^1.1", - "symfony/phpunit-bridge": "^4.2 || ^5.0 || ^6.0" + "symfony/phpunit-bridge": "^6.0" }, "type": "library", "autoload": { @@ -207,7 +207,7 @@ "support": { "irc": "irc://irc.freenode.org/composer", "issues": "https://github.com/composer/xdebug-handler/issues", - "source": "https://github.com/composer/xdebug-handler/tree/2.0.5" + "source": "https://github.com/composer/xdebug-handler/tree/3.0.3" }, "funding": [ { @@ -223,7 +223,7 @@ "type": "tidelift" } ], - "time": "2022-02-24T20:20:32+00:00" + "time": "2022-02-25T21:32:43+00:00" }, { "name": "doctrine/annotations", @@ -375,85 +375,65 @@ }, { "name": "friendsofphp/php-cs-fixer", - "version": "v2.19.3", + "version": "v3.8.0", "source": { "type": "git", "url": "https://github.com/FriendsOfPHP/PHP-CS-Fixer.git", - "reference": "75ac86f33fab4714ea5a39a396784d83ae3b5ed8" + "reference": "cbad1115aac4b5c3c5540e7210d3c9fba2f81fa3" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/FriendsOfPHP/PHP-CS-Fixer/zipball/75ac86f33fab4714ea5a39a396784d83ae3b5ed8", - "reference": "75ac86f33fab4714ea5a39a396784d83ae3b5ed8", + "url": "https://api.github.com/repos/FriendsOfPHP/PHP-CS-Fixer/zipball/cbad1115aac4b5c3c5540e7210d3c9fba2f81fa3", + "reference": "cbad1115aac4b5c3c5540e7210d3c9fba2f81fa3", "shasum": "" }, "require": { - "composer/semver": "^1.4 || ^2.0 || ^3.0", - "composer/xdebug-handler": "^1.2 || ^2.0", - "doctrine/annotations": "^1.2", + "composer/semver": "^3.2", + "composer/xdebug-handler": "^3.0.3", + "doctrine/annotations": "^1.13", "ext-json": "*", "ext-tokenizer": "*", - "php": "^5.6 || ^7.0 || ^8.0", - "php-cs-fixer/diff": "^1.3", - "symfony/console": "^3.4.43 || ^4.1.6 || ^5.0", - "symfony/event-dispatcher": "^3.0 || ^4.0 || ^5.0", - "symfony/filesystem": "^3.0 || ^4.0 || ^5.0", - "symfony/finder": "^3.0 || ^4.0 || ^5.0", - "symfony/options-resolver": "^3.0 || ^4.0 || ^5.0", - "symfony/polyfill-php70": "^1.0", - "symfony/polyfill-php72": "^1.4", - "symfony/process": "^3.0 || ^4.0 || ^5.0", - "symfony/stopwatch": "^3.0 || ^4.0 || ^5.0" + "php": "^7.4 || ^8.0", + "php-cs-fixer/diff": "^2.0", + "symfony/console": "^5.4 || ^6.0", + "symfony/event-dispatcher": "^5.4 || ^6.0", + "symfony/filesystem": "^5.4 || ^6.0", + "symfony/finder": "^5.4 || ^6.0", + "symfony/options-resolver": "^5.4 || ^6.0", + "symfony/polyfill-mbstring": "^1.23", + "symfony/polyfill-php80": "^1.25", + "symfony/polyfill-php81": "^1.25", + "symfony/process": "^5.4 || ^6.0", + "symfony/stopwatch": "^5.4 || ^6.0" }, "require-dev": { - "justinrainbow/json-schema": "^5.0", - "keradus/cli-executor": "^1.4", - "mikey179/vfsstream": "^1.6", - "php-coveralls/php-coveralls": "^2.4.2", - "php-cs-fixer/accessible-object": "^1.0", + "justinrainbow/json-schema": "^5.2", + "keradus/cli-executor": "^1.5", + "mikey179/vfsstream": "^1.6.10", + "php-coveralls/php-coveralls": "^2.5.2", + "php-cs-fixer/accessible-object": "^1.1", "php-cs-fixer/phpunit-constraint-isidenticalstring": "^1.2", "php-cs-fixer/phpunit-constraint-xmlmatchesxsd": "^1.2.1", - "phpspec/prophecy-phpunit": "^1.1 || ^2.0", - "phpunit/phpunit": "^5.7.27 || ^6.5.14 || ^7.5.20 || ^8.5.13 || ^9.5", + "phpspec/prophecy": "^1.15", + "phpspec/prophecy-phpunit": "^2.0", + "phpunit/phpunit": "^9.5", "phpunitgoodpractices/polyfill": "^1.5", "phpunitgoodpractices/traits": "^1.9.1", - "sanmai/phpunit-legacy-adapter": "^6.4 || ^8.2.1", - "symfony/phpunit-bridge": "^5.2.1", - "symfony/yaml": "^3.0 || ^4.0 || ^5.0" + "symfony/phpunit-bridge": "^6.0", + "symfony/yaml": "^5.4 || ^6.0" }, "suggest": { "ext-dom": "For handling output formats in XML", - "ext-mbstring": "For handling non-UTF8 characters.", - "php-cs-fixer/phpunit-constraint-isidenticalstring": "For IsIdenticalString constraint.", - "php-cs-fixer/phpunit-constraint-xmlmatchesxsd": "For XmlMatchesXsd constraint.", - "symfony/polyfill-mbstring": "When enabling `ext-mbstring` is not possible." + "ext-mbstring": "For handling non-UTF8 characters." }, "bin": [ "php-cs-fixer" ], "type": "application", - "extra": { - "branch-alias": { - "dev-master": "2.19-dev" - } - }, "autoload": { "psr-4": { "PhpCsFixer\\": "src/" - }, - "classmap": [ - "tests/Test/AbstractFixerTestCase.php", - "tests/Test/AbstractIntegrationCaseFactory.php", - "tests/Test/AbstractIntegrationTestCase.php", - "tests/Test/Assert/AssertTokensTrait.php", - "tests/Test/IntegrationCase.php", - "tests/Test/IntegrationCaseFactory.php", - "tests/Test/IntegrationCaseFactoryInterface.php", - "tests/Test/InternalIntegrationCaseFactory.php", - "tests/Test/IsIdenticalConstraint.php", - "tests/Test/TokensWithObservedTransformers.php", - "tests/TestCase.php" - ] + } }, "notification-url": "https://packagist.org/downloads/", "license": [ @@ -472,7 +452,7 @@ "description": "A tool to automatically fix PHP code style", "support": { "issues": "https://github.com/FriendsOfPHP/PHP-CS-Fixer/issues", - "source": "https://github.com/FriendsOfPHP/PHP-CS-Fixer/tree/v2.19.3" + "source": "https://github.com/FriendsOfPHP/PHP-CS-Fixer/tree/v3.8.0" }, "funding": [ { @@ -480,20 +460,20 @@ "type": "github" } ], - "time": "2021-11-15T17:17:55+00:00" + "time": "2022-03-18T17:20:59+00:00" }, { "name": "php-cs-fixer/diff", - "version": "v1.3.1", + "version": "v2.0.2", "source": { "type": "git", "url": "https://github.com/PHP-CS-Fixer/diff.git", - "reference": "dbd31aeb251639ac0b9e7e29405c1441907f5759" + "reference": "29dc0d507e838c4580d018bd8b5cb412474f7ec3" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/PHP-CS-Fixer/diff/zipball/dbd31aeb251639ac0b9e7e29405c1441907f5759", - "reference": "dbd31aeb251639ac0b9e7e29405c1441907f5759", + "url": "https://api.github.com/repos/PHP-CS-Fixer/diff/zipball/29dc0d507e838c4580d018bd8b5cb412474f7ec3", + "reference": "29dc0d507e838c4580d018bd8b5cb412474f7ec3", "shasum": "" }, "require": { @@ -521,21 +501,18 @@ { "name": "Kore Nordmann", "email": "mail@kore-nordmann.de" - }, - { - "name": "SpacePossum" } ], - "description": "sebastian/diff v2 backport support for PHP5.6", + "description": "sebastian/diff v3 backport support for PHP 5.6+", "homepage": "https://github.com/PHP-CS-Fixer", "keywords": [ "diff" ], "support": { "issues": "https://github.com/PHP-CS-Fixer/diff/issues", - "source": "https://github.com/PHP-CS-Fixer/diff/tree/v1.3.1" + "source": "https://github.com/PHP-CS-Fixer/diff/tree/v2.0.2" }, - "time": "2020-10-14T08:39:05+00:00" + "time": "2020-10-14T08:32:19+00:00" }, { "name": "psr/cache", @@ -691,16 +668,16 @@ }, { "name": "psr/log", - "version": "2.0.0", + "version": "3.0.0", "source": { "type": "git", "url": "https://github.com/php-fig/log.git", - "reference": "ef29f6d262798707a9edd554e2b82517ef3a9376" + "reference": "fe5ea303b0887d5caefd3d431c3e61ad47037001" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/php-fig/log/zipball/ef29f6d262798707a9edd554e2b82517ef3a9376", - "reference": "ef29f6d262798707a9edd554e2b82517ef3a9376", + "url": "https://api.github.com/repos/php-fig/log/zipball/fe5ea303b0887d5caefd3d431c3e61ad47037001", + "reference": "fe5ea303b0887d5caefd3d431c3e61ad47037001", "shasum": "" }, "require": { @@ -709,7 +686,7 @@ "type": "library", "extra": { "branch-alias": { - "dev-master": "2.0.x-dev" + "dev-master": "3.x-dev" } }, "autoload": { @@ -735,52 +712,49 @@ "psr-3" ], "support": { - "source": "https://github.com/php-fig/log/tree/2.0.0" + "source": "https://github.com/php-fig/log/tree/3.0.0" }, - "time": "2021-07-14T16:41:46+00:00" + "time": "2021-07-14T16:46:02+00:00" }, { "name": "symfony/console", - "version": "v5.4.7", + "version": "v6.1.0", "source": { "type": "git", "url": "https://github.com/symfony/console.git", - "reference": "900275254f0a1a2afff1ab0e11abd5587a10e1d6" + "reference": "c9646197ef43b0e2ff44af61e7f0571526fd4170" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/console/zipball/900275254f0a1a2afff1ab0e11abd5587a10e1d6", - "reference": "900275254f0a1a2afff1ab0e11abd5587a10e1d6", + "url": "https://api.github.com/repos/symfony/console/zipball/c9646197ef43b0e2ff44af61e7f0571526fd4170", + "reference": "c9646197ef43b0e2ff44af61e7f0571526fd4170", "shasum": "" }, "require": { - "php": ">=7.2.5", + "php": ">=8.1", "symfony/deprecation-contracts": "^2.1|^3", "symfony/polyfill-mbstring": "~1.0", - "symfony/polyfill-php73": "^1.9", - "symfony/polyfill-php80": "^1.16", "symfony/service-contracts": "^1.1|^2|^3", - "symfony/string": "^5.1|^6.0" + "symfony/string": "^5.4|^6.0" }, "conflict": { - "psr/log": ">=3", - "symfony/dependency-injection": "<4.4", - "symfony/dotenv": "<5.1", - "symfony/event-dispatcher": "<4.4", - "symfony/lock": "<4.4", - "symfony/process": "<4.4" + "symfony/dependency-injection": "<5.4", + "symfony/dotenv": "<5.4", + "symfony/event-dispatcher": "<5.4", + "symfony/lock": "<5.4", + "symfony/process": "<5.4" }, "provide": { - "psr/log-implementation": "1.0|2.0" + "psr/log-implementation": "1.0|2.0|3.0" }, "require-dev": { - "psr/log": "^1|^2", - "symfony/config": "^4.4|^5.0|^6.0", - "symfony/dependency-injection": "^4.4|^5.0|^6.0", - "symfony/event-dispatcher": "^4.4|^5.0|^6.0", - "symfony/lock": "^4.4|^5.0|^6.0", - "symfony/process": "^4.4|^5.0|^6.0", - "symfony/var-dumper": "^4.4|^5.0|^6.0" + "psr/log": "^1|^2|^3", + "symfony/config": "^5.4|^6.0", + "symfony/dependency-injection": "^5.4|^6.0", + "symfony/event-dispatcher": "^5.4|^6.0", + "symfony/lock": "^5.4|^6.0", + "symfony/process": "^5.4|^6.0", + "symfony/var-dumper": "^5.4|^6.0" }, "suggest": { "psr/log": "For using the console logger", @@ -820,7 +794,7 @@ "terminal" ], "support": { - "source": "https://github.com/symfony/console/tree/v5.4.7" + "source": "https://github.com/symfony/console/tree/v6.1.0" }, "funding": [ { @@ -836,29 +810,29 @@ "type": "tidelift" } ], - "time": "2022-03-31T17:09:19+00:00" + "time": "2022-05-27T06:34:22+00:00" }, { "name": "symfony/deprecation-contracts", - "version": "v3.0.0", + "version": "v3.1.0", "source": { "type": "git", "url": "https://github.com/symfony/deprecation-contracts.git", - "reference": "c726b64c1ccfe2896cb7df2e1331c357ad1c8ced" + "reference": "07f1b9cc2ffee6aaafcf4b710fbc38ff736bd918" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/deprecation-contracts/zipball/c726b64c1ccfe2896cb7df2e1331c357ad1c8ced", - "reference": "c726b64c1ccfe2896cb7df2e1331c357ad1c8ced", + "url": "https://api.github.com/repos/symfony/deprecation-contracts/zipball/07f1b9cc2ffee6aaafcf4b710fbc38ff736bd918", + "reference": "07f1b9cc2ffee6aaafcf4b710fbc38ff736bd918", "shasum": "" }, "require": { - "php": ">=8.0.2" + "php": ">=8.1" }, "type": "library", "extra": { "branch-alias": { - "dev-main": "3.0-dev" + "dev-main": "3.1-dev" }, "thanks": { "name": "symfony/contracts", @@ -887,7 +861,7 @@ "description": "A generic function and convention to trigger deprecation notices", "homepage": "https://symfony.com", "support": { - "source": "https://github.com/symfony/deprecation-contracts/tree/v3.0.0" + "source": "https://github.com/symfony/deprecation-contracts/tree/v3.1.0" }, "funding": [ { @@ -903,44 +877,42 @@ "type": "tidelift" } ], - "time": "2021-11-01T23:48:49+00:00" + "time": "2022-02-25T11:15:52+00:00" }, { "name": "symfony/event-dispatcher", - "version": "v5.4.3", + "version": "v6.1.0", "source": { "type": "git", "url": "https://github.com/symfony/event-dispatcher.git", - "reference": "dec8a9f58d20df252b9cd89f1c6c1530f747685d" + "reference": "a0449a7ad7daa0f7c0acd508259f80544ab5a347" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/event-dispatcher/zipball/dec8a9f58d20df252b9cd89f1c6c1530f747685d", - "reference": "dec8a9f58d20df252b9cd89f1c6c1530f747685d", + "url": "https://api.github.com/repos/symfony/event-dispatcher/zipball/a0449a7ad7daa0f7c0acd508259f80544ab5a347", + "reference": "a0449a7ad7daa0f7c0acd508259f80544ab5a347", "shasum": "" }, "require": { - "php": ">=7.2.5", - "symfony/deprecation-contracts": "^2.1|^3", - "symfony/event-dispatcher-contracts": "^2|^3", - "symfony/polyfill-php80": "^1.16" + "php": ">=8.1", + "symfony/event-dispatcher-contracts": "^2|^3" }, "conflict": { - "symfony/dependency-injection": "<4.4" + "symfony/dependency-injection": "<5.4" }, "provide": { "psr/event-dispatcher-implementation": "1.0", - "symfony/event-dispatcher-implementation": "2.0" + "symfony/event-dispatcher-implementation": "2.0|3.0" }, "require-dev": { "psr/log": "^1|^2|^3", - "symfony/config": "^4.4|^5.0|^6.0", - "symfony/dependency-injection": "^4.4|^5.0|^6.0", - "symfony/error-handler": "^4.4|^5.0|^6.0", - "symfony/expression-language": "^4.4|^5.0|^6.0", - "symfony/http-foundation": "^4.4|^5.0|^6.0", + "symfony/config": "^5.4|^6.0", + "symfony/dependency-injection": "^5.4|^6.0", + "symfony/error-handler": "^5.4|^6.0", + "symfony/expression-language": "^5.4|^6.0", + "symfony/http-foundation": "^5.4|^6.0", "symfony/service-contracts": "^1.1|^2|^3", - "symfony/stopwatch": "^4.4|^5.0|^6.0" + "symfony/stopwatch": "^5.4|^6.0" }, "suggest": { "symfony/dependency-injection": "", @@ -972,7 +944,7 @@ "description": "Provides tools that allow your application components to communicate with each other by dispatching events and listening to them", "homepage": "https://symfony.com", "support": { - "source": "https://github.com/symfony/event-dispatcher/tree/v5.4.3" + "source": "https://github.com/symfony/event-dispatcher/tree/v6.1.0" }, "funding": [ { @@ -988,24 +960,24 @@ "type": "tidelift" } ], - "time": "2022-01-02T09:53:40+00:00" + "time": "2022-05-05T16:51:07+00:00" }, { "name": "symfony/event-dispatcher-contracts", - "version": "v3.0.0", + "version": "v3.1.0", "source": { "type": "git", "url": "https://github.com/symfony/event-dispatcher-contracts.git", - "reference": "aa5422287b75594b90ee9cd807caf8f0df491385" + "reference": "02ff5eea2f453731cfbc6bc215e456b781480448" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/event-dispatcher-contracts/zipball/aa5422287b75594b90ee9cd807caf8f0df491385", - "reference": "aa5422287b75594b90ee9cd807caf8f0df491385", + "url": "https://api.github.com/repos/symfony/event-dispatcher-contracts/zipball/02ff5eea2f453731cfbc6bc215e456b781480448", + "reference": "02ff5eea2f453731cfbc6bc215e456b781480448", "shasum": "" }, "require": { - "php": ">=8.0.2", + "php": ">=8.1", "psr/event-dispatcher": "^1" }, "suggest": { @@ -1014,7 +986,7 @@ "type": "library", "extra": { "branch-alias": { - "dev-main": "3.0-dev" + "dev-main": "3.1-dev" }, "thanks": { "name": "symfony/contracts", @@ -1051,7 +1023,7 @@ "standards" ], "support": { - "source": "https://github.com/symfony/event-dispatcher-contracts/tree/v3.0.0" + "source": "https://github.com/symfony/event-dispatcher-contracts/tree/v3.1.0" }, "funding": [ { @@ -1067,27 +1039,26 @@ "type": "tidelift" } ], - "time": "2021-07-15T12:33:35+00:00" + "time": "2022-02-25T11:15:52+00:00" }, { "name": "symfony/filesystem", - "version": "v5.4.7", + "version": "v6.1.0", "source": { "type": "git", "url": "https://github.com/symfony/filesystem.git", - "reference": "3a4442138d80c9f7b600fb297534ac718b61d37f" + "reference": "3132d2f43ca799c2aa099f9738d98228c56baa5d" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/filesystem/zipball/3a4442138d80c9f7b600fb297534ac718b61d37f", - "reference": "3a4442138d80c9f7b600fb297534ac718b61d37f", + "url": "https://api.github.com/repos/symfony/filesystem/zipball/3132d2f43ca799c2aa099f9738d98228c56baa5d", + "reference": "3132d2f43ca799c2aa099f9738d98228c56baa5d", "shasum": "" }, "require": { - "php": ">=7.2.5", + "php": ">=8.1", "symfony/polyfill-ctype": "~1.8", - "symfony/polyfill-mbstring": "~1.8", - "symfony/polyfill-php80": "^1.16" + "symfony/polyfill-mbstring": "~1.8" }, "type": "library", "autoload": { @@ -1115,7 +1086,7 @@ "description": "Provides basic utilities for the filesystem", "homepage": "https://symfony.com", "support": { - "source": "https://github.com/symfony/filesystem/tree/v5.4.7" + "source": "https://github.com/symfony/filesystem/tree/v6.1.0" }, "funding": [ { @@ -1131,26 +1102,27 @@ "type": "tidelift" } ], - "time": "2022-04-01T12:33:59+00:00" + "time": "2022-05-21T13:34:40+00:00" }, { "name": "symfony/finder", - "version": "v5.4.3", + "version": "v6.1.0", "source": { "type": "git", "url": "https://github.com/symfony/finder.git", - "reference": "231313534dded84c7ecaa79d14bc5da4ccb69b7d" + "reference": "45b8beb69d6eb3b05a65689ebfd4222326773f8f" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/finder/zipball/231313534dded84c7ecaa79d14bc5da4ccb69b7d", - "reference": "231313534dded84c7ecaa79d14bc5da4ccb69b7d", + "url": "https://api.github.com/repos/symfony/finder/zipball/45b8beb69d6eb3b05a65689ebfd4222326773f8f", + "reference": "45b8beb69d6eb3b05a65689ebfd4222326773f8f", "shasum": "" }, "require": { - "php": ">=7.2.5", - "symfony/deprecation-contracts": "^2.1|^3", - "symfony/polyfill-php80": "^1.16" + "php": ">=8.1" + }, + "require-dev": { + "symfony/filesystem": "^6.0" }, "type": "library", "autoload": { @@ -1178,7 +1150,7 @@ "description": "Finds files and directories via an intuitive fluent interface", "homepage": "https://symfony.com", "support": { - "source": "https://github.com/symfony/finder/tree/v5.4.3" + "source": "https://github.com/symfony/finder/tree/v6.1.0" }, "funding": [ { @@ -1194,27 +1166,25 @@ "type": "tidelift" } ], - "time": "2022-01-26T16:34:36+00:00" + "time": "2022-04-15T08:08:08+00:00" }, { "name": "symfony/options-resolver", - "version": "v5.4.3", + "version": "v6.1.0", "source": { "type": "git", "url": "https://github.com/symfony/options-resolver.git", - "reference": "cc1147cb11af1b43f503ac18f31aa3bec213aba8" + "reference": "a3016f5442e28386ded73c43a32a5b68586dd1c4" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/options-resolver/zipball/cc1147cb11af1b43f503ac18f31aa3bec213aba8", - "reference": "cc1147cb11af1b43f503ac18f31aa3bec213aba8", + "url": "https://api.github.com/repos/symfony/options-resolver/zipball/a3016f5442e28386ded73c43a32a5b68586dd1c4", + "reference": "a3016f5442e28386ded73c43a32a5b68586dd1c4", "shasum": "" }, "require": { - "php": ">=7.2.5", - "symfony/deprecation-contracts": "^2.1|^3", - "symfony/polyfill-php73": "~1.0", - "symfony/polyfill-php80": "^1.16" + "php": ">=8.1", + "symfony/deprecation-contracts": "^2.1|^3" }, "type": "library", "autoload": { @@ -1247,7 +1217,7 @@ "options" ], "support": { - "source": "https://github.com/symfony/options-resolver/tree/v5.4.3" + "source": "https://github.com/symfony/options-resolver/tree/v6.1.0" }, "funding": [ { @@ -1263,7 +1233,7 @@ "type": "tidelift" } ], - "time": "2022-01-02T09:53:40+00:00" + "time": "2022-02-25T11:15:52+00:00" }, { "name": "symfony/polyfill-ctype", @@ -1595,229 +1565,6 @@ ], "time": "2021-11-30T18:21:41+00:00" }, - { - "name": "symfony/polyfill-php70", - "version": "v1.20.0", - "source": { - "type": "git", - "url": "https://github.com/symfony/polyfill-php70.git", - "reference": "5f03a781d984aae42cebd18e7912fa80f02ee644" - }, - "dist": { - "type": "zip", - "url": "https://api.github.com/repos/symfony/polyfill-php70/zipball/5f03a781d984aae42cebd18e7912fa80f02ee644", - "reference": "5f03a781d984aae42cebd18e7912fa80f02ee644", - "shasum": "" - }, - "require": { - "php": ">=7.1" - }, - "type": "metapackage", - "extra": { - "branch-alias": { - "dev-main": "1.20-dev" - }, - "thanks": { - "name": "symfony/polyfill", - "url": "https://github.com/symfony/polyfill" - } - }, - "notification-url": "https://packagist.org/downloads/", - "license": [ - "MIT" - ], - "authors": [ - { - "name": "Nicolas Grekas", - "email": "p@tchwork.com" - }, - { - "name": "Symfony Community", - "homepage": "https://symfony.com/contributors" - } - ], - "description": "Symfony polyfill backporting some PHP 7.0+ features to lower PHP versions", - "homepage": "https://symfony.com", - "keywords": [ - "compatibility", - "polyfill", - "portable", - "shim" - ], - "support": { - "source": "https://github.com/symfony/polyfill-php70/tree/v1.20.0" - }, - "funding": [ - { - "url": "https://symfony.com/sponsor", - "type": "custom" - }, - { - "url": "https://github.com/fabpot", - "type": "github" - }, - { - "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", - "type": "tidelift" - } - ], - "time": "2020-10-23T14:02:19+00:00" - }, - { - "name": "symfony/polyfill-php72", - "version": "v1.25.0", - "source": { - "type": "git", - "url": "https://github.com/symfony/polyfill-php72.git", - "reference": "9a142215a36a3888e30d0a9eeea9766764e96976" - }, - "dist": { - "type": "zip", - "url": "https://api.github.com/repos/symfony/polyfill-php72/zipball/9a142215a36a3888e30d0a9eeea9766764e96976", - "reference": "9a142215a36a3888e30d0a9eeea9766764e96976", - "shasum": "" - }, - "require": { - "php": ">=7.1" - }, - "type": "library", - "extra": { - "branch-alias": { - "dev-main": "1.23-dev" - }, - "thanks": { - "name": "symfony/polyfill", - "url": "https://github.com/symfony/polyfill" - } - }, - "autoload": { - "files": [ - "bootstrap.php" - ], - "psr-4": { - "Symfony\\Polyfill\\Php72\\": "" - } - }, - "notification-url": "https://packagist.org/downloads/", - "license": [ - "MIT" - ], - "authors": [ - { - "name": "Nicolas Grekas", - "email": "p@tchwork.com" - }, - { - "name": "Symfony Community", - "homepage": "https://symfony.com/contributors" - } - ], - "description": "Symfony polyfill backporting some PHP 7.2+ features to lower PHP versions", - "homepage": "https://symfony.com", - "keywords": [ - "compatibility", - "polyfill", - "portable", - "shim" - ], - "support": { - "source": "https://github.com/symfony/polyfill-php72/tree/v1.25.0" - }, - "funding": [ - { - "url": "https://symfony.com/sponsor", - "type": "custom" - }, - { - "url": "https://github.com/fabpot", - "type": "github" - }, - { - "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", - "type": "tidelift" - } - ], - "time": "2021-05-27T09:17:38+00:00" - }, - { - "name": "symfony/polyfill-php73", - "version": "v1.25.0", - "source": { - "type": "git", - "url": "https://github.com/symfony/polyfill-php73.git", - "reference": "cc5db0e22b3cb4111010e48785a97f670b350ca5" - }, - "dist": { - "type": "zip", - "url": "https://api.github.com/repos/symfony/polyfill-php73/zipball/cc5db0e22b3cb4111010e48785a97f670b350ca5", - "reference": "cc5db0e22b3cb4111010e48785a97f670b350ca5", - "shasum": "" - }, - "require": { - "php": ">=7.1" - }, - "type": "library", - "extra": { - "branch-alias": { - "dev-main": "1.23-dev" - }, - "thanks": { - "name": "symfony/polyfill", - "url": "https://github.com/symfony/polyfill" - } - }, - "autoload": { - "files": [ - "bootstrap.php" - ], - "psr-4": { - "Symfony\\Polyfill\\Php73\\": "" - }, - "classmap": [ - "Resources/stubs" - ] - }, - "notification-url": "https://packagist.org/downloads/", - "license": [ - "MIT" - ], - "authors": [ - { - "name": "Nicolas Grekas", - "email": "p@tchwork.com" - }, - { - "name": "Symfony Community", - "homepage": "https://symfony.com/contributors" - } - ], - "description": "Symfony polyfill backporting some PHP 7.3+ features to lower PHP versions", - "homepage": "https://symfony.com", - "keywords": [ - "compatibility", - "polyfill", - "portable", - "shim" - ], - "support": { - "source": "https://github.com/symfony/polyfill-php73/tree/v1.25.0" - }, - "funding": [ - { - "url": "https://symfony.com/sponsor", - "type": "custom" - }, - { - "url": "https://github.com/fabpot", - "type": "github" - }, - { - "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", - "type": "tidelift" - } - ], - "time": "2021-06-05T21:20:04+00:00" - }, { "name": "symfony/polyfill-php80", "version": "v1.25.0", @@ -1902,22 +1649,100 @@ "time": "2022-03-04T08:16:47+00:00" }, { - "name": "symfony/process", - "version": "v5.4.7", + "name": "symfony/polyfill-php81", + "version": "v1.25.0", "source": { "type": "git", - "url": "https://github.com/symfony/process.git", - "reference": "38a44b2517b470a436e1c944bf9b9ba3961137fb" + "url": "https://github.com/symfony/polyfill-php81.git", + "reference": "5de4ba2d41b15f9bd0e19b2ab9674135813ec98f" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/process/zipball/38a44b2517b470a436e1c944bf9b9ba3961137fb", - "reference": "38a44b2517b470a436e1c944bf9b9ba3961137fb", + "url": "https://api.github.com/repos/symfony/polyfill-php81/zipball/5de4ba2d41b15f9bd0e19b2ab9674135813ec98f", + "reference": "5de4ba2d41b15f9bd0e19b2ab9674135813ec98f", "shasum": "" }, "require": { - "php": ">=7.2.5", - "symfony/polyfill-php80": "^1.16" + "php": ">=7.1" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "1.23-dev" + }, + "thanks": { + "name": "symfony/polyfill", + "url": "https://github.com/symfony/polyfill" + } + }, + "autoload": { + "files": [ + "bootstrap.php" + ], + "psr-4": { + "Symfony\\Polyfill\\Php81\\": "" + }, + "classmap": [ + "Resources/stubs" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Symfony polyfill backporting some PHP 8.1+ features to lower PHP versions", + "homepage": "https://symfony.com", + "keywords": [ + "compatibility", + "polyfill", + "portable", + "shim" + ], + "support": { + "source": "https://github.com/symfony/polyfill-php81/tree/v1.25.0" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2021-09-13T13:58:11+00:00" + }, + { + "name": "symfony/process", + "version": "v6.1.0", + "source": { + "type": "git", + "url": "https://github.com/symfony/process.git", + "reference": "318718453c2be58266f1a9e74063d13cb8dd4165" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/process/zipball/318718453c2be58266f1a9e74063d13cb8dd4165", + "reference": "318718453c2be58266f1a9e74063d13cb8dd4165", + "shasum": "" + }, + "require": { + "php": ">=8.1" }, "type": "library", "autoload": { @@ -1945,7 +1770,7 @@ "description": "Executes commands in sub-processes", "homepage": "https://symfony.com", "support": { - "source": "https://github.com/symfony/process/tree/v5.4.7" + "source": "https://github.com/symfony/process/tree/v6.1.0" }, "funding": [ { @@ -1961,24 +1786,24 @@ "type": "tidelift" } ], - "time": "2022-03-18T16:18:52+00:00" + "time": "2022-05-11T12:12:29+00:00" }, { "name": "symfony/service-contracts", - "version": "v3.0.0", + "version": "v3.1.0", "source": { "type": "git", "url": "https://github.com/symfony/service-contracts.git", - "reference": "36715ebf9fb9db73db0cb24263c79077c6fe8603" + "reference": "d66cd8ab656780f62c4215b903a420eb86358957" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/service-contracts/zipball/36715ebf9fb9db73db0cb24263c79077c6fe8603", - "reference": "36715ebf9fb9db73db0cb24263c79077c6fe8603", + "url": "https://api.github.com/repos/symfony/service-contracts/zipball/d66cd8ab656780f62c4215b903a420eb86358957", + "reference": "d66cd8ab656780f62c4215b903a420eb86358957", "shasum": "" }, "require": { - "php": ">=8.0.2", + "php": ">=8.1", "psr/container": "^2.0" }, "conflict": { @@ -1990,7 +1815,7 @@ "type": "library", "extra": { "branch-alias": { - "dev-main": "3.0-dev" + "dev-main": "3.1-dev" }, "thanks": { "name": "symfony/contracts", @@ -2000,7 +1825,10 @@ "autoload": { "psr-4": { "Symfony\\Contracts\\Service\\": "" - } + }, + "exclude-from-classmap": [ + "/Test/" + ] }, "notification-url": "https://packagist.org/downloads/", "license": [ @@ -2027,7 +1855,7 @@ "standards" ], "support": { - "source": "https://github.com/symfony/service-contracts/tree/v3.0.0" + "source": "https://github.com/symfony/service-contracts/tree/v3.1.0" }, "funding": [ { @@ -2043,24 +1871,24 @@ "type": "tidelift" } ], - "time": "2021-11-04T17:53:12+00:00" + "time": "2022-05-07T08:07:09+00:00" }, { "name": "symfony/stopwatch", - "version": "v5.4.5", + "version": "v6.1.0", "source": { "type": "git", "url": "https://github.com/symfony/stopwatch.git", - "reference": "4d04b5c24f3c9a1a168a131f6cbe297155bc0d30" + "reference": "77dedae82ce2a26e2e9b481855473fc3b3e4e54d" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/stopwatch/zipball/4d04b5c24f3c9a1a168a131f6cbe297155bc0d30", - "reference": "4d04b5c24f3c9a1a168a131f6cbe297155bc0d30", + "url": "https://api.github.com/repos/symfony/stopwatch/zipball/77dedae82ce2a26e2e9b481855473fc3b3e4e54d", + "reference": "77dedae82ce2a26e2e9b481855473fc3b3e4e54d", "shasum": "" }, "require": { - "php": ">=7.2.5", + "php": ">=8.1", "symfony/service-contracts": "^1|^2|^3" }, "type": "library", @@ -2089,7 +1917,7 @@ "description": "Provides a way to profile code", "homepage": "https://symfony.com", "support": { - "source": "https://github.com/symfony/stopwatch/tree/v5.4.5" + "source": "https://github.com/symfony/stopwatch/tree/v6.1.0" }, "funding": [ { @@ -2105,24 +1933,24 @@ "type": "tidelift" } ], - "time": "2022-02-18T16:06:09+00:00" + "time": "2022-02-25T11:15:52+00:00" }, { "name": "symfony/string", - "version": "v6.0.3", + "version": "v6.1.0", "source": { "type": "git", "url": "https://github.com/symfony/string.git", - "reference": "522144f0c4c004c80d56fa47e40e17028e2eefc2" + "reference": "d3edc75baf9f1d4f94879764dda2e1ac33499529" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/string/zipball/522144f0c4c004c80d56fa47e40e17028e2eefc2", - "reference": "522144f0c4c004c80d56fa47e40e17028e2eefc2", + "url": "https://api.github.com/repos/symfony/string/zipball/d3edc75baf9f1d4f94879764dda2e1ac33499529", + "reference": "d3edc75baf9f1d4f94879764dda2e1ac33499529", "shasum": "" }, "require": { - "php": ">=8.0.2", + "php": ">=8.1", "symfony/polyfill-ctype": "~1.8", "symfony/polyfill-intl-grapheme": "~1.0", "symfony/polyfill-intl-normalizer": "~1.0", @@ -2174,7 +2002,7 @@ "utf8" ], "support": { - "source": "https://github.com/symfony/string/tree/v6.0.3" + "source": "https://github.com/symfony/string/tree/v6.1.0" }, "funding": [ { @@ -2190,7 +2018,7 @@ "type": "tidelift" } ], - "time": "2022-01-02T09:55:41+00:00" + "time": "2022-04-22T08:18:23+00:00" } ], "aliases": [], @@ -2200,5 +2028,5 @@ "prefer-lowest": false, "platform": [], "platform-dev": [], - "plugin-api-version": "2.2.0" + "plugin-api-version": "2.3.0" } diff --git a/vendor-bin/daux/composer.lock b/vendor-bin/daux/composer.lock index b7d0ac28..85825bfe 100644 --- a/vendor-bin/daux/composer.lock +++ b/vendor-bin/daux/composer.lock @@ -83,16 +83,16 @@ }, { "name": "guzzlehttp/guzzle", - "version": "7.4.2", + "version": "7.4.3", "source": { "type": "git", "url": "https://github.com/guzzle/guzzle.git", - "reference": "ac1ec1cd9b5624694c3a40be801d94137afb12b4" + "reference": "74a8602c6faec9ef74b7a9391ac82c5e65b1cdab" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/guzzle/guzzle/zipball/ac1ec1cd9b5624694c3a40be801d94137afb12b4", - "reference": "ac1ec1cd9b5624694c3a40be801d94137afb12b4", + "url": "https://api.github.com/repos/guzzle/guzzle/zipball/74a8602c6faec9ef74b7a9391ac82c5e65b1cdab", + "reference": "74a8602c6faec9ef74b7a9391ac82c5e65b1cdab", "shasum": "" }, "require": { @@ -187,7 +187,7 @@ ], "support": { "issues": "https://github.com/guzzle/guzzle/issues", - "source": "https://github.com/guzzle/guzzle/tree/7.4.2" + "source": "https://github.com/guzzle/guzzle/tree/7.4.3" }, "funding": [ { @@ -203,7 +203,7 @@ "type": "tidelift" } ], - "time": "2022-03-20T14:16:28+00:00" + "time": "2022-05-25T13:24:33+00:00" }, { "name": "guzzlehttp/promises", @@ -898,16 +898,16 @@ }, { "name": "symfony/console", - "version": "v5.4.7", + "version": "v5.4.9", "source": { "type": "git", "url": "https://github.com/symfony/console.git", - "reference": "900275254f0a1a2afff1ab0e11abd5587a10e1d6" + "reference": "829d5d1bf60b2efeb0887b7436873becc71a45eb" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/console/zipball/900275254f0a1a2afff1ab0e11abd5587a10e1d6", - "reference": "900275254f0a1a2afff1ab0e11abd5587a10e1d6", + "url": "https://api.github.com/repos/symfony/console/zipball/829d5d1bf60b2efeb0887b7436873becc71a45eb", + "reference": "829d5d1bf60b2efeb0887b7436873becc71a45eb", "shasum": "" }, "require": { @@ -977,7 +977,7 @@ "terminal" ], "support": { - "source": "https://github.com/symfony/console/tree/v5.4.7" + "source": "https://github.com/symfony/console/tree/v5.4.9" }, "funding": [ { @@ -993,29 +993,29 @@ "type": "tidelift" } ], - "time": "2022-03-31T17:09:19+00:00" + "time": "2022-05-18T06:17:34+00:00" }, { "name": "symfony/deprecation-contracts", - "version": "v3.0.0", + "version": "v3.1.0", "source": { "type": "git", "url": "https://github.com/symfony/deprecation-contracts.git", - "reference": "c726b64c1ccfe2896cb7df2e1331c357ad1c8ced" + "reference": "07f1b9cc2ffee6aaafcf4b710fbc38ff736bd918" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/deprecation-contracts/zipball/c726b64c1ccfe2896cb7df2e1331c357ad1c8ced", - "reference": "c726b64c1ccfe2896cb7df2e1331c357ad1c8ced", + "url": "https://api.github.com/repos/symfony/deprecation-contracts/zipball/07f1b9cc2ffee6aaafcf4b710fbc38ff736bd918", + "reference": "07f1b9cc2ffee6aaafcf4b710fbc38ff736bd918", "shasum": "" }, "require": { - "php": ">=8.0.2" + "php": ">=8.1" }, "type": "library", "extra": { "branch-alias": { - "dev-main": "3.0-dev" + "dev-main": "3.1-dev" }, "thanks": { "name": "symfony/contracts", @@ -1044,7 +1044,7 @@ "description": "A generic function and convention to trigger deprecation notices", "homepage": "https://symfony.com", "support": { - "source": "https://github.com/symfony/deprecation-contracts/tree/v3.0.0" + "source": "https://github.com/symfony/deprecation-contracts/tree/v3.1.0" }, "funding": [ { @@ -1060,20 +1060,20 @@ "type": "tidelift" } ], - "time": "2021-11-01T23:48:49+00:00" + "time": "2022-02-25T11:15:52+00:00" }, { "name": "symfony/http-foundation", - "version": "v5.4.6", + "version": "v5.4.9", "source": { "type": "git", "url": "https://github.com/symfony/http-foundation.git", - "reference": "34e89bc147633c0f9dd6caaaf56da3b806a21465" + "reference": "6b0d0e4aca38d57605dcd11e2416994b38774522" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/http-foundation/zipball/34e89bc147633c0f9dd6caaaf56da3b806a21465", - "reference": "34e89bc147633c0f9dd6caaaf56da3b806a21465", + "url": "https://api.github.com/repos/symfony/http-foundation/zipball/6b0d0e4aca38d57605dcd11e2416994b38774522", + "reference": "6b0d0e4aca38d57605dcd11e2416994b38774522", "shasum": "" }, "require": { @@ -1117,7 +1117,7 @@ "description": "Defines an object-oriented layer for the HTTP specification", "homepage": "https://symfony.com", "support": { - "source": "https://github.com/symfony/http-foundation/tree/v5.4.6" + "source": "https://github.com/symfony/http-foundation/tree/v5.4.9" }, "funding": [ { @@ -1133,20 +1133,20 @@ "type": "tidelift" } ], - "time": "2022-03-05T21:03:43+00:00" + "time": "2022-05-17T15:07:29+00:00" }, { "name": "symfony/mime", - "version": "v5.4.7", + "version": "v5.4.9", "source": { "type": "git", "url": "https://github.com/symfony/mime.git", - "reference": "92d27a34dea2e199fa9b687e3fff3a7d169b7b1c" + "reference": "2b3802a24e48d0cfccf885173d2aac91e73df92e" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/mime/zipball/92d27a34dea2e199fa9b687e3fff3a7d169b7b1c", - "reference": "92d27a34dea2e199fa9b687e3fff3a7d169b7b1c", + "url": "https://api.github.com/repos/symfony/mime/zipball/2b3802a24e48d0cfccf885173d2aac91e73df92e", + "reference": "2b3802a24e48d0cfccf885173d2aac91e73df92e", "shasum": "" }, "require": { @@ -1200,7 +1200,7 @@ "mime-type" ], "support": { - "source": "https://github.com/symfony/mime/tree/v5.4.7" + "source": "https://github.com/symfony/mime/tree/v5.4.9" }, "funding": [ { @@ -1216,7 +1216,7 @@ "type": "tidelift" } ], - "time": "2022-03-11T16:08:05+00:00" + "time": "2022-05-21T10:24:18+00:00" }, { "name": "symfony/polyfill-ctype", @@ -1962,16 +1962,16 @@ }, { "name": "symfony/process", - "version": "v5.4.7", + "version": "v5.4.8", "source": { "type": "git", "url": "https://github.com/symfony/process.git", - "reference": "38a44b2517b470a436e1c944bf9b9ba3961137fb" + "reference": "597f3fff8e3e91836bb0bd38f5718b56ddbde2f3" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/process/zipball/38a44b2517b470a436e1c944bf9b9ba3961137fb", - "reference": "38a44b2517b470a436e1c944bf9b9ba3961137fb", + "url": "https://api.github.com/repos/symfony/process/zipball/597f3fff8e3e91836bb0bd38f5718b56ddbde2f3", + "reference": "597f3fff8e3e91836bb0bd38f5718b56ddbde2f3", "shasum": "" }, "require": { @@ -2004,7 +2004,7 @@ "description": "Executes commands in sub-processes", "homepage": "https://symfony.com", "support": { - "source": "https://github.com/symfony/process/tree/v5.4.7" + "source": "https://github.com/symfony/process/tree/v5.4.8" }, "funding": [ { @@ -2020,24 +2020,24 @@ "type": "tidelift" } ], - "time": "2022-03-18T16:18:52+00:00" + "time": "2022-04-08T05:07:18+00:00" }, { "name": "symfony/service-contracts", - "version": "v3.0.0", + "version": "v3.1.0", "source": { "type": "git", "url": "https://github.com/symfony/service-contracts.git", - "reference": "36715ebf9fb9db73db0cb24263c79077c6fe8603" + "reference": "d66cd8ab656780f62c4215b903a420eb86358957" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/service-contracts/zipball/36715ebf9fb9db73db0cb24263c79077c6fe8603", - "reference": "36715ebf9fb9db73db0cb24263c79077c6fe8603", + "url": "https://api.github.com/repos/symfony/service-contracts/zipball/d66cd8ab656780f62c4215b903a420eb86358957", + "reference": "d66cd8ab656780f62c4215b903a420eb86358957", "shasum": "" }, "require": { - "php": ">=8.0.2", + "php": ">=8.1", "psr/container": "^2.0" }, "conflict": { @@ -2049,7 +2049,7 @@ "type": "library", "extra": { "branch-alias": { - "dev-main": "3.0-dev" + "dev-main": "3.1-dev" }, "thanks": { "name": "symfony/contracts", @@ -2059,7 +2059,10 @@ "autoload": { "psr-4": { "Symfony\\Contracts\\Service\\": "" - } + }, + "exclude-from-classmap": [ + "/Test/" + ] }, "notification-url": "https://packagist.org/downloads/", "license": [ @@ -2086,7 +2089,7 @@ "standards" ], "support": { - "source": "https://github.com/symfony/service-contracts/tree/v3.0.0" + "source": "https://github.com/symfony/service-contracts/tree/v3.1.0" }, "funding": [ { @@ -2102,24 +2105,24 @@ "type": "tidelift" } ], - "time": "2021-11-04T17:53:12+00:00" + "time": "2022-05-07T08:07:09+00:00" }, { "name": "symfony/string", - "version": "v6.0.3", + "version": "v6.1.0", "source": { "type": "git", "url": "https://github.com/symfony/string.git", - "reference": "522144f0c4c004c80d56fa47e40e17028e2eefc2" + "reference": "d3edc75baf9f1d4f94879764dda2e1ac33499529" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/string/zipball/522144f0c4c004c80d56fa47e40e17028e2eefc2", - "reference": "522144f0c4c004c80d56fa47e40e17028e2eefc2", + "url": "https://api.github.com/repos/symfony/string/zipball/d3edc75baf9f1d4f94879764dda2e1ac33499529", + "reference": "d3edc75baf9f1d4f94879764dda2e1ac33499529", "shasum": "" }, "require": { - "php": ">=8.0.2", + "php": ">=8.1", "symfony/polyfill-ctype": "~1.8", "symfony/polyfill-intl-grapheme": "~1.0", "symfony/polyfill-intl-normalizer": "~1.0", @@ -2171,7 +2174,7 @@ "utf8" ], "support": { - "source": "https://github.com/symfony/string/tree/v6.0.3" + "source": "https://github.com/symfony/string/tree/v6.1.0" }, "funding": [ { @@ -2187,7 +2190,7 @@ "type": "tidelift" } ], - "time": "2022-01-02T09:55:41+00:00" + "time": "2022-04-22T08:18:23+00:00" }, { "name": "symfony/yaml", @@ -2344,5 +2347,5 @@ "prefer-lowest": false, "platform": [], "platform-dev": [], - "plugin-api-version": "2.2.0" + "plugin-api-version": "2.3.0" } diff --git a/vendor-bin/phpstan/composer.lock b/vendor-bin/phpstan/composer.lock index c735af88..defcc9ba 100644 --- a/vendor-bin/phpstan/composer.lock +++ b/vendor-bin/phpstan/composer.lock @@ -79,5 +79,5 @@ "prefer-lowest": false, "platform": [], "platform-dev": [], - "plugin-api-version": "2.2.0" + "plugin-api-version": "2.3.0" } diff --git a/vendor-bin/phpunit/composer.lock b/vendor-bin/phpunit/composer.lock index 4fa10765..838d5501 100644 --- a/vendor-bin/phpunit/composer.lock +++ b/vendor-bin/phpunit/composer.lock @@ -448,16 +448,16 @@ }, { "name": "nikic/php-parser", - "version": "v4.13.2", + "version": "v4.14.0", "source": { "type": "git", "url": "https://github.com/nikic/PHP-Parser.git", - "reference": "210577fe3cf7badcc5814d99455df46564f3c077" + "reference": "34bea19b6e03d8153165d8f30bba4c3be86184c1" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/nikic/PHP-Parser/zipball/210577fe3cf7badcc5814d99455df46564f3c077", - "reference": "210577fe3cf7badcc5814d99455df46564f3c077", + "url": "https://api.github.com/repos/nikic/PHP-Parser/zipball/34bea19b6e03d8153165d8f30bba4c3be86184c1", + "reference": "34bea19b6e03d8153165d8f30bba4c3be86184c1", "shasum": "" }, "require": { @@ -498,9 +498,9 @@ ], "support": { "issues": "https://github.com/nikic/PHP-Parser/issues", - "source": "https://github.com/nikic/PHP-Parser/tree/v4.13.2" + "source": "https://github.com/nikic/PHP-Parser/tree/v4.14.0" }, - "time": "2021-11-30T19:35:32+00:00" + "time": "2022-05-31T20:59:12+00:00" }, { "name": "phar-io/manifest", @@ -2417,21 +2417,20 @@ }, { "name": "webmozart/glob", - "version": "4.4.0", + "version": "4.6.0", "source": { "type": "git", "url": "https://github.com/webmozarts/glob.git", - "reference": "539b5dbc10021d3f9242e7a9e9b6b37843179e83" + "reference": "3c17f7dec3d9d0e87b575026011f2e75a56ed655" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/webmozarts/glob/zipball/539b5dbc10021d3f9242e7a9e9b6b37843179e83", - "reference": "539b5dbc10021d3f9242e7a9e9b6b37843179e83", + "url": "https://api.github.com/repos/webmozarts/glob/zipball/3c17f7dec3d9d0e87b575026011f2e75a56ed655", + "reference": "3c17f7dec3d9d0e87b575026011f2e75a56ed655", "shasum": "" }, "require": { - "php": "^7.3 || ^8.0.0", - "webmozart/path-util": "^2.2" + "php": "^7.3 || ^8.0.0" }, "require-dev": { "phpunit/phpunit": "^9.5", @@ -2461,60 +2460,9 @@ "description": "A PHP implementation of Ant's glob.", "support": { "issues": "https://github.com/webmozarts/glob/issues", - "source": "https://github.com/webmozarts/glob/tree/4.4.0" + "source": "https://github.com/webmozarts/glob/tree/4.6.0" }, - "time": "2021-10-07T16:13:08+00:00" - }, - { - "name": "webmozart/path-util", - "version": "2.3.0", - "source": { - "type": "git", - "url": "https://github.com/webmozart/path-util.git", - "reference": "d939f7edc24c9a1bb9c0dee5cb05d8e859490725" - }, - "dist": { - "type": "zip", - "url": "https://api.github.com/repos/webmozart/path-util/zipball/d939f7edc24c9a1bb9c0dee5cb05d8e859490725", - "reference": "d939f7edc24c9a1bb9c0dee5cb05d8e859490725", - "shasum": "" - }, - "require": { - "php": ">=5.3.3", - "webmozart/assert": "~1.0" - }, - "require-dev": { - "phpunit/phpunit": "^4.6", - "sebastian/version": "^1.0.1" - }, - "type": "library", - "extra": { - "branch-alias": { - "dev-master": "2.3-dev" - } - }, - "autoload": { - "psr-4": { - "Webmozart\\PathUtil\\": "src/" - } - }, - "notification-url": "https://packagist.org/downloads/", - "license": [ - "MIT" - ], - "authors": [ - { - "name": "Bernhard Schussek", - "email": "bschussek@gmail.com" - } - ], - "description": "A robust cross-platform utility for normalizing, comparing and modifying file paths.", - "support": { - "issues": "https://github.com/webmozart/path-util/issues", - "source": "https://github.com/webmozart/path-util/tree/2.3.0" - }, - "abandoned": "symfony/filesystem", - "time": "2015-12-17T08:42:14+00:00" + "time": "2022-05-24T19:45:58+00:00" } ], "aliases": [], @@ -2524,5 +2472,5 @@ "prefer-lowest": false, "platform": [], "platform-dev": [], - "plugin-api-version": "2.2.0" + "plugin-api-version": "2.3.0" } diff --git a/vendor-bin/robo/composer.lock b/vendor-bin/robo/composer.lock index 1849110a..97b62cd4 100644 --- a/vendor-bin/robo/composer.lock +++ b/vendor-bin/robo/composer.lock @@ -90,22 +90,22 @@ }, { "name": "consolidation/annotated-command", - "version": "4.5.3", + "version": "4.5.5", "source": { "type": "git", "url": "https://github.com/consolidation/annotated-command.git", - "reference": "1941a743e63993288e09d0686a4cb7ed47813213" + "reference": "67cea8e8e7656b74da651ea6f49321853996c0fd" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/consolidation/annotated-command/zipball/1941a743e63993288e09d0686a4cb7ed47813213", - "reference": "1941a743e63993288e09d0686a4cb7ed47813213", + "url": "https://api.github.com/repos/consolidation/annotated-command/zipball/67cea8e8e7656b74da651ea6f49321853996c0fd", + "reference": "67cea8e8e7656b74da651ea6f49321853996c0fd", "shasum": "" }, "require": { "consolidation/output-formatters": "^4.1.1", "php": ">=7.1.3", - "psr/log": "^1|^2", + "psr/log": "^1|^2|^3", "symfony/console": "^4.4.8|^5|^6", "symfony/event-dispatcher": "^4.4.8|^5|^6", "symfony/finder": "^4.4.8|^5|^6" @@ -119,7 +119,7 @@ "type": "library", "extra": { "branch-alias": { - "dev-main": "5.x-dev" + "dev-main": "4.x-dev" } }, "autoload": { @@ -140,9 +140,9 @@ "description": "Initialize Symfony Console commands from annotated command class methods.", "support": { "issues": "https://github.com/consolidation/annotated-command/issues", - "source": "https://github.com/consolidation/annotated-command/tree/4.5.3" + "source": "https://github.com/consolidation/annotated-command/tree/4.5.5" }, - "time": "2022-04-02T00:17:53+00:00" + "time": "2022-04-26T16:18:25+00:00" }, { "name": "consolidation/config", @@ -546,26 +546,25 @@ }, { "name": "grasmash/expander", - "version": "2.0.2", + "version": "2.0.3", "source": { "type": "git", "url": "https://github.com/grasmash/expander.git", - "reference": "f4df21d01d1fbda38269cca89e3dbb6ba223da7f" + "reference": "b7cbc1f2fdf9a9c0e253a424c2a4058316b7cb6e" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/grasmash/expander/zipball/f4df21d01d1fbda38269cca89e3dbb6ba223da7f", - "reference": "f4df21d01d1fbda38269cca89e3dbb6ba223da7f", + "url": "https://api.github.com/repos/grasmash/expander/zipball/b7cbc1f2fdf9a9c0e253a424c2a4058316b7cb6e", + "reference": "b7cbc1f2fdf9a9c0e253a424c2a4058316b7cb6e", "shasum": "" }, "require": { "dflydev/dot-access-data": "^3.0.0", - "php": ">=5.6", - "psr/log": "^1 | ^2" + "php": ">=7.1", + "psr/log": "^1 | ^2 | ^3" }, "require-dev": { "greg-1-anderson/composer-test-scenarios": "^1", - "php-coveralls/php-coveralls": "^2.0", "phpunit/phpunit": "^6.0 || ^8.0 || ^9", "squizlabs/php_codesniffer": "^2.7 || ^3.3" }, @@ -592,9 +591,9 @@ "description": "Expands internal property references in PHP arrays file.", "support": { "issues": "https://github.com/grasmash/expander/issues", - "source": "https://github.com/grasmash/expander/tree/2.0.2" + "source": "https://github.com/grasmash/expander/tree/2.0.3" }, - "time": "2022-02-24T03:58:20+00:00" + "time": "2022-04-25T22:17:46+00:00" }, { "name": "league/container", @@ -1071,20 +1070,21 @@ }, { "name": "symfony/console", - "version": "v6.0.7", + "version": "v6.1.0", "source": { "type": "git", "url": "https://github.com/symfony/console.git", - "reference": "70dcf7b2ca2ea08ad6ebcc475f104a024fb5632e" + "reference": "c9646197ef43b0e2ff44af61e7f0571526fd4170" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/console/zipball/70dcf7b2ca2ea08ad6ebcc475f104a024fb5632e", - "reference": "70dcf7b2ca2ea08ad6ebcc475f104a024fb5632e", + "url": "https://api.github.com/repos/symfony/console/zipball/c9646197ef43b0e2ff44af61e7f0571526fd4170", + "reference": "c9646197ef43b0e2ff44af61e7f0571526fd4170", "shasum": "" }, "require": { - "php": ">=8.0.2", + "php": ">=8.1", + "symfony/deprecation-contracts": "^2.1|^3", "symfony/polyfill-mbstring": "~1.0", "symfony/service-contracts": "^1.1|^2|^3", "symfony/string": "^5.4|^6.0" @@ -1146,7 +1146,7 @@ "terminal" ], "support": { - "source": "https://github.com/symfony/console/tree/v6.0.7" + "source": "https://github.com/symfony/console/tree/v6.1.0" }, "funding": [ { @@ -1162,24 +1162,91 @@ "type": "tidelift" } ], - "time": "2022-03-31T17:18:25+00:00" + "time": "2022-05-27T06:34:22+00:00" }, { - "name": "symfony/event-dispatcher", - "version": "v6.0.3", + "name": "symfony/deprecation-contracts", + "version": "v3.1.0", "source": { "type": "git", - "url": "https://github.com/symfony/event-dispatcher.git", - "reference": "6472ea2dd415e925b90ca82be64b8bc6157f3934" + "url": "https://github.com/symfony/deprecation-contracts.git", + "reference": "07f1b9cc2ffee6aaafcf4b710fbc38ff736bd918" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/event-dispatcher/zipball/6472ea2dd415e925b90ca82be64b8bc6157f3934", - "reference": "6472ea2dd415e925b90ca82be64b8bc6157f3934", + "url": "https://api.github.com/repos/symfony/deprecation-contracts/zipball/07f1b9cc2ffee6aaafcf4b710fbc38ff736bd918", + "reference": "07f1b9cc2ffee6aaafcf4b710fbc38ff736bd918", "shasum": "" }, "require": { - "php": ">=8.0.2", + "php": ">=8.1" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "3.1-dev" + }, + "thanks": { + "name": "symfony/contracts", + "url": "https://github.com/symfony/contracts" + } + }, + "autoload": { + "files": [ + "function.php" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "A generic function and convention to trigger deprecation notices", + "homepage": "https://symfony.com", + "support": { + "source": "https://github.com/symfony/deprecation-contracts/tree/v3.1.0" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2022-02-25T11:15:52+00:00" + }, + { + "name": "symfony/event-dispatcher", + "version": "v6.1.0", + "source": { + "type": "git", + "url": "https://github.com/symfony/event-dispatcher.git", + "reference": "a0449a7ad7daa0f7c0acd508259f80544ab5a347" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/event-dispatcher/zipball/a0449a7ad7daa0f7c0acd508259f80544ab5a347", + "reference": "a0449a7ad7daa0f7c0acd508259f80544ab5a347", + "shasum": "" + }, + "require": { + "php": ">=8.1", "symfony/event-dispatcher-contracts": "^2|^3" }, "conflict": { @@ -1229,7 +1296,7 @@ "description": "Provides tools that allow your application components to communicate with each other by dispatching events and listening to them", "homepage": "https://symfony.com", "support": { - "source": "https://github.com/symfony/event-dispatcher/tree/v6.0.3" + "source": "https://github.com/symfony/event-dispatcher/tree/v6.1.0" }, "funding": [ { @@ -1245,24 +1312,24 @@ "type": "tidelift" } ], - "time": "2022-01-02T09:55:41+00:00" + "time": "2022-05-05T16:51:07+00:00" }, { "name": "symfony/event-dispatcher-contracts", - "version": "v3.0.0", + "version": "v3.1.0", "source": { "type": "git", "url": "https://github.com/symfony/event-dispatcher-contracts.git", - "reference": "aa5422287b75594b90ee9cd807caf8f0df491385" + "reference": "02ff5eea2f453731cfbc6bc215e456b781480448" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/event-dispatcher-contracts/zipball/aa5422287b75594b90ee9cd807caf8f0df491385", - "reference": "aa5422287b75594b90ee9cd807caf8f0df491385", + "url": "https://api.github.com/repos/symfony/event-dispatcher-contracts/zipball/02ff5eea2f453731cfbc6bc215e456b781480448", + "reference": "02ff5eea2f453731cfbc6bc215e456b781480448", "shasum": "" }, "require": { - "php": ">=8.0.2", + "php": ">=8.1", "psr/event-dispatcher": "^1" }, "suggest": { @@ -1271,7 +1338,7 @@ "type": "library", "extra": { "branch-alias": { - "dev-main": "3.0-dev" + "dev-main": "3.1-dev" }, "thanks": { "name": "symfony/contracts", @@ -1308,7 +1375,7 @@ "standards" ], "support": { - "source": "https://github.com/symfony/event-dispatcher-contracts/tree/v3.0.0" + "source": "https://github.com/symfony/event-dispatcher-contracts/tree/v3.1.0" }, "funding": [ { @@ -1324,24 +1391,24 @@ "type": "tidelift" } ], - "time": "2021-07-15T12:33:35+00:00" + "time": "2022-02-25T11:15:52+00:00" }, { "name": "symfony/filesystem", - "version": "v6.0.7", + "version": "v6.1.0", "source": { "type": "git", "url": "https://github.com/symfony/filesystem.git", - "reference": "6c9e4c41f2c51dfde3db298594ed9cba55dbf5ff" + "reference": "3132d2f43ca799c2aa099f9738d98228c56baa5d" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/filesystem/zipball/6c9e4c41f2c51dfde3db298594ed9cba55dbf5ff", - "reference": "6c9e4c41f2c51dfde3db298594ed9cba55dbf5ff", + "url": "https://api.github.com/repos/symfony/filesystem/zipball/3132d2f43ca799c2aa099f9738d98228c56baa5d", + "reference": "3132d2f43ca799c2aa099f9738d98228c56baa5d", "shasum": "" }, "require": { - "php": ">=8.0.2", + "php": ">=8.1", "symfony/polyfill-ctype": "~1.8", "symfony/polyfill-mbstring": "~1.8" }, @@ -1371,7 +1438,7 @@ "description": "Provides basic utilities for the filesystem", "homepage": "https://symfony.com", "support": { - "source": "https://github.com/symfony/filesystem/tree/v6.0.7" + "source": "https://github.com/symfony/filesystem/tree/v6.1.0" }, "funding": [ { @@ -1387,24 +1454,27 @@ "type": "tidelift" } ], - "time": "2022-04-01T12:54:51+00:00" + "time": "2022-05-21T13:34:40+00:00" }, { "name": "symfony/finder", - "version": "v6.0.3", + "version": "v6.1.0", "source": { "type": "git", "url": "https://github.com/symfony/finder.git", - "reference": "8661b74dbabc23223f38c9b99d3f8ade71170430" + "reference": "45b8beb69d6eb3b05a65689ebfd4222326773f8f" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/finder/zipball/8661b74dbabc23223f38c9b99d3f8ade71170430", - "reference": "8661b74dbabc23223f38c9b99d3f8ade71170430", + "url": "https://api.github.com/repos/symfony/finder/zipball/45b8beb69d6eb3b05a65689ebfd4222326773f8f", + "reference": "45b8beb69d6eb3b05a65689ebfd4222326773f8f", "shasum": "" }, "require": { - "php": ">=8.0.2" + "php": ">=8.1" + }, + "require-dev": { + "symfony/filesystem": "^6.0" }, "type": "library", "autoload": { @@ -1432,7 +1502,7 @@ "description": "Finds files and directories via an intuitive fluent interface", "homepage": "https://symfony.com", "support": { - "source": "https://github.com/symfony/finder/tree/v6.0.3" + "source": "https://github.com/symfony/finder/tree/v6.1.0" }, "funding": [ { @@ -1448,7 +1518,7 @@ "type": "tidelift" } ], - "time": "2022-01-26T17:23:29+00:00" + "time": "2022-04-15T08:08:08+00:00" }, { "name": "symfony/polyfill-ctype", @@ -1782,20 +1852,20 @@ }, { "name": "symfony/process", - "version": "v6.0.7", + "version": "v6.1.0", "source": { "type": "git", "url": "https://github.com/symfony/process.git", - "reference": "e13f6757e267d687e20ec5b26ccfcbbe511cd8f4" + "reference": "318718453c2be58266f1a9e74063d13cb8dd4165" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/process/zipball/e13f6757e267d687e20ec5b26ccfcbbe511cd8f4", - "reference": "e13f6757e267d687e20ec5b26ccfcbbe511cd8f4", + "url": "https://api.github.com/repos/symfony/process/zipball/318718453c2be58266f1a9e74063d13cb8dd4165", + "reference": "318718453c2be58266f1a9e74063d13cb8dd4165", "shasum": "" }, "require": { - "php": ">=8.0.2" + "php": ">=8.1" }, "type": "library", "autoload": { @@ -1823,7 +1893,7 @@ "description": "Executes commands in sub-processes", "homepage": "https://symfony.com", "support": { - "source": "https://github.com/symfony/process/tree/v6.0.7" + "source": "https://github.com/symfony/process/tree/v6.1.0" }, "funding": [ { @@ -1839,24 +1909,24 @@ "type": "tidelift" } ], - "time": "2022-03-18T16:21:55+00:00" + "time": "2022-05-11T12:12:29+00:00" }, { "name": "symfony/service-contracts", - "version": "v3.0.0", + "version": "v3.1.0", "source": { "type": "git", "url": "https://github.com/symfony/service-contracts.git", - "reference": "36715ebf9fb9db73db0cb24263c79077c6fe8603" + "reference": "d66cd8ab656780f62c4215b903a420eb86358957" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/service-contracts/zipball/36715ebf9fb9db73db0cb24263c79077c6fe8603", - "reference": "36715ebf9fb9db73db0cb24263c79077c6fe8603", + "url": "https://api.github.com/repos/symfony/service-contracts/zipball/d66cd8ab656780f62c4215b903a420eb86358957", + "reference": "d66cd8ab656780f62c4215b903a420eb86358957", "shasum": "" }, "require": { - "php": ">=8.0.2", + "php": ">=8.1", "psr/container": "^2.0" }, "conflict": { @@ -1868,7 +1938,7 @@ "type": "library", "extra": { "branch-alias": { - "dev-main": "3.0-dev" + "dev-main": "3.1-dev" }, "thanks": { "name": "symfony/contracts", @@ -1878,7 +1948,10 @@ "autoload": { "psr-4": { "Symfony\\Contracts\\Service\\": "" - } + }, + "exclude-from-classmap": [ + "/Test/" + ] }, "notification-url": "https://packagist.org/downloads/", "license": [ @@ -1905,7 +1978,7 @@ "standards" ], "support": { - "source": "https://github.com/symfony/service-contracts/tree/v3.0.0" + "source": "https://github.com/symfony/service-contracts/tree/v3.1.0" }, "funding": [ { @@ -1921,24 +1994,24 @@ "type": "tidelift" } ], - "time": "2021-11-04T17:53:12+00:00" + "time": "2022-05-07T08:07:09+00:00" }, { "name": "symfony/string", - "version": "v6.0.3", + "version": "v6.1.0", "source": { "type": "git", "url": "https://github.com/symfony/string.git", - "reference": "522144f0c4c004c80d56fa47e40e17028e2eefc2" + "reference": "d3edc75baf9f1d4f94879764dda2e1ac33499529" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/string/zipball/522144f0c4c004c80d56fa47e40e17028e2eefc2", - "reference": "522144f0c4c004c80d56fa47e40e17028e2eefc2", + "url": "https://api.github.com/repos/symfony/string/zipball/d3edc75baf9f1d4f94879764dda2e1ac33499529", + "reference": "d3edc75baf9f1d4f94879764dda2e1ac33499529", "shasum": "" }, "require": { - "php": ">=8.0.2", + "php": ">=8.1", "symfony/polyfill-ctype": "~1.8", "symfony/polyfill-intl-grapheme": "~1.0", "symfony/polyfill-intl-normalizer": "~1.0", @@ -1990,7 +2063,7 @@ "utf8" ], "support": { - "source": "https://github.com/symfony/string/tree/v6.0.3" + "source": "https://github.com/symfony/string/tree/v6.1.0" }, "funding": [ { @@ -2006,24 +2079,24 @@ "type": "tidelift" } ], - "time": "2022-01-02T09:55:41+00:00" + "time": "2022-04-22T08:18:23+00:00" }, { "name": "symfony/yaml", - "version": "v6.0.3", + "version": "v6.1.0", "source": { "type": "git", "url": "https://github.com/symfony/yaml.git", - "reference": "e77f3ea0b21141d771d4a5655faa54f692b34af5" + "reference": "84ce4f9d2d68f306f971a39d949d8f4b5550dba2" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/symfony/yaml/zipball/e77f3ea0b21141d771d4a5655faa54f692b34af5", - "reference": "e77f3ea0b21141d771d4a5655faa54f692b34af5", + "url": "https://api.github.com/repos/symfony/yaml/zipball/84ce4f9d2d68f306f971a39d949d8f4b5550dba2", + "reference": "84ce4f9d2d68f306f971a39d949d8f4b5550dba2", "shasum": "" }, "require": { - "php": ">=8.0.2", + "php": ">=8.1", "symfony/polyfill-ctype": "^1.8" }, "conflict": { @@ -2064,7 +2137,7 @@ "description": "Loads and dumps YAML files", "homepage": "https://symfony.com", "support": { - "source": "https://github.com/symfony/yaml/tree/v6.0.3" + "source": "https://github.com/symfony/yaml/tree/v6.1.0" }, "funding": [ { @@ -2080,7 +2153,7 @@ "type": "tidelift" } ], - "time": "2022-01-26T17:23:29+00:00" + "time": "2022-04-15T14:25:02+00:00" } ], "aliases": [], @@ -2090,5 +2163,5 @@ "prefer-lowest": false, "platform": [], "platform-dev": [], - "plugin-api-version": "2.2.0" + "plugin-api-version": "2.3.0" } From 9ac615e4a4634d296e51d5020a8f92db527d22ea Mon Sep 17 00:00:00 2001 From: "J. King" Date: Tue, 31 May 2022 23:55:04 -0400 Subject: [PATCH 04/10] Apply more PSR-12 style rules --- .php-cs-fixer.dist.php | 26 +++++++-------------- lib/Db/PostgreSQL/PDOResult.php | 1 - lib/Misc/URL.php | 1 - lib/Misc/ValueInfo.php | 16 ++++++------- lib/REST.php | 4 ++-- lib/Service.php | 6 ++--- tests/cases/Exception/TestException.php | 2 +- tests/cases/Feed/TestException.php | 8 +++---- tests/cases/Misc/TestContext.php | 4 ++-- tests/cases/Misc/TestURL.php | 1 - tests/cases/REST/Miniflux/TestV1.php | 2 +- tests/cases/REST/NextcloudNews/TestV1_2.php | 2 +- tests/cases/REST/TestREST.php | 9 ++++--- tests/cases/REST/TinyTinyRSS/TestAPI.php | 2 +- tests/cases/REST/TinyTinyRSS/TestIcon.php | 2 +- tests/cases/Service/TestService.php | 2 +- tests/lib/AbstractTest.php | 22 ++++++++--------- 17 files changed, 48 insertions(+), 62 deletions(-) diff --git a/.php-cs-fixer.dist.php b/.php-cs-fixer.dist.php index 925d4d19..d1706cb3 100644 --- a/.php-cs-fixer.dist.php +++ b/.php-cs-fixer.dist.php @@ -16,6 +16,8 @@ $paths = [ BASE."tests", ]; $rules = [ + // PSR standard to apply + '@PSR12' => true, // house rules where PSR series is silent 'align_multiline_comment' => ['comment_type' => "phpdocs_only"], 'array_syntax' => ['syntax' => "short"], @@ -51,27 +53,15 @@ $rules = [ 'trailing_comma_in_multiline' => true, 'unary_operator_spaces' => true, 'yoda_style' => false, - // PSR standard to apply - '@PSR2' => true, - // PSR-12 rules; php-cs-fixer does not yet support PSR-12 natively - 'compact_nullable_typehint' => true, - 'declare_equal_normalize' => ['space' => "none"], - 'function_typehint_space' => true, - 'lowercase_cast' => true, - 'lowercase_static_reference' => true, - 'no_alternative_syntax' => true, - 'no_empty_statement' => true, - 'no_leading_import_slash' => true, - 'no_leading_namespace_whitespace' => true, - 'no_whitespace_in_blank_line' => true, - 'return_type_declaration' => ['space_before' => "none"], - 'single_trait_insert_per_statement' => true, - 'short_scalar_cast' => true, - 'visibility_required' => ['elements' => ["const", "property", "method"]], // house exceptions to PSR rules 'braces' => ['position_after_functions_and_oop_constructs' => "same"], 'function_declaration' => ['closure_function_spacing' => "none"], - 'new_with_braces' => false, // no option to specify absence of braces + 'new_with_braces' => [ + 'anonymous_class' => false, + 'named_class' => false, + ], + 'single_blank_line_before_namespace' => false, + 'blank_line_after_opening_tag' => false, ]; $finder = \PhpCsFixer\Finder::create(); diff --git a/lib/Db/PostgreSQL/PDOResult.php b/lib/Db/PostgreSQL/PDOResult.php index 4920776f..77cf88f0 100644 --- a/lib/Db/PostgreSQL/PDOResult.php +++ b/lib/Db/PostgreSQL/PDOResult.php @@ -7,7 +7,6 @@ declare(strict_types=1); namespace JKingWeb\Arsse\Db\PostgreSQL; class PDOResult extends \JKingWeb\Arsse\Db\PDOResult { - // This method exists to transparent handle byte-array results public function valid() { diff --git a/lib/Misc/URL.php b/lib/Misc/URL.php index a1cad01b..6b22b63e 100644 --- a/lib/Misc/URL.php +++ b/lib/Misc/URL.php @@ -10,7 +10,6 @@ namespace JKingWeb\Arsse\Misc; * A collection of functions for manipulating URLs */ class URL { - /** Returns whether a URL is absolute i.e. has a scheme */ public static function absolute(string $url): bool { return (bool) strlen((string) parse_url($url, \PHP_URL_SCHEME)); diff --git a/lib/Misc/ValueInfo.php b/lib/Misc/ValueInfo.php index d03949cd..b3cdb56e 100644 --- a/lib/Misc/ValueInfo.php +++ b/lib/Misc/ValueInfo.php @@ -107,7 +107,7 @@ class ValueInfo { if ($strict && !$drop) { throw new ExceptionType("strictFailure", $type); } - return (!$drop) ? (int) $value->getTimestamp(): null; + return (!$drop) ? (int) $value->getTimestamp() : null; } elseif ($value instanceof \DateInterval) { if ($strict && !$drop) { throw new ExceptionType("strictFailure", $type); @@ -159,7 +159,7 @@ class ValueInfo { if ($strict && !$drop) { throw new ExceptionType("strictFailure", $type); } - return (!$drop) ? (float) $value->getTimestamp(): null; + return (!$drop) ? (float) $value->getTimestamp() : null; } elseif ($value instanceof \DateInterval) { if ($drop) { return null; @@ -203,13 +203,13 @@ class ValueInfo { if ($value->days) { $dateSpec = $value->days."D"; } else { - $dateSpec .= $value->y ? $value->y."Y": ""; - $dateSpec .= $value->m ? $value->m."M": ""; - $dateSpec .= $value->d ? $value->d."D": ""; + $dateSpec .= $value->y ? $value->y."Y" : ""; + $dateSpec .= $value->m ? $value->m."M" : ""; + $dateSpec .= $value->d ? $value->d."D" : ""; } - $timeSpec .= $value->h ? $value->h."H": ""; - $timeSpec .= $value->i ? $value->i."M": ""; - $timeSpec .= $value->s ? $value->s."S": ""; + $timeSpec .= $value->h ? $value->h."H" : ""; + $timeSpec .= $value->i ? $value->i."M" : ""; + $timeSpec .= $value->s ? $value->s."S" : ""; $timeSpec = $timeSpec ? "T".$timeSpec : ""; if (!$dateSpec && !$timeSpec) { return "PT0S"; diff --git a/lib/REST.php b/lib/REST.php index f7818e56..adc56ac5 100644 --- a/lib/REST.php +++ b/lib/REST.php @@ -125,14 +125,14 @@ class REST { $target = substr($url, strlen($api['strip'])); } else { // if the match fails we are not able to handle the request - throw new REST\Exception501(); + throw new REST\Exception501; } // return the API name, stripped URL, and API class name return [$id, $target, $api['class']]; } } // or throw an exception otherwise - throw new REST\Exception501(); + throw new REST\Exception501; } public function authenticateRequest(ServerRequestInterface $req): ServerRequestInterface { diff --git a/lib/Service.php b/lib/Service.php index 7eb31779..79a53467 100644 --- a/lib/Service.php +++ b/lib/Service.php @@ -21,13 +21,13 @@ class Service { public function __construct() { $driver = Arsse::$conf->serviceDriver; - $this->drv = new $driver(); + $this->drv = new $driver; } public function watch(bool $loop = true): \DateTimeInterface { $this->loop = $loop; $this->signalInit(); - $t = new \DateTime(); + $t = new \DateTime; do { $this->checkIn(); static::cleanupPre(); @@ -80,7 +80,7 @@ class Service { // get the checking interval $int = Arsse::$conf->serviceFrequency; // subtract twice the checking interval from the current time to yield the earliest acceptable check-in time - $limit = new \DateTime(); + $limit = new \DateTime; $limit->sub($int); $limit->sub($int); // return whether the check-in time is within the acceptable limit diff --git a/tests/cases/Exception/TestException.php b/tests/cases/Exception/TestException.php index c85bff8d..3ab7d3b6 100644 --- a/tests/cases/Exception/TestException.php +++ b/tests/cases/Exception/TestException.php @@ -31,7 +31,7 @@ class TestException extends \JKingWeb\Arsse\Test\AbstractTest { */ public function testBaseClassWithoutMessage(): void { $this->assertException("unknown"); - throw new Exception(); + throw new Exception; } /** diff --git a/tests/cases/Feed/TestException.php b/tests/cases/Feed/TestException.php index b28d0d1d..b39a32f0 100644 --- a/tests/cases/Feed/TestException.php +++ b/tests/cases/Feed/TestException.php @@ -150,10 +150,10 @@ class TestException extends \JKingWeb\Arsse\Test\AbstractTest { public function providePicoFeedException() { return [ - 'Failed feed discovery' => [new \PicoFeed\Reader\SubscriptionNotFoundException(), "subscriptionNotFound"], - 'Unsupported format' => [new \PicoFeed\Reader\UnsupportedFeedFormatException(), "unsupportedFeedFormat"], - 'Malformed XML' => [new \PicoFeed\Parser\MalformedXmlException(), "malformedXml"], - 'XML entity expansion' => [new \PicoFeed\Parser\XmlEntityException(), "xmlEntity"], + 'Failed feed discovery' => [new \PicoFeed\Reader\SubscriptionNotFoundException, "subscriptionNotFound"], + 'Unsupported format' => [new \PicoFeed\Reader\UnsupportedFeedFormatException, "unsupportedFeedFormat"], + 'Malformed XML' => [new \PicoFeed\Parser\MalformedXmlException, "malformedXml"], + 'XML entity expansion' => [new \PicoFeed\Parser\XmlEntityException, "xmlEntity"], ]; } diff --git a/tests/cases/Misc/TestContext.php b/tests/cases/Misc/TestContext.php index f038594c..97882311 100644 --- a/tests/cases/Misc/TestContext.php +++ b/tests/cases/Misc/TestContext.php @@ -103,7 +103,7 @@ class TestContext extends \JKingWeb\Arsse\Test\AbstractTest { public function testCleanIdArrayValues(): void { $methods = ["articles", "editions", "tags", "labels", "subscriptions"]; - $in = [1, "2", 3.5, 4.0, 4, "ook", 0, -20, true, false, null, new \DateTime(), -1.0]; + $in = [1, "2", 3.5, 4.0, 4, "ook", 0, -20, true, false, null, new \DateTime, -1.0]; $out = [1, 2, 4]; $c = new Context; foreach ($methods as $method) { @@ -113,7 +113,7 @@ class TestContext extends \JKingWeb\Arsse\Test\AbstractTest { public function testCleanFolderIdArrayValues(): void { $methods = ["folders", "foldersShallow"]; - $in = [1, "2", 3.5, 4.0, 4, "ook", 0, -20, true, false, null, new \DateTime(), -1.0]; + $in = [1, "2", 3.5, 4.0, 4, "ook", 0, -20, true, false, null, new \DateTime, -1.0]; $out = [1, 2, 4, 0]; $c = new Context; foreach ($methods as $method) { diff --git a/tests/cases/Misc/TestURL.php b/tests/cases/Misc/TestURL.php index eddc67ed..a5de76ac 100644 --- a/tests/cases/Misc/TestURL.php +++ b/tests/cases/Misc/TestURL.php @@ -10,7 +10,6 @@ use JKingWeb\Arsse\Misc\URL; /** @covers \JKingWeb\Arsse\Misc\URL */ class TestURL extends \JKingWeb\Arsse\Test\AbstractTest { - /** @dataProvider provideNormalizations */ public function testNormalizeAUrl(string $url, string $exp, string $user = null, string $pass = null): void { $this->assertSame($exp, URL::normalize($url, $user, $pass)); diff --git a/tests/cases/REST/Miniflux/TestV1.php b/tests/cases/REST/Miniflux/TestV1.php index 9623fd3d..3bae78e8 100644 --- a/tests/cases/REST/Miniflux/TestV1.php +++ b/tests/cases/REST/Miniflux/TestV1.php @@ -93,7 +93,7 @@ class TestV1 extends \JKingWeb\Arsse\Test\AbstractTest { Arsse::$user->method("propertiesGet")->willReturn(['num' => 42, 'admin' => false, 'root_folder_name' => null, 'tz' => "Asia/Gaza"]); Arsse::$user->method("begin")->willReturn($this->transaction->get()); //initialize a handler - $this->h = new V1(); + $this->h = new V1; } protected function v($value) { diff --git a/tests/cases/REST/NextcloudNews/TestV1_2.php b/tests/cases/REST/NextcloudNews/TestV1_2.php index f58f87d6..34720e31 100644 --- a/tests/cases/REST/NextcloudNews/TestV1_2.php +++ b/tests/cases/REST/NextcloudNews/TestV1_2.php @@ -328,7 +328,7 @@ class TestV1_2 extends \JKingWeb\Arsse\Test\AbstractTest { $this->dbMock = $this->mock(Database::class); $this->dbMock->begin->returns($this->mock(Transaction::class)); //initialize a handler - $this->h = new V1_2(); + $this->h = new V1_2; } protected function v($value) { diff --git a/tests/cases/REST/TestREST.php b/tests/cases/REST/TestREST.php index 0ba6eada..347ab7fe 100644 --- a/tests/cases/REST/TestREST.php +++ b/tests/cases/REST/TestREST.php @@ -22,7 +22,6 @@ use Laminas\Diactoros\Response\EmptyResponse; /** @covers \JKingWeb\Arsse\REST */ class TestREST extends \JKingWeb\Arsse\Test\AbstractTest { - /** @dataProvider provideApiMatchData */ public function testMatchAUrlToAnApi($apiList, string $input, array $exp): void { $r = new REST($apiList); @@ -61,7 +60,7 @@ class TestREST extends \JKingWeb\Arsse\Test\AbstractTest { /** @dataProvider provideAuthenticableRequests */ public function testAuthenticateRequests(array $serverParams, array $expAttr): void { - $r = new REST(); + $r = new REST; // create a mock user manager $this->userMock = $this->mock(User::class); $this->userMock->auth->returns(false); @@ -95,7 +94,7 @@ class TestREST extends \JKingWeb\Arsse\Test\AbstractTest { public function testSendAuthenticationChallenges(): void { self::setConf(); - $r = new REST(); + $r = new REST; $in = new EmptyResponse(401); $exp = $in->withHeader("WWW-Authenticate", 'Basic realm="OOK", charset="UTF-8"'); $act = $r->challenge($in, "OOK"); @@ -107,7 +106,7 @@ class TestREST extends \JKingWeb\Arsse\Test\AbstractTest { /** @dataProvider provideUnnormalizedOrigins */ public function testNormalizeOrigins(string $origin, string $exp, array $ports = null): void { - $r = new REST(); + $r = new REST; $act = $r->corsNormalizeOrigin($origin, $ports); $this->assertSame($exp, $act); } @@ -188,7 +187,7 @@ class TestREST extends \JKingWeb\Arsse\Test\AbstractTest { /** @dataProvider provideCorsHeaders */ public function testAddCorsHeaders(string $reqMethod, array $reqHeaders, array $resHeaders, array $expHeaders): void { - $r = new REST(); + $r = new REST; $req = new Request("", $reqMethod, "php://memory", $reqHeaders); $res = new EmptyResponse(204, $resHeaders); $exp = new EmptyResponse(204, $expHeaders); diff --git a/tests/cases/REST/TinyTinyRSS/TestAPI.php b/tests/cases/REST/TinyTinyRSS/TestAPI.php index 77497a61..9a10d6a2 100644 --- a/tests/cases/REST/TinyTinyRSS/TestAPI.php +++ b/tests/cases/REST/TinyTinyRSS/TestAPI.php @@ -147,7 +147,7 @@ LONG_STRING; 'expires' => "2112-12-21 21:12:00", 'user' => $this->userId, ]); - $this->h = new API(); + $this->h = new API; } protected function req($data, string $method = "POST", string $target = "", string $strData = null, string $user = null): ResponseInterface { diff --git a/tests/cases/REST/TinyTinyRSS/TestIcon.php b/tests/cases/REST/TinyTinyRSS/TestIcon.php index f541f504..f9d4b277 100644 --- a/tests/cases/REST/TinyTinyRSS/TestIcon.php +++ b/tests/cases/REST/TinyTinyRSS/TestIcon.php @@ -25,7 +25,7 @@ class TestIcon extends \JKingWeb\Arsse\Test\AbstractTest { Arsse::$user = $this->mock(User::class)->get(); // create a mock database interface $this->dbMock = $this->mock(Database::class); - $this->h = new Icon(); + $this->h = new Icon; } protected function req(string $target, string $method = "GET", string $user = null): ResponseInterface { diff --git a/tests/cases/Service/TestService.php b/tests/cases/Service/TestService.php index 277df8f1..8ac4e4a1 100644 --- a/tests/cases/Service/TestService.php +++ b/tests/cases/Service/TestService.php @@ -20,7 +20,7 @@ class TestService extends \JKingWeb\Arsse\Test\AbstractTest { self::setConf(); $this->dbMock = $this->mock(Database::class); Arsse::$db = $this->dbMock->get(); - $this->srv = new Service(); + $this->srv = new Service; } public function testCheckIn(): void { diff --git a/tests/lib/AbstractTest.php b/tests/lib/AbstractTest.php index 409a6e95..627e888a 100644 --- a/tests/lib/AbstractTest.php +++ b/tests/lib/AbstractTest.php @@ -54,7 +54,7 @@ abstract class AbstractTest extends \PHPUnit\Framework\TestCase { Arsse::$$prop = null; } if ($loadLang) { - Arsse::$lang = new \JKingWeb\Arsse\Lang(); + Arsse::$lang = new \JKingWeb\Arsse\Lang; } } @@ -62,17 +62,17 @@ abstract class AbstractTest extends \PHPUnit\Framework\TestCase { $defaults = [ 'dbSQLite3File' => ":memory:", 'dbSQLite3Timeout' => 0, - 'dbPostgreSQLHost' => $_ENV['ARSSE_TEST_PGSQL_HOST'] ?: "", - 'dbPostgreSQLPort' => $_ENV['ARSSE_TEST_PGSQL_PORT'] ?: 5432, - 'dbPostgreSQLUser' => $_ENV['ARSSE_TEST_PGSQL_USER'] ?: "arsse_test", - 'dbPostgreSQLPass' => $_ENV['ARSSE_TEST_PGSQL_PASS'] ?: "arsse_test", - 'dbPostgreSQLDb' => $_ENV['ARSSE_TEST_PGSQL_DB'] ?: "arsse_test", + 'dbPostgreSQLHost' => $_ENV['ARSSE_TEST_PGSQL_HOST'] ?: "", + 'dbPostgreSQLPort' => $_ENV['ARSSE_TEST_PGSQL_PORT'] ?: 5432, + 'dbPostgreSQLUser' => $_ENV['ARSSE_TEST_PGSQL_USER'] ?: "arsse_test", + 'dbPostgreSQLPass' => $_ENV['ARSSE_TEST_PGSQL_PASS'] ?: "arsse_test", + 'dbPostgreSQLDb' => $_ENV['ARSSE_TEST_PGSQL_DB'] ?: "arsse_test", 'dbPostgreSQLSchema' => $_ENV['ARSSE_TEST_PGSQL_SCHEMA'] ?: "arsse_test", - 'dbMySQLHost' => $_ENV['ARSSE_TEST_MYSQL_HOST'] ?: "localhost", - 'dbMySQLPort' => $_ENV['ARSSE_TEST_MYSQL_PORT'] ?: 3306, - 'dbMySQLUser' => $_ENV['ARSSE_TEST_MYSQL_USER'] ?: "arsse_test", - 'dbMySQLPass' => $_ENV['ARSSE_TEST_MYSQL_PASS'] ?: "arsse_test", - 'dbMySQLDb' => $_ENV['ARSSE_TEST_MYSQL_DB'] ?: "arsse_test", + 'dbMySQLHost' => $_ENV['ARSSE_TEST_MYSQL_HOST'] ?: "localhost", + 'dbMySQLPort' => $_ENV['ARSSE_TEST_MYSQL_PORT'] ?: 3306, + 'dbMySQLUser' => $_ENV['ARSSE_TEST_MYSQL_USER'] ?: "arsse_test", + 'dbMySQLPass' => $_ENV['ARSSE_TEST_MYSQL_PASS'] ?: "arsse_test", + 'dbMySQLDb' => $_ENV['ARSSE_TEST_MYSQL_DB'] ?: "arsse_test", ]; Arsse::$conf = (($force ? null : Arsse::$conf) ?? (new Conf))->import($defaults)->import($conf); } From 51ce4ae92ba9a130ca3ff39ba2766a40bdeac937 Mon Sep 17 00:00:00 2001 From: "J. King" Date: Fri, 3 Jun 2022 22:10:49 -0400 Subject: [PATCH 05/10] Partial rewrite of database table comparison Contents still need to be sorted for tests to pass. --- tests/cases/Database/TestDatabase.php | 146 ---------------- tests/lib/AbstractTest.php | 230 ++++++++++++++++++++++---- 2 files changed, 195 insertions(+), 181 deletions(-) diff --git a/tests/cases/Database/TestDatabase.php b/tests/cases/Database/TestDatabase.php index f39425d8..00838b3a 100644 --- a/tests/cases/Database/TestDatabase.php +++ b/tests/cases/Database/TestDatabase.php @@ -10,152 +10,6 @@ use JKingWeb\Arsse\Database; /** @covers \JKingWeb\Arsse\Database */ class TestDatabase extends \JKingWeb\Arsse\Test\AbstractTest { - protected const COL_DEFS = [ - 'arsse_meta' => [ - 'key' => "strict str", - 'value' => "str", - ], - 'arsse_users' => [ - 'id' => "strict str", - 'password' => "str", - 'num' => "strict int", - 'admin' => "strict bool", - ], - 'arsse_user_meta' => [ - 'owner' => "strict str", - 'key' => "strict str", - 'modified' => "strict datetime", - 'value' => "str", - ], - 'arsse_sessions' => [ - 'id' => "strict str", - 'created' => "strict datetime", - 'expires' => "strict datetime", - 'user' => "strict str", - ], - 'arsse_tokens' => [ - 'id' => "strict str", - 'class' => "strict str", - 'user' => "strict str", - 'created' => "strict datetime", - 'expires' => "datetime", - 'data' => "str", - ], - 'arsse_feeds' => [ - 'id' => "int", - 'url' => "strict str", - 'title' => "str", - 'source' => "str", - 'updated' => "datetime", - 'modified' => "datetime", - 'next_fetch' => "datetime", - 'orphaned' => "datetime", - 'etag' => "strict str", - 'err_count' => "strict int", - 'err_msg' => "str", - 'username' => "strict str", - 'password' => "strict str", - 'size' => "strict int", - 'icon' => "int", - ], - 'arsse_icons' => [ - 'id' => "int", - 'url' => "strict str", - 'modified' => "datetime", - 'etag' => "strict str", - 'next_fetch' => "datetime", - 'orphaned' => "datetime", - 'type' => "str", - 'data' => "blob", - ], - 'arsse_articles' => [ - 'id' => "int", - 'feed' => "strict int", - 'url' => "str", - 'title' => "str", - 'author' => "str", - 'published' => "datetime", - 'edited' => "datetime", - 'modified' => "strict datetime", - 'guid' => "str", - 'url_title_hash' => "strict str", - 'url_content_hash' => "strict str", - 'title_content_hash' => "strict str", - 'content_scraped' => "str", - 'content' => "str", - ], - 'arsse_editions' => [ - 'id' => "int", - 'article' => "strict int", - 'modified' => "strict datetime", - ], - 'arsse_enclosures' => [ - 'article' => "strict int", - 'url' => "str", - 'type' => "str", - ], - 'arsse_categories' => [ - 'article' => "strict int", - 'name' => "str", - ], - 'arsse_marks' => [ - 'article' => "strict int", - 'subscription' => "strict int", - 'read' => "strict bool", - 'starred' => "strict bool", - 'modified' => "datetime", - 'note' => "strict str", - 'touched' => "strict bool", - 'hidden' => "strict bool", - ], - 'arsse_subscriptions' => [ - 'id' => "int", - 'owner' => "strict str", - 'feed' => "strict int", - 'added' => "strict datetime", - 'modified' => "strict datetime", - 'title' => "str", - 'order_type' => "strict int", - 'pinned' => "strict bool", - 'folder' => "int", - 'keep_rule' => "str", - 'block_rule' => "str", - 'scrape' => "strict bool", - ], - 'arsse_folders' => [ - 'id' => "int", - 'owner' => "strict str", - 'parent' => "int", - 'name' => "strict str", - 'modified' => "strict datetime", - ], - 'arsse_tags' => [ - 'id' => "int", - 'owner' => "strict str", - 'name' => "strict str", - 'modified' => "strict datetime", - ], - 'arsse_tag_members' => [ - 'tag' => "strict int", - 'subscription' => "strict int", - 'assigned' => "strict bool", - 'modified' => "strict datetime", - ], - 'arsse_labels' => [ - 'id' => "int", - 'owner' => "strict str", - 'name' => "strict str", - 'modified' => "strict datetime", - ], - 'arsse_label_members' => [ - 'label' => "strict int", - 'article' => "strict int", - 'subscription' => "strict int", - 'assigned' => "strict bool", - 'modified' => "strict datetime", - ], - ]; - protected $db = null; public function setUp(): void { diff --git a/tests/lib/AbstractTest.php b/tests/lib/AbstractTest.php index 627e888a..f874a107 100644 --- a/tests/lib/AbstractTest.php +++ b/tests/lib/AbstractTest.php @@ -31,6 +31,152 @@ use Laminas\Diactoros\Response\XmlResponse; abstract class AbstractTest extends \PHPUnit\Framework\TestCase { use \DMS\PHPUnitExtensions\ArraySubset\ArraySubsetAsserts; + protected const COL_DEFS = [ + 'arsse_meta' => [ + 'key' => "str", + 'value' => "str", + ], + 'arsse_users' => [ + 'id' => "str", + 'password' => "str", + 'num' => "int", + 'admin' => "bool", + ], + 'arsse_user_meta' => [ + 'owner' => "str", + 'key' => "str", + 'modified' => "datetime", + 'value' => "str", + ], + 'arsse_sessions' => [ + 'id' => "str", + 'created' => "datetime", + 'expires' => "datetime", + 'user' => "str", + ], + 'arsse_tokens' => [ + 'id' => "str", + 'class' => "str", + 'user' => "str", + 'created' => "datetime", + 'expires' => "datetime", + 'data' => "str", + ], + 'arsse_feeds' => [ + 'id' => "int", + 'url' => "str", + 'title' => "str", + 'source' => "str", + 'updated' => "datetime", + 'modified' => "datetime", + 'next_fetch' => "datetime", + 'orphaned' => "datetime", + 'etag' => "str", + 'err_count' => "int", + 'err_msg' => "str", + 'username' => "str", + 'password' => "str", + 'size' => "int", + 'icon' => "int", + ], + 'arsse_icons' => [ + 'id' => "int", + 'url' => "str", + 'modified' => "datetime", + 'etag' => "str", + 'next_fetch' => "datetime", + 'orphaned' => "datetime", + 'type' => "str", + 'data' => "blob", + ], + 'arsse_articles' => [ + 'id' => "int", + 'feed' => "int", + 'url' => "str", + 'title' => "str", + 'author' => "str", + 'published' => "datetime", + 'edited' => "datetime", + 'modified' => "datetime", + 'guid' => "str", + 'url_title_hash' => "str", + 'url_content_hash' => "str", + 'title_content_hash' => "str", + 'content_scraped' => "str", + 'content' => "str", + ], + 'arsse_editions' => [ + 'id' => "int", + 'article' => "int", + 'modified' => "datetime", + ], + 'arsse_enclosures' => [ + 'article' => "int", + 'url' => "str", + 'type' => "str", + ], + 'arsse_categories' => [ + 'article' => "int", + 'name' => "str", + ], + 'arsse_marks' => [ + 'article' => "int", + 'subscription' => "int", + 'read' => "bool", + 'starred' => "bool", + 'modified' => "datetime", + 'note' => "str", + 'touched' => "bool", + 'hidden' => "bool", + ], + 'arsse_subscriptions' => [ + 'id' => "int", + 'owner' => "str", + 'feed' => "int", + 'added' => "datetime", + 'modified' => "datetime", + 'title' => "str", + 'order_type' => "int", + 'pinned' => "bool", + 'folder' => "int", + 'keep_rule' => "str", + 'block_rule' => "str", + 'scrape' => "bool", + ], + 'arsse_folders' => [ + 'id' => "int", + 'owner' => "str", + 'parent' => "int", + 'name' => "str", + 'modified' => "datetime", + ], + 'arsse_tags' => [ + 'id' => "int", + 'owner' => "str", + 'name' => "str", + 'modified' => "datetime", + ], + 'arsse_tag_members' => [ + 'tag' => "int", + 'subscription' => "int", + 'assigned' => "bool", + 'modified' => "datetime", + ], + 'arsse_labels' => [ + 'id' => "int", + 'owner' => "str", + 'name' => "str", + 'modified' => "datetime", + ], + 'arsse_label_members' => [ + 'label' => "int", + 'article' => "int", + 'subscription' => "int", + 'assigned' => "bool", + 'modified' => "datetime", + ], + ]; + protected $objMock; protected $confMock; protected $langMock; @@ -260,47 +406,61 @@ abstract class AbstractTest extends \PHPUnit\Framework\TestCase { return true; } - public function compareExpectations(Driver $drv, array $expected): bool { + public function compareExpectations(Driver $drv, array $expected): void { foreach ($expected as $table => $info) { - $cols = array_map(function($v) { - return '"'.str_replace('"', '""', $v).'"'; - }, array_keys($info['columns'])); - $cols = implode(",", $cols); - $types = $info['columns']; - $data = $drv->prepare("SELECT $cols from $table")->run()->getAll(); - $cols = array_keys($info['columns']); - foreach ($info['rows'] as $index => $row) { - $this->assertCount(sizeof($cols), $row, "The number of columns in array index $index of expectations for table $table does not match its definition"); - $row = array_combine($cols, $row); - foreach ($data as $index => $test) { - foreach ($test as $col => $value) { - switch ($types[$col]) { - case "datetime": - $test[$col] = $this->approximateTime($row[$col], $value); - break; - case "int": - $test[$col] = ValueInfo::normalize($value, ValueInfo::T_INT | ValueInfo::M_DROP | valueInfo::M_NULL); - break; - case "float": - $test[$col] = ValueInfo::normalize($value, ValueInfo::T_FLOAT | ValueInfo::M_DROP | valueInfo::M_NULL); - break; - case "bool": - $test[$col] = (int) ValueInfo::normalize($value, ValueInfo::T_BOOL | ValueInfo::M_DROP | valueInfo::M_NULL); - break; - } + // serialize the rows of the expected output + $types = array_values($info['columns']); + $exp = []; + $dates = []; + foreach ($info['rows'] as $r) { + $row = []; + foreach ($r as $c => $v) { + if ($types[$c] === "datetime") { + $dates[] = $v; } - if ($row === $test) { - $data[$index] = $test; - break; + if ($v === null) { + $row[] = ""; + } elseif (static::$stringOutput || is_string($v)) { + $row[] = '"'.str_replace('"', '""', (string) $v).'"'; + } else { + $row[] = (string) $v; } } - $this->assertContains($row, $data, "Actual Table $table does not contain record at expected array index $index"); - $found = array_search($row, $data, true); - unset($data[$found]); + $exp[] = implode(",", $row); } - $this->assertSame([], $data, "Actual table $table contains extra rows not in expectations"); + // serialize the rows of the actual output + $cols = implode(",", array_map(function($v) { + return '"'.str_replace('"', '""', $v).'"'; + }, array_keys($info['columns']))); + $data = $drv->prepare("SELECT $cols from $table")->run()->getAll(); + $types = $info['columns']; + $act = []; + foreach ($data as $r) { + $row = []; + foreach ($r as $c => $v) { + if ($types[$c] === "datetime") { + if (array_search($v, $dates, true) === false) { + $v = Date::transform(Date::sub("PT1S", $v), "sql"); + if (array_search($v, $dates, true) === false) { + $v = Date::transform(Date::add("PT2S", $v), "sql"); + if (array_search($v, $dates, true) === false) { + $v = Date::transform(Date::sub("PT1S", $v), "sql"); + } + } + } + } + if ($v === null) { + $row[] = ""; + } elseif (is_string($v)) { + $row[] = '"'.str_replace('"', '""', (string) $v).'"'; + } else { + $row[] = (string) $v; + } + } + $act[] = implode(",", $row); + } + $this->assertSame($exp, $act, "Actual table $table does not match expectations"); } - return true; } public function primeExpectations(array $source, array $tableSpecs): array { From 2822864a85bb7a9190742ffa45b70c197c2958fe Mon Sep 17 00:00:00 2001 From: "J. King" Date: Sat, 4 Jun 2022 20:16:22 -0400 Subject: [PATCH 06/10] Fix most test failures MySQL is still being stubborn. It may be a type-conversion issue. --- lib/Db/MySQL/Driver.php | 2 +- tests/lib/AbstractTest.php | 25 ++++++++++++++++++++++++- tests/lib/DatabaseDrivers/MySQL.php | 4 ++-- 3 files changed, 27 insertions(+), 4 deletions(-) diff --git a/lib/Db/MySQL/Driver.php b/lib/Db/MySQL/Driver.php index de3bc3d6..532d05d3 100644 --- a/lib/Db/MySQL/Driver.php +++ b/lib/Db/MySQL/Driver.php @@ -165,7 +165,7 @@ class Driver extends \JKingWeb\Arsse\Db\AbstractDriver { $drv->report_mode = \MYSQLI_REPORT_OFF; $this->db = mysqli_init(); $this->db->options(\MYSQLI_SET_CHARSET_NAME, "utf8mb4"); - $this->db->options(\MYSQLI_OPT_INT_AND_FLOAT_NATIVE, false); + $this->db->options(\MYSQLI_OPT_INT_AND_FLOAT_NATIVE, true); $this->db->options(\MYSQLI_OPT_CONNECT_TIMEOUT, ceil(Arsse::$conf->dbTimeoutConnect)); @$this->db->real_connect($host, $user, $password, $db, $port, $socket); if ($this->db->connect_errno) { diff --git a/tests/lib/AbstractTest.php b/tests/lib/AbstractTest.php index f874a107..d352bd6d 100644 --- a/tests/lib/AbstractTest.php +++ b/tests/lib/AbstractTest.php @@ -415,9 +415,11 @@ abstract class AbstractTest extends \PHPUnit\Framework\TestCase { foreach ($info['rows'] as $r) { $row = []; foreach ($r as $c => $v) { + // store any date values for later comparison if ($types[$c] === "datetime") { $dates[] = $v; } + // serialize to CSV, null being represented by no value if ($v === null) { $row[] = ""; } elseif (static::$stringOutput || is_string($v)) { @@ -435,9 +437,11 @@ abstract class AbstractTest extends \PHPUnit\Framework\TestCase { $data = $drv->prepare("SELECT $cols from $table")->run()->getAll(); $types = $info['columns']; $act = []; + $extra = []; foreach ($data as $r) { $row = []; foreach ($r as $c => $v) { + // account for dates which might be off by one second if ($types[$c] === "datetime") { if (array_search($v, $dates, true) === false) { $v = Date::transform(Date::sub("PT1S", $v), "sql"); @@ -457,8 +461,27 @@ abstract class AbstractTest extends \PHPUnit\Framework\TestCase { $row[] = (string) $v; } } - $act[] = implode(",", $row); + $row = implode(",", $row); + // now search for the actual output row in the expected output + $found = array_keys($exp, $row, true); + foreach ($found as $k) { + if(!isset($act[$k])) { + $act[$k] = $row; + // skip to the next row + continue 2; + } + } + // if the row was not found, add it to a buffer which will be added to the actual output once all found rows are processed + $extra[] = $row; } + // add any unfound rows to the end of the actual array + $base = sizeof($exp) + 1; + foreach ($extra as $k => $v) { + $act[$base + $k] = $v; + } + // sort the actual output by keys + ksort($act); + // finally perform the comparison to be shown to the tester $this->assertSame($exp, $act, "Actual table $table does not match expectations"); } } diff --git a/tests/lib/DatabaseDrivers/MySQL.php b/tests/lib/DatabaseDrivers/MySQL.php index 2b6c0164..ab772be0 100644 --- a/tests/lib/DatabaseDrivers/MySQL.php +++ b/tests/lib/DatabaseDrivers/MySQL.php @@ -16,7 +16,7 @@ trait MySQL { protected static $dbResultClass = \JKingWeb\Arsse\Db\MySQL\Result::class; protected static $dbStatementClass = \JKingWeb\Arsse\Db\MySQL\Statement::class; protected static $dbDriverClass = \JKingWeb\Arsse\Db\MySQL\Driver::class; - protected static $stringOutput = true; + protected static $stringOutput = false; public static function dbInterface() { if (!class_exists("mysqli")) { @@ -25,7 +25,7 @@ trait MySQL { $drv = new \mysqli_driver; $drv->report_mode = \MYSQLI_REPORT_OFF; $d = mysqli_init(); - $d->options(\MYSQLI_OPT_INT_AND_FLOAT_NATIVE, false); + $d->options(\MYSQLI_OPT_INT_AND_FLOAT_NATIVE, true); $d->options(\MYSQLI_SET_CHARSET_NAME, "utf8mb4"); @$d->real_connect(Arsse::$conf->dbMySQLHost, Arsse::$conf->dbMySQLUser, Arsse::$conf->dbMySQLPass, Arsse::$conf->dbMySQLDb, Arsse::$conf->dbMySQLPort); if ($d->connect_errno) { From c40f39e34e8b5d1405e9030c018ef1a9da750b8a Mon Sep 17 00:00:00 2001 From: "J. King" Date: Sat, 4 Jun 2022 22:07:21 -0400 Subject: [PATCH 07/10] Work around MySQL absurdities --- lib/Database.php | 10 ++++++---- lib/Db/MySQL/Driver.php | 2 ++ tests/cases/Db/BaseDriver.php | 2 ++ tests/lib/AbstractTest.php | 4 ---- 4 files changed, 10 insertions(+), 8 deletions(-) diff --git a/lib/Database.php b/lib/Database.php index 87a3c56f..68cc43df 100644 --- a/lib/Database.php +++ b/lib/Database.php @@ -813,7 +813,8 @@ class Database { public function subscriptionList(string $user, $folder = null, bool $recursive = true, int $id = null): Db\Result { // validate inputs $folder = $this->folderValidateId($user, $folder)['id']; - // create a complex query + // compile the query + $integerType = $this->db->sqlToken("integer"); $q = new Query( "WITH RECURSIVE topmost(f_id, top) as ( @@ -834,7 +835,7 @@ class Database { i.url as icon_url, folder, t.top as top_folder, d.name as folder_name, dt.name as top_folder_name, coalesce(s.title, f.title) as title, - coalesce((articles - hidden - marked), coalesce(articles,0)) as unread + cast(coalesce((articles - hidden - marked), coalesce(articles,0)) as $integerType) as unread -- this cast is required for MySQL for unclear reasons from arsse_subscriptions as s join arsse_feeds as f on f.id = s.feed left join topmost as t on t.f_id = s.folder @@ -2209,13 +2210,14 @@ class Database { * @param boolean $includeEmpty Whether to include (true) or supress (false) labels which have no articles assigned to them */ public function labelList(string $user, bool $includeEmpty = true): Db\Result { + $integerType = $this->db->sqlToken("integer"); return $this->db->prepareArray( "SELECT * FROM ( SELECT id, name, - coalesce(articles - coalesce(hidden, 0), 0) as articles, - coalesce(marked, 0) as \"read\" + cast(coalesce(articles - coalesce(hidden, 0), 0) as $integerType) as articles, -- this cast is required for MySQL for unclear reasons + cast(coalesce(marked, 0) as $integerType) as \"read\" -- this cast is required for MySQL for unclear reasons from arsse_labels left join ( SELECT label, sum(assigned) as articles from arsse_label_members group by label diff --git a/lib/Db/MySQL/Driver.php b/lib/Db/MySQL/Driver.php index 532d05d3..10bc766d 100644 --- a/lib/Db/MySQL/Driver.php +++ b/lib/Db/MySQL/Driver.php @@ -83,6 +83,8 @@ class Driver extends \JKingWeb\Arsse\Db\AbstractDriver { return '"utf8mb4_unicode_ci"'; case "asc": return ""; + case "integer": + return "signed integer"; default: return $token; } diff --git a/tests/cases/Db/BaseDriver.php b/tests/cases/Db/BaseDriver.php index 0999d3a7..fe7f344c 100644 --- a/tests/cases/Db/BaseDriver.php +++ b/tests/cases/Db/BaseDriver.php @@ -384,6 +384,7 @@ abstract class BaseDriver extends \JKingWeb\Arsse\Test\AbstractTest { $greatest = $this->drv->sqlToken("GrEatESt"); $nocase = $this->drv->sqlToken("noCASE"); $like = $this->drv->sqlToken("liKe"); + $integer = $this->drv->sqlToken("InTEGer"); $asc = $this->drv->sqlToken("asc"); $desc = $this->drv->sqlToken("desc"); $least = $this->drv->sqlToken("leASt"); @@ -394,6 +395,7 @@ abstract class BaseDriver extends \JKingWeb\Arsse\Test\AbstractTest { $this->assertSame("Z", $this->drv->query("SELECT $greatest('Z', 'A')")->getValue()); $this->assertSame("Z", $this->drv->query("SELECT 'Z' collate $nocase")->getValue()); $this->assertSame("Z", $this->drv->query("SELECT 'Z' where 'Z' $like 'z'")->getValue()); + $this->assertEquals(1, $this->drv->query("SELECT CAST((1=1) as $integer)")->getValue()); $this->assertEquals([null, 1, 2], array_column($this->drv->query("SELECT 1 as t union select null as t union select 2 as t order by t $asc")->getAll(), "t")); $this->assertEquals([2, 1, null], array_column($this->drv->query("SELECT 1 as t union select null as t union select 2 as t order by t $desc")->getAll(), "t")); } diff --git a/tests/lib/AbstractTest.php b/tests/lib/AbstractTest.php index d352bd6d..a22913df 100644 --- a/tests/lib/AbstractTest.php +++ b/tests/lib/AbstractTest.php @@ -518,10 +518,6 @@ abstract class AbstractTest extends \PHPUnit\Framework\TestCase { // stringify our expectations if necessary if (static::$stringOutput ?? false) { $expected = $this->stringify($expected); - // MySQL is extra-special and mixes strings and integers, so we cast the data, too - if ((static::$implementation ?? "") === "MySQL") { - $data = $this->stringify($data); - } } $this->assertCount(sizeof($expected), $data, "Number of result rows (".sizeof($data).") differs from number of expected rows (".sizeof($expected).")"); if (sizeof($expected)) { From 0f2da754c5b02d46f2277216edb89186f40cbe52 Mon Sep 17 00:00:00 2001 From: "J. King" Date: Sat, 4 Jun 2022 23:02:24 -0400 Subject: [PATCH 08/10] Fix remaining test problems --- lib/Db/AbstractDriver.php | 4 ++++ lib/Db/Driver.php | 4 ++++ lib/Db/MySQL/Driver.php | 2 +- lib/Db/MySQL/PDODriver.php | 2 +- lib/Db/PostgreSQL/Driver.php | 4 ++++ lib/Db/PostgreSQL/PDODriver.php | 4 ++++ lib/Db/SQLite3/PDODriver.php | 4 ++++ tests/lib/AbstractTest.php | 2 +- tests/lib/DatabaseDrivers/MySQLPDO.php | 4 ++-- 9 files changed, 25 insertions(+), 5 deletions(-) diff --git a/lib/Db/AbstractDriver.php b/lib/Db/AbstractDriver.php index 2a6a973e..66e65f27 100644 --- a/lib/Db/AbstractDriver.php +++ b/lib/Db/AbstractDriver.php @@ -204,4 +204,8 @@ abstract class AbstractDriver implements Driver { public function prepare(string $query, ...$paramType): Statement { return $this->prepareArray($query, $paramType); } + + public function stringOutput(): bool { + return false; + } } diff --git a/lib/Db/Driver.php b/lib/Db/Driver.php index 09f16e78..6f1e311f 100644 --- a/lib/Db/Driver.php +++ b/lib/Db/Driver.php @@ -72,6 +72,7 @@ interface Driver { * The tokens the implementation must understand are: * * - "greatest": the GREATEST function implemented by PostgreSQL and MySQL + * - "least": the LEAST function implemented by PostgreSQL and MySQL * - "nocase": the name of a general-purpose case-insensitive collation sequence * - "like": the case-insensitive LIKE operator * - "integer": the integer type to use for explicit casts @@ -91,4 +92,7 @@ interface Driver { * This should be restricted to quick maintenance; in SQLite terms it might include ANALYZE, but not VACUUM */ public function maintenance(): bool; + + /** Reports whether the implementation will coerce integer and float values to text (string) */ + public function stringOutput(): bool; } diff --git a/lib/Db/MySQL/Driver.php b/lib/Db/MySQL/Driver.php index 10bc766d..fa7a9752 100644 --- a/lib/Db/MySQL/Driver.php +++ b/lib/Db/MySQL/Driver.php @@ -12,7 +12,7 @@ use JKingWeb\Arsse\Db\Exception; class Driver extends \JKingWeb\Arsse\Db\AbstractDriver { use ExceptionBuilder; - protected const SQL_MODE = "ANSI_QUOTES,HIGH_NOT_PRECEDENCE,NO_BACKSLASH_ESCAPES,NO_ENGINE_SUBSTITUTION,PIPES_AS_CONCAT,STRICT_ALL_TABLES"; + protected const SQL_MODE = "ANSI_QUOTES,HIGH_NOT_PRECEDENCE,NO_BACKSLASH_ESCAPES,NO_ENGINE_SUBSTITUTION,PIPES_AS_CONCAT,STRICT_ALL_TABLES,NO_UNSIGNED_SUBTRACTION"; protected const TRANSACTIONAL_LOCKS = false; /** @var \mysqli */ diff --git a/lib/Db/MySQL/PDODriver.php b/lib/Db/MySQL/PDODriver.php index 18cda0be..590b73c4 100644 --- a/lib/Db/MySQL/PDODriver.php +++ b/lib/Db/MySQL/PDODriver.php @@ -29,7 +29,7 @@ class PDODriver extends Driver { try { $this->db = new \PDO($dsn, $user, $password, [ \PDO::ATTR_ERRMODE => \PDO::ERRMODE_EXCEPTION, - \PDO::ATTR_STRINGIFY_FETCHES => true, + \PDO::ATTR_STRINGIFY_FETCHES => false, ]); } catch (\PDOException $e) { $msg = $e->getMessage(); diff --git a/lib/Db/PostgreSQL/Driver.php b/lib/Db/PostgreSQL/Driver.php index f78855cf..e290967a 100644 --- a/lib/Db/PostgreSQL/Driver.php +++ b/lib/Db/PostgreSQL/Driver.php @@ -232,4 +232,8 @@ class Driver extends \JKingWeb\Arsse\Db\AbstractDriver { $this->exec("ANALYZE"); return true; } + + public function stringOutput(): bool { + return true; + } } diff --git a/lib/Db/PostgreSQL/PDODriver.php b/lib/Db/PostgreSQL/PDODriver.php index 93daf667..71d18eb9 100644 --- a/lib/Db/PostgreSQL/PDODriver.php +++ b/lib/Db/PostgreSQL/PDODriver.php @@ -60,4 +60,8 @@ class PDODriver extends Driver { public function prepareArray(string $query, array $paramTypes): \JKingWeb\Arsse\Db\Statement { return new PDOStatement($this->db, $query, $paramTypes); } + + public function stringOutput(): bool { + return false; + } } diff --git a/lib/Db/SQLite3/PDODriver.php b/lib/Db/SQLite3/PDODriver.php index 86ab1cdc..87ecfc30 100644 --- a/lib/Db/SQLite3/PDODriver.php +++ b/lib/Db/SQLite3/PDODriver.php @@ -81,4 +81,8 @@ class PDODriver extends AbstractPDODriver { } } } + + public function stringOutput(): bool { + return true; + } } diff --git a/tests/lib/AbstractTest.php b/tests/lib/AbstractTest.php index a22913df..f498a543 100644 --- a/tests/lib/AbstractTest.php +++ b/tests/lib/AbstractTest.php @@ -422,7 +422,7 @@ abstract class AbstractTest extends \PHPUnit\Framework\TestCase { // serialize to CSV, null being represented by no value if ($v === null) { $row[] = ""; - } elseif (static::$stringOutput || is_string($v)) { + } elseif ($drv->stringOutput() || is_string($v)) { $row[] = '"'.str_replace('"', '""', (string) $v).'"'; } else { $row[] = (string) $v; diff --git a/tests/lib/DatabaseDrivers/MySQLPDO.php b/tests/lib/DatabaseDrivers/MySQLPDO.php index 46248684..ba01dd4e 100644 --- a/tests/lib/DatabaseDrivers/MySQLPDO.php +++ b/tests/lib/DatabaseDrivers/MySQLPDO.php @@ -16,7 +16,7 @@ trait MySQLPDO { protected static $dbResultClass = \JKingWeb\Arsse\Db\PDOResult::class; protected static $dbStatementClass = \JKingWeb\Arsse\Db\MySQL\PDOStatement::class; protected static $dbDriverClass = \JKingWeb\Arsse\Db\MySQL\PDODriver::class; - protected static $stringOutput = true; + protected static $stringOutput = false; public static function dbInterface() { try { @@ -33,7 +33,7 @@ trait MySQLPDO { $dsn = "mysql:".implode(";", $dsn); $d = new \PDO($dsn, Arsse::$conf->dbMySQLUser, Arsse::$conf->dbMySQLPass, [ \PDO::ATTR_ERRMODE => \PDO::ERRMODE_EXCEPTION, - \PDO::ATTR_STRINGIFY_FETCHES => true, + \PDO::ATTR_STRINGIFY_FETCHES => false, \PDO::MYSQL_ATTR_MULTI_STATEMENTS => false, ]); foreach (\JKingWeb\Arsse\Db\MySQL\PDODriver::makeSetupQueries() as $q) { From 07bac4ead385048ed0409bdf96aa07947b300d71 Mon Sep 17 00:00:00 2001 From: "J. King" Date: Sun, 5 Jun 2022 17:41:32 -0400 Subject: [PATCH 09/10] Remove colukmn types from test data For some reason this breaks PostgreSQL tests; this will have to be fixed --- tests/cases/Database/SeriesArticle.php | 124 ++++-------------- tests/cases/Database/SeriesCleanup.php | 64 ++------- tests/cases/Database/SeriesFeed.php | 75 ++--------- tests/cases/Database/SeriesFolder.php | 26 +--- tests/cases/Database/SeriesIcon.php | 31 +---- tests/cases/Database/SeriesLabel.php | 74 ++--------- tests/cases/Database/SeriesMeta.php | 9 +- tests/cases/Database/SeriesSession.php | 13 +- tests/cases/Database/SeriesSubscription.php | 82 ++---------- tests/cases/Database/SeriesTag.php | 31 +---- tests/cases/Database/SeriesToken.php | 14 +- tests/cases/Database/SeriesUser.php | 13 +- tests/cases/ImportExport/TestImportExport.php | 39 +----- tests/lib/AbstractTest.php | 57 +++++--- 14 files changed, 132 insertions(+), 520 deletions(-) diff --git a/tests/cases/Database/SeriesArticle.php b/tests/cases/Database/SeriesArticle.php index efd78e18..4dea4baf 100644 --- a/tests/cases/Database/SeriesArticle.php +++ b/tests/cases/Database/SeriesArticle.php @@ -18,11 +18,7 @@ trait SeriesArticle { protected function setUpSeriesArticle(): void { $this->data = [ 'arsse_users' => [ - 'columns' => [ - 'id' => 'str', - 'password' => 'str', - 'num' => 'int', - ], + 'columns' => ["id", "password", "num"], 'rows' => [ ["jane.doe@example.com", "", 1], ["john.doe@example.com", "", 2], @@ -32,11 +28,7 @@ trait SeriesArticle { ], ], 'arsse_feeds' => [ - 'columns' => [ - 'id' => "int", - 'url' => "str", - 'title' => "str", - ], + 'columns' => ["id", "url", "title"], 'rows' => [ [1,"http://example.com/1", "Feed 1"], [2,"http://example.com/2", "Feed 2"], @@ -54,12 +46,7 @@ trait SeriesArticle { ], ], 'arsse_folders' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'parent' => "int", - 'name' => "str", - ], + 'columns' => ["id", "owner", "parent", "name"], 'rows' => [ [1, "john.doe@example.com", null, "Technology"], [2, "john.doe@example.com", 1, "Software"], @@ -73,11 +60,7 @@ trait SeriesArticle { ], ], 'arsse_tags' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'name' => "str", - ], + 'columns' => ["id", "owner", "name"], 'rows' => [ [1, "john.doe@example.com", "Technology"], [2, "john.doe@example.com", "Software"], @@ -90,38 +73,27 @@ trait SeriesArticle { ], ], 'arsse_subscriptions' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'feed' => "int", - 'folder' => "int", - 'title' => "str", - 'scrape' => "bool", - ], + 'columns' => ["id", "owner", "feed", "folder", "title", "scrape"], 'rows' => [ - [1, "john.doe@example.com",1, null,"Subscription 1",0], - [2, "john.doe@example.com",2, null,null,0], - [3, "john.doe@example.com",3, 1,"Subscription 3",0], - [4, "john.doe@example.com",4, 6,null,0], - [5, "john.doe@example.com",10, 5,"Subscription 5",0], - [6, "jane.doe@example.com",1, null,null,0], - [7, "jane.doe@example.com",10,null,"Subscription 7",0], - [8, "john.doe@example.org",11,null,null,0], - [9, "john.doe@example.org",12,null,"Subscription 9",0], - [10,"john.doe@example.org",13,null,null,0], + [1, "john.doe@example.com",1, null,"Subscription 1", 0], + [2, "john.doe@example.com",2, null,null, 0], + [3, "john.doe@example.com",3, 1,"Subscription 3", 0], + [4, "john.doe@example.com",4, 6,null, 0], + [5, "john.doe@example.com",10, 5,"Subscription 5", 0], + [6, "jane.doe@example.com",1, null,null, 0], + [7, "jane.doe@example.com",10,null,"Subscription 7", 0], + [8, "john.doe@example.org",11,null,null, 0], + [9, "john.doe@example.org",12,null,"Subscription 9", 0], + [10,"john.doe@example.org",13,null,null, 0], [11,"john.doe@example.net",10,null,"Subscription 11",0], - [12,"john.doe@example.net",2, 9,null,0], + [12,"john.doe@example.net",2, 9,null, 0], [13,"john.doe@example.net",3, 8,"Subscription 13",0], - [14,"john.doe@example.net",4, 7,null,0], - [15,"jill.doe@example.com",11,null,null,1], + [14,"john.doe@example.net",4, 7,null, 0], + [15,"jill.doe@example.com",11,null,null, 1], ], ], 'arsse_tag_members' => [ - 'columns' => [ - 'tag' => "int", - 'subscription' => "int", - 'assigned' => "bool", - ], + 'columns' => ["tag", "subscription", "assigned"], 'rows' => [ [1,3,1], [1,4,1], @@ -137,20 +109,8 @@ trait SeriesArticle { ], 'arsse_articles' => [ 'columns' => [ - 'id' => "int", - 'feed' => "int", - 'url' => "str", - 'title' => "str", - 'author' => "str", - 'published' => "datetime", - 'edited' => "datetime", - 'content' => "str", - 'guid' => "str", - 'url_title_hash' => "str", - 'url_content_hash' => "str", - 'title_content_hash' => "str", - 'modified' => "datetime", - 'content_scraped' => "str", + "id", "feed", "url", "title", "author", "published", "edited", "content", "guid", + "url_title_hash", "url_content_hash", "title_content_hash", "modified", "content_scraped" ], 'rows' => [ [1,1,null,"Title one", null,null,null,"First article", null,"","","","2000-01-01T00:00:00Z",null], @@ -181,11 +141,7 @@ trait SeriesArticle { ], ], 'arsse_enclosures' => [ - 'columns' => [ - 'article' => "int", - 'url' => "str", - 'type' => "str", - ], + 'columns' => ["article", "url", "type"], 'rows' => [ [102,"http://example.com/text","text/plain"], [103,"http://example.com/video","video/webm"], @@ -195,10 +151,7 @@ trait SeriesArticle { ], ], 'arsse_editions' => [ - 'columns' => [ - 'id' => "int", - 'article' => "int", - ], + 'columns' => ["id", "article"], 'rows' => [ [1,1], [2,2], @@ -234,15 +187,7 @@ trait SeriesArticle { ], ], 'arsse_marks' => [ - 'columns' => [ - 'subscription' => "int", - 'article' => "int", - 'read' => "bool", - 'starred' => "bool", - 'modified' => "datetime", - 'note' => "str", - 'hidden' => "bool", - ], + 'columns' => ["subscription", "article", "read", "starred", "modified", "note", "hidden"], 'rows' => [ [1, 1,1,1,'2000-01-01 00:00:00','',0], [5, 19,1,0,'2016-01-01 00:00:00','',0], @@ -263,10 +208,7 @@ trait SeriesArticle { ], ], 'arsse_categories' => [ // author-supplied categories - 'columns' => [ - 'article' => "int", - 'name' => "str", - ], + 'columns' => ["article", "name"], 'rows' => [ [19,"Fascinating"], [19,"Logical"], @@ -274,12 +216,8 @@ trait SeriesArticle { [20,"Logical"], ], ], - 'arsse_labels' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'name' => "str", - ], + 'arsse_labels' => [ // labels applied to articles + 'columns' => ["id", "owner", "name"], 'rows' => [ [1,"john.doe@example.com","Interesting"], [2,"john.doe@example.com","Fascinating"], @@ -288,13 +226,7 @@ trait SeriesArticle { ], ], 'arsse_label_members' => [ - 'columns' => [ - 'label' => "int", - 'article' => "int", - 'subscription' => "int", - 'assigned' => "bool", - 'modified' => "datetime", - ], + 'columns' => ["label", "article", "subscription", "assigned", "modified"], 'rows' => [ [1, 1,1,1,'2000-01-01 00:00:00'], [2, 1,1,1,'2000-01-01 00:00:00'], diff --git a/tests/cases/Database/SeriesCleanup.php b/tests/cases/Database/SeriesCleanup.php index d863a644..8850ecd2 100644 --- a/tests/cases/Database/SeriesCleanup.php +++ b/tests/cases/Database/SeriesCleanup.php @@ -27,23 +27,14 @@ trait SeriesCleanup { $faroff = (new Date("now + 1 hour", $tz))->format("Y-m-d H:i:s"); $this->data = [ 'arsse_users' => [ - 'columns' => [ - 'id' => 'str', - 'password' => 'str', - 'num' => 'int', - ], + 'columns' => ["id", "password", "num"], 'rows' => [ ["jane.doe@example.com", "",1], ["john.doe@example.com", "",2], ], ], 'arsse_sessions' => [ - 'columns' => [ - 'id' => "str", - 'created' => "datetime", - 'expires' => "datetime", - 'user' => "str", - ], + 'columns' => ["id", "created", "expires", "user"], 'rows' => [ ["a", $nowish, $faroff, "jane.doe@example.com"], // not expired and recently created, thus kept ["b", $nowish, $soon, "jane.doe@example.com"], // not expired and recently created, thus kept @@ -53,12 +44,7 @@ trait SeriesCleanup { ], ], 'arsse_tokens' => [ - 'columns' => [ - 'id' => "str", - 'class' => "str", - 'user' => "str", - 'expires' => "datetime", - ], + 'columns' => ["id", "class", "user", "expires"], 'rows' => [ ["80fa94c1a11f11e78667001e673b2560", "fever.login", "jane.doe@example.com", $faroff], ["27c6de8da13311e78667001e673b2560", "fever.login", "jane.doe@example.com", $weeksago], // expired @@ -67,11 +53,7 @@ trait SeriesCleanup { ], ], 'arsse_icons' => [ - 'columns' => [ - 'id' => "int", - 'url' => "str", - 'orphaned' => "datetime", - ], + 'columns' => ["id", "url", "orphaned"], 'rows' => [ [1,'http://localhost:8000/Icon/PNG',$daybefore], [2,'http://localhost:8000/Icon/GIF',$daybefore], @@ -79,14 +61,7 @@ trait SeriesCleanup { ], ], 'arsse_feeds' => [ - 'columns' => [ - 'id' => "int", - 'url' => "str", - 'title' => "str", - 'orphaned' => "datetime", - 'size' => "int", - 'icon' => "int", - ], + 'columns' => ["id", "url", "title", "orphaned", "size", "icon"], 'rows' => [ [1,"http://example.com/1","",$daybefore,2,null], //latest two articles should be kept [2,"http://example.com/2","",$yesterday,0,2], @@ -95,11 +70,7 @@ trait SeriesCleanup { ], ], 'arsse_subscriptions' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'feed' => "int", - ], + 'columns' => ["id", "owner", "feed"], 'rows' => [ // one feed previously marked for deletion has a subscription again, and so should not be deleted [1,'jane.doe@example.com',1], @@ -108,14 +79,7 @@ trait SeriesCleanup { ], ], 'arsse_articles' => [ - 'columns' => [ - 'id' => "int", - 'feed' => "int", - 'url_title_hash' => "str", - 'url_content_hash' => "str", - 'title_content_hash' => "str", - 'modified' => "datetime", - ], + 'columns' => ["id", "feed", "url_title_hash", "url_content_hash", "title_content_hash", "modified"], 'rows' => [ [1,1,"","","",$weeksago], // is the latest article, thus is kept [2,1,"","","",$weeksago], // is the second latest article, thus is kept @@ -129,10 +93,7 @@ trait SeriesCleanup { ], ], 'arsse_editions' => [ - 'columns' => [ - 'id' => "int", - 'article' => "int", - ], + 'columns' => ["id", "article"], 'rows' => [ [1,1], [2,2], @@ -143,14 +104,7 @@ trait SeriesCleanup { ], ], 'arsse_marks' => [ - 'columns' => [ - 'article' => "int", - 'subscription' => "int", - 'read' => "bool", - 'starred' => "bool", - 'hidden' => "bool", - 'modified' => "datetime", - ], + 'columns' => ["article", "subscription", "read", "starred", "hidden", "modified"], 'rows' => [ [3,1,0,1,0,$weeksago], [4,1,1,0,0,$daysago], diff --git a/tests/cases/Database/SeriesFeed.php b/tests/cases/Database/SeriesFeed.php index 5cc0d84c..67fb77a6 100644 --- a/tests/cases/Database/SeriesFeed.php +++ b/tests/cases/Database/SeriesFeed.php @@ -17,23 +17,14 @@ trait SeriesFeed { $now = gmdate("Y-m-d H:i:s", strtotime("now")); $this->data = [ 'arsse_users' => [ - 'columns' => [ - 'id' => 'str', - 'password' => 'str', - 'num' => 'int', - ], + 'columns' => ["id", "password", "num"], 'rows' => [ ["jane.doe@example.com", "",1], ["john.doe@example.com", "",2], ], ], 'arsse_icons' => [ - 'columns' => [ - 'id' => "int", - 'url' => "str", - 'type' => "str", - 'data' => "blob", - ], + 'columns' => ["id", "url", "type", "data"], 'rows' => [ [1,'http://localhost:8000/Icon/PNG','image/png',base64_decode("iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAAZdEVYdFNvZnR3YXJlAHBhaW50Lm5ldCA0LjAuMjHxIGmVAAAADUlEQVQYV2NgYGBgAAAABQABijPjAAAAAABJRU5ErkJggg==")], [2,'http://localhost:8000/Icon/GIF','image/gif',base64_decode("R0lGODlhAQABAIABAAAAAP///yH5BAEKAAEALAAAAAABAAEAAAICTAEAOw==")], @@ -42,17 +33,7 @@ trait SeriesFeed { ], ], 'arsse_feeds' => [ - 'columns' => [ - 'id' => "int", - 'url' => "str", - 'title' => "str", - 'err_count' => "int", - 'err_msg' => "str", - 'modified' => "datetime", - 'next_fetch' => "datetime", - 'size' => "int", - 'icon' => "int", - ], + 'columns' => ["id", "url", "title", "err_count", "err_msg", "modified", "next_fetch", "size", "icon"], 'rows' => [ [1,"http://localhost:8000/Feed/Matching/3","Ook",0,"",$past,$past,0,null], [2,"http://localhost:8000/Feed/Matching/1","Eek",5,"There was an error last time",$past,$future,0,null], @@ -67,13 +48,7 @@ trait SeriesFeed { ], ], 'arsse_subscriptions' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'feed' => "int", - 'keep_rule' => "str", - 'block_rule' => "str", - ], + 'columns' => ["id", "owner", "feed", "keep_rule", "block_rule"], 'rows' => [ [1,'john.doe@example.com',1,null,'^Sport$'], [2,'john.doe@example.com',2,"",null], @@ -84,21 +59,7 @@ trait SeriesFeed { ], ], 'arsse_articles' => [ - 'columns' => [ - 'id' => "int", - 'feed' => "int", - 'url' => "str", - 'title' => "str", - 'author' => "str", - 'published' => "datetime", - 'edited' => "datetime", - 'content' => "str", - 'guid' => "str", - 'url_title_hash' => "str", - 'url_content_hash' => "str", - 'title_content_hash' => "str", - 'modified' => "datetime", - ], + 'columns' => ["id", "feed", "url", "title", "author", "published", "edited", "content", "guid", "url_title_hash", "url_content_hash", "title_content_hash", "modified"], 'rows' => [ [1,1,'http://example.com/1','Article title 1','','2000-01-01 00:00:00','2000-01-01 00:00:00','

Article content 1

','e433653cef2e572eee4215fa299a4a5af9137b2cefd6283c85bd69a32915beda','f5cb8bfc1c7396dc9816af212a3e2ac5221585c2a00bf7ccb6aabd95dcfcd6a6','fb0bc8f8cb08913dc5a497db700e327f1d34e4987402687d494a5891f24714d4','18fdd4fa93d693128c43b004399e5c9cea6c261ddfa002518d3669f55d8c2207',$past], [2,1,'http://example.com/2','Article title 2','','2000-01-02 00:00:00','2000-01-02 00:00:00','

Article content 2

','5be8a5a46ecd52ed132191c8d27fb1af6b3d4edc00234c5d9f8f0e10562ed3b7','0e86d2de822a174fe3c44a466953e63ca1f1a58a19cbf475fce0855d4e3d5153','13075894189c47ffcfafd1dfe7fbb539f7c74a69d35a399b3abf8518952714f9','2abd0a8cba83b8214a66c8f0293ba63e467d720540e29ff8ddcdab069d4f1c9e',$past], @@ -110,11 +71,7 @@ trait SeriesFeed { ], ], 'arsse_editions' => [ - 'columns' => [ - 'id' => "int", - 'article' => "int", - 'modified' => "datetime", - ], + 'columns' => ["id", "article", "modified"], 'rows' => [ [1,1,$past], [2,2,$past], @@ -124,14 +81,7 @@ trait SeriesFeed { ], ], 'arsse_marks' => [ - 'columns' => [ - 'article' => "int", - 'subscription' => "int", - 'read' => "bool", - 'starred' => "bool", - 'hidden' => "bool", - 'modified' => "datetime", - ], + 'columns' => ["article", "subscription", "read", "starred", "hidden", "modified"], 'rows' => [ // Jane's marks [1,6,1,0,0,$past], @@ -146,20 +96,13 @@ trait SeriesFeed { ], ], 'arsse_enclosures' => [ - 'columns' => [ - 'article' => "int", - 'url' => "str", - 'type' => "str", - ], + 'columns' => ["article", "url", "type"], 'rows' => [ [7,'http://example.com/png','image/png'], ], ], 'arsse_categories' => [ - 'columns' => [ - 'article' => "int", - 'name' => "str", - ], + 'columns' => ["article", "name"], 'rows' => [ [7,'Syrinx'], ], diff --git a/tests/cases/Database/SeriesFolder.php b/tests/cases/Database/SeriesFolder.php index 4c488ced..4e0eec4b 100644 --- a/tests/cases/Database/SeriesFolder.php +++ b/tests/cases/Database/SeriesFolder.php @@ -12,23 +12,14 @@ trait SeriesFolder { protected function setUpSeriesFolder(): void { $this->data = [ 'arsse_users' => [ - 'columns' => [ - 'id' => 'str', - 'password' => 'str', - 'num' => 'int', - ], + 'columns' => ["id", "password", "num"], 'rows' => [ ["jane.doe@example.com", "",1], ["john.doe@example.com", "",2], ], ], 'arsse_folders' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'parent' => "int", - 'name' => "str", - ], + 'columns' => ["id", "owner", "parent", "name"], /* Layout translates to: Jane Politics @@ -49,11 +40,7 @@ trait SeriesFolder { ], ], 'arsse_feeds' => [ - 'columns' => [ - 'id' => "int", - 'url' => "str", - 'title' => "str", - ], + 'columns' => ["id", "url", "title"], 'rows' => [ [1,"http://example.com/1", "Feed 1"], [2,"http://example.com/2", "Feed 2"], @@ -71,12 +58,7 @@ trait SeriesFolder { ], ], 'arsse_subscriptions' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'feed' => "int", - 'folder' => "int", - ], + 'columns' => ["id", "owner", "feed", "folder"], 'rows' => [ [1, "john.doe@example.com",1, null], [2, "john.doe@example.com",2, null], diff --git a/tests/cases/Database/SeriesIcon.php b/tests/cases/Database/SeriesIcon.php index 667651f2..73b6cf4e 100644 --- a/tests/cases/Database/SeriesIcon.php +++ b/tests/cases/Database/SeriesIcon.php @@ -16,23 +16,14 @@ trait SeriesIcon { $now = gmdate("Y-m-d H:i:s", strtotime("now")); $this->data = [ 'arsse_users' => [ - 'columns' => [ - 'id' => 'str', - 'password' => 'str', - 'num' => 'int', - ], + 'columns' => ["id", "password", "num"], 'rows' => [ ["jane.doe@example.com", "",1], ["john.doe@example.com", "",2], ], ], 'arsse_icons' => [ - 'columns' => [ - 'id' => "int", - 'url' => "str", - 'type' => "str", - 'data' => "blob", - ], + 'columns' => ["id", "url", "type", "data"], 'rows' => [ [1,'http://localhost:8000/Icon/PNG','image/png',base64_decode("iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAAZdEVYdFNvZnR3YXJlAHBhaW50Lm5ldCA0LjAuMjHxIGmVAAAADUlEQVQYV2NgYGBgAAAABQABijPjAAAAAABJRU5ErkJggg==")], [2,'http://localhost:8000/Icon/GIF','image/gif',base64_decode("R0lGODlhAQABAIABAAAAAP///yH5BAEKAAEALAAAAAABAAEAAAICTAEAOw==")], @@ -41,17 +32,7 @@ trait SeriesIcon { ], ], 'arsse_feeds' => [ - 'columns' => [ - 'id' => "int", - 'url' => "str", - 'title' => "str", - 'err_count' => "int", - 'err_msg' => "str", - 'modified' => "datetime", - 'next_fetch' => "datetime", - 'size' => "int", - 'icon' => "int", - ], + 'columns' => ["id", "url", "title", "err_count", "err_msg", "modified", "next_fetch", "size", "icon"], 'rows' => [ [1,"http://localhost:8000/Feed/Matching/3","Ook",0,"",$past,$past,0,1], [2,"http://localhost:8000/Feed/Matching/1","Eek",5,"There was an error last time",$past,$future,0,2], @@ -61,11 +42,7 @@ trait SeriesIcon { ], ], 'arsse_subscriptions' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'feed' => "int", - ], + 'columns' => ["id", "owner", "feed"], 'rows' => [ [1,'john.doe@example.com',1], [2,'john.doe@example.com',2], diff --git a/tests/cases/Database/SeriesLabel.php b/tests/cases/Database/SeriesLabel.php index 4a4fac66..d4ffa721 100644 --- a/tests/cases/Database/SeriesLabel.php +++ b/tests/cases/Database/SeriesLabel.php @@ -14,11 +14,7 @@ trait SeriesLabel { protected function setUpSeriesLabel(): void { $this->data = [ 'arsse_users' => [ - 'columns' => [ - 'id' => 'str', - 'password' => 'str', - 'num' => 'int', - ], + 'columns' => ["id", "password", "num"], 'rows' => [ ["jane.doe@example.com", "",1], ["john.doe@example.com", "",2], @@ -27,12 +23,7 @@ trait SeriesLabel { ], ], 'arsse_folders' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'parent' => "int", - 'name' => "str", - ], + 'columns' => ["id", "owner", "parent", "name"], 'rows' => [ [1, "john.doe@example.com", null, "Technology"], [2, "john.doe@example.com", 1, "Software"], @@ -46,10 +37,7 @@ trait SeriesLabel { ], ], 'arsse_feeds' => [ - 'columns' => [ - 'id' => "int", - 'url' => "str", - ], + 'columns' => ["id", "url"], 'rows' => [ [1,"http://example.com/1"], [2,"http://example.com/2"], @@ -67,12 +55,7 @@ trait SeriesLabel { ], ], 'arsse_subscriptions' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'feed' => "int", - 'folder' => "int", - ], + 'columns' => ["id", "owner", "feed", "folder"], 'rows' => [ [1,"john.doe@example.com",1,null], [2,"john.doe@example.com",2,null], @@ -91,21 +74,7 @@ trait SeriesLabel { ], ], 'arsse_articles' => [ - 'columns' => [ - 'id' => "int", - 'feed' => "int", - 'url' => "str", - 'title' => "str", - 'author' => "str", - 'published' => "datetime", - 'edited' => "datetime", - 'content' => "str", - 'guid' => "str", - 'url_title_hash' => "str", - 'url_content_hash' => "str", - 'title_content_hash' => "str", - 'modified' => "datetime", - ], + 'columns' => ["id", "feed", "url", "title", "author", "published", "edited", "content", "guid", "url_title_hash", "url_content_hash", "title_content_hash", "modified"], 'rows' => [ [1,1,null,null,null,null,null,null,null,"","","","2000-01-01T00:00:00Z"], [2,1,null,null,null,null,null,null,null,"","","","2010-01-01T00:00:00Z"], @@ -135,11 +104,7 @@ trait SeriesLabel { ], ], 'arsse_enclosures' => [ - 'columns' => [ - 'article' => "int", - 'url' => "str", - 'type' => "str", - ], + 'columns' => ["article", "url", "type"], 'rows' => [ [102,"http://example.com/text","text/plain"], [103,"http://example.com/video","video/webm"], @@ -149,10 +114,7 @@ trait SeriesLabel { ], ], 'arsse_editions' => [ - 'columns' => [ - 'id' => "int", - 'article' => "int", - ], + 'columns' => ["id", "article"], 'rows' => [ [1,1], [2,2], @@ -188,14 +150,7 @@ trait SeriesLabel { ], ], 'arsse_marks' => [ - 'columns' => [ - 'subscription' => "int", - 'article' => "int", - 'read' => "bool", - 'starred' => "bool", - 'modified' => "datetime", - 'hidden' => "bool", - ], + 'columns' => ["subscription", "article", "read", "starred", "modified", "hidden"], 'rows' => [ [1, 1,1,1,'2000-01-01 00:00:00',0], [5, 19,1,0,'2000-01-01 00:00:00',0], @@ -213,11 +168,7 @@ trait SeriesLabel { ], ], 'arsse_labels' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'name' => "str", - ], + 'columns' => ["id", "owner", "name"], 'rows' => [ [1,"john.doe@example.com","Interesting"], [2,"john.doe@example.com","Fascinating"], @@ -226,12 +177,7 @@ trait SeriesLabel { ], ], 'arsse_label_members' => [ - 'columns' => [ - 'label' => "int", - 'article' => "int", - 'subscription' => "int", - 'assigned' => "bool", - ], + 'columns' => ["label", "article", "subscription", "assigned"], 'rows' => [ [1, 1,1,1], [2, 1,1,1], diff --git a/tests/cases/Database/SeriesMeta.php b/tests/cases/Database/SeriesMeta.php index b1d19743..aeac6b79 100644 --- a/tests/cases/Database/SeriesMeta.php +++ b/tests/cases/Database/SeriesMeta.php @@ -13,13 +13,10 @@ trait SeriesMeta { protected function setUpSeriesMeta(): void { $dataBare = [ 'arsse_meta' => [ - 'columns' => [ - 'key' => 'str', - 'value' => 'str', - ], + 'columns' => ["key", "value"], 'rows' => [ - //['schema_version', "".\JKingWeb\Arsse\Database::SCHEMA_VERSION], - ['album',"A Farewell to Kings"], + //['schema_version', "".\JKingWeb\Arsse\Database::SCHEMA_VERSION], + ['album',"A Farewell to Kings"], ], ], ]; diff --git a/tests/cases/Database/SeriesSession.php b/tests/cases/Database/SeriesSession.php index 1db319f8..ffbba56e 100644 --- a/tests/cases/Database/SeriesSession.php +++ b/tests/cases/Database/SeriesSession.php @@ -23,23 +23,14 @@ trait SeriesSession { $old = gmdate("Y-m-d H:i:s", strtotime("now - 2 days")); $this->data = [ 'arsse_users' => [ - 'columns' => [ - 'id' => 'str', - 'password' => 'str', - 'num' => 'int', - ], + 'columns' => ["id", "password", "num"], 'rows' => [ ["jane.doe@example.com", "",1], ["john.doe@example.com", "",2], ], ], 'arsse_sessions' => [ - 'columns' => [ - 'id' => "str", - 'user' => "str", - 'created' => "datetime", - 'expires' => "datetime", - ], + 'columns' => ["id", "user", "created", "expires"], 'rows' => [ ["80fa94c1a11f11e78667001e673b2560", "jane.doe@example.com", $past, $faroff], ["27c6de8da13311e78667001e673b2560", "jane.doe@example.com", $past, $past], // expired diff --git a/tests/cases/Database/SeriesSubscription.php b/tests/cases/Database/SeriesSubscription.php index 0b5f6512..0b279702 100644 --- a/tests/cases/Database/SeriesSubscription.php +++ b/tests/cases/Database/SeriesSubscription.php @@ -15,11 +15,7 @@ trait SeriesSubscription { public function setUpSeriesSubscription(): void { $this->data = [ 'arsse_users' => [ - 'columns' => [ - 'id' => 'str', - 'password' => 'str', - 'num' => 'int', - ], + 'columns' => ["id", "password", "num"], 'rows' => [ ["jane.doe@example.com", "", 1], ["john.doe@example.com", "", 2], @@ -28,12 +24,7 @@ trait SeriesSubscription { ], ], 'arsse_folders' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'parent' => "int", - 'name' => "str", - ], + 'columns' => ["id", "owner", "parent", "name"], 'rows' => [ [1, "john.doe@example.com", null, "Technology"], [2, "john.doe@example.com", 1, "Software"], @@ -44,27 +35,14 @@ trait SeriesSubscription { ], ], 'arsse_icons' => [ - 'columns' => [ - 'id' => "int", - 'url' => "str", - 'data' => "blob", - ], + 'columns' => ["id", "url", "data"], 'rows' => [ [1,"http://example.com/favicon.ico", "ICON DATA"], [2,"http://example.net/favicon.ico", null], ], ], 'arsse_feeds' => [ - 'columns' => [ - 'id' => "int", - 'url' => "str", - 'title' => "str", - 'username' => "str", - 'password' => "str", - 'updated' => "datetime", - 'next_fetch' => "datetime", - 'icon' => "int", - ], + 'columns' => ["id", "url", "title", "username", "password", "updated", "next_fetch", "icon"], 'rows' => [ [1,"http://example.com/feed1", "Ook", "", "",strtotime("now"),strtotime("now"),null], [2,"http://example.com/feed2", "eek", "", "",strtotime("now - 1 hour"),strtotime("now - 1 hour"),1], @@ -73,18 +51,7 @@ trait SeriesSubscription { ], ], 'arsse_subscriptions' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'feed' => "int", - 'title' => "str", - 'folder' => "int", - 'pinned' => "bool", - 'order_type' => "int", - 'keep_rule' => "str", - 'block_rule' => "str", - 'scrape' => "bool", - ], + 'columns' => ["id", "owner", "feed", "title", "folder", "pinned", "order_type", "keep_rule", "block_rule", "scrape"], 'rows' => [ [1,"john.doe@example.com",2,null,null,1,2,null,null,0], [2,"jane.doe@example.com",2,null,null,0,0,null,null,0], @@ -95,11 +62,7 @@ trait SeriesSubscription { ], ], 'arsse_tags' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'name' => "str", - ], + 'columns' => ["id", "owner", "name"], 'rows' => [ [1,"john.doe@example.com","Interesting"], [2,"john.doe@example.com","Fascinating"], @@ -108,11 +71,7 @@ trait SeriesSubscription { ], ], 'arsse_tag_members' => [ - 'columns' => [ - 'tag' => "int", - 'subscription' => "int", - 'assigned' => "bool", - ], + 'columns' => ["tag", "subscription", "assigned"], 'rows' => [ [1,1,1], [1,3,0], @@ -122,14 +81,7 @@ trait SeriesSubscription { ], ], 'arsse_articles' => [ - 'columns' => [ - 'id' => "int", - 'feed' => "int", - 'url_title_hash' => "str", - 'url_content_hash' => "str", - 'title_content_hash' => "str", - 'title' => "str", - ], + 'columns' => ["id", "feed", "url_title_hash", "url_content_hash", "title_content_hash", "title"], 'rows' => [ [1,2,"","","","Title 1"], [2,2,"","","","Title 2"], @@ -142,10 +94,7 @@ trait SeriesSubscription { ], ], 'arsse_editions' => [ - 'columns' => [ - 'id' => "int", - 'article' => "int", - ], + 'columns' => ["id", "article"], 'rows' => [ [1,1], [2,2], @@ -158,10 +107,7 @@ trait SeriesSubscription { ], ], 'arsse_categories' => [ - 'columns' => [ - 'article' => "int", - 'name' => "str", - ], + 'columns' => ["article", "name"], 'rows' => [ [1,"A"], [2,"B"], @@ -173,13 +119,7 @@ trait SeriesSubscription { ], ], 'arsse_marks' => [ - 'columns' => [ - 'article' => "int", - 'subscription' => "int", - 'read' => "bool", - 'starred' => "bool", - 'hidden' => "bool", - ], + 'columns' => ["article", "subscription", "read", "starred", "hidden"], 'rows' => [ [1,2,1,0,0], [2,2,1,0,0], diff --git a/tests/cases/Database/SeriesTag.php b/tests/cases/Database/SeriesTag.php index 1f2ea9cd..47c9fa7c 100644 --- a/tests/cases/Database/SeriesTag.php +++ b/tests/cases/Database/SeriesTag.php @@ -13,11 +13,7 @@ trait SeriesTag { protected function setUpSeriesTag(): void { $this->data = [ 'arsse_users' => [ - 'columns' => [ - 'id' => 'str', - 'password' => 'str', - 'num' => 'int', - ], + 'columns' => ["id", "password", "num"], 'rows' => [ ["jane.doe@example.com", "",1], ["john.doe@example.com", "",2], @@ -26,11 +22,7 @@ trait SeriesTag { ], ], 'arsse_feeds' => [ - 'columns' => [ - 'id' => "int", - 'url' => "str", - 'title' => "str", - ], + 'columns' => ["id", "url", "title"], 'rows' => [ [1,"http://example.com/1",""], [2,"http://example.com/2",""], @@ -48,12 +40,7 @@ trait SeriesTag { ], ], 'arsse_subscriptions' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'feed' => "int", - 'title' => "str", - ], + 'columns' => ["id", "owner", "feed", "title"], 'rows' => [ [1, "john.doe@example.com", 1,"Lord of Carrots"], [2, "john.doe@example.com", 2,null], @@ -72,11 +59,7 @@ trait SeriesTag { ], ], 'arsse_tags' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'name' => "str", - ], + 'columns' => ["id", "owner", "name"], 'rows' => [ [1,"john.doe@example.com","Interesting"], [2,"john.doe@example.com","Fascinating"], @@ -85,11 +68,7 @@ trait SeriesTag { ], ], 'arsse_tag_members' => [ - 'columns' => [ - 'tag' => "int", - 'subscription' => "int", - 'assigned' => "bool", - ], + 'columns' => ["tag", "subscription", "assigned"], 'rows' => [ [1,1,1], [1,3,0], diff --git a/tests/cases/Database/SeriesToken.php b/tests/cases/Database/SeriesToken.php index 7a14ed0d..ab90244e 100644 --- a/tests/cases/Database/SeriesToken.php +++ b/tests/cases/Database/SeriesToken.php @@ -17,24 +17,14 @@ trait SeriesToken { $old = gmdate("Y-m-d H:i:s", strtotime("now - 2 days")); $this->data = [ 'arsse_users' => [ - 'columns' => [ - 'id' => 'str', - 'password' => 'str', - 'num' => 'int', - ], + 'columns' => ["id", "password", "num"], 'rows' => [ ["jane.doe@example.com", "",1], ["john.doe@example.com", "",2], ], ], 'arsse_tokens' => [ - 'columns' => [ - 'id' => "str", - 'class' => "str", - 'user' => "str", - 'expires' => "datetime", - 'data' => "str", - ], + 'columns' => ["id", "class", "user", "expires", "data"], 'rows' => [ ["80fa94c1a11f11e78667001e673b2560", "fever.login", "jane.doe@example.com", $faroff, null], ["27c6de8da13311e78667001e673b2560", "fever.login", "jane.doe@example.com", $past, null], // expired diff --git a/tests/cases/Database/SeriesUser.php b/tests/cases/Database/SeriesUser.php index 031e5161..2caba860 100644 --- a/tests/cases/Database/SeriesUser.php +++ b/tests/cases/Database/SeriesUser.php @@ -12,12 +12,7 @@ trait SeriesUser { protected function setUpSeriesUser(): void { $this->data = [ 'arsse_users' => [ - 'columns' => [ - 'id' => 'str', - 'password' => 'str', - 'num' => 'int', - 'admin' => 'bool', - ], + 'columns' => ["id", "password", "num", "admin"], 'rows' => [ ["admin@example.net", '$2y$10$PbcG2ZR3Z8TuPzM7aHTF8.v61dtCjzjK78gdZJcp4UePE8T9jEgBW', 1, 1], // password is hash of "secret" ["jane.doe@example.com", "", 2, 0], @@ -25,11 +20,7 @@ trait SeriesUser { ], ], 'arsse_user_meta' => [ - 'columns' => [ - 'owner' => "str", - 'key' => "str", - 'value' => "str", - ], + 'columns' => ["owner", "key", "value"], 'rows' => [ ["admin@example.net", "lang", "en"], ["admin@example.net", "tz", "America/Toronto"], diff --git a/tests/cases/ImportExport/TestImportExport.php b/tests/cases/ImportExport/TestImportExport.php index ae0c7a4a..a6d6194b 100644 --- a/tests/cases/ImportExport/TestImportExport.php +++ b/tests/cases/ImportExport/TestImportExport.php @@ -41,23 +41,14 @@ class TestImportExport extends \JKingWeb\Arsse\Test\AbstractTest { Arsse::$db->driverSchemaUpdate(); $this->data = [ 'arsse_users' => [ - 'columns' => [ - 'id' => 'str', - 'password' => 'str', - 'num' => 'int', - ], + 'columns' => ["id", "password", "num"], 'rows' => [ ["john.doe@example.com", "", 1], ["jane.doe@example.com", "", 2], ], ], 'arsse_folders' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'parent' => "int", - 'name' => "str", - ], + 'columns' => ["id", "owner", "parent", "name"], 'rows' => [ [1, "john.doe@example.com", null, "Science"], [2, "john.doe@example.com", 1, "Rocketry"], @@ -68,11 +59,7 @@ class TestImportExport extends \JKingWeb\Arsse\Test\AbstractTest { ], ], 'arsse_feeds' => [ - 'columns' => [ - 'id' => "int", - 'url' => "str", - 'title' => "str", - ], + 'columns' => ["id", "url", "title"], 'rows' => [ [1, "http://localhost:8000/Import/nasa-jpl", "NASA JPL"], [2, "http://localhost:8000/Import/torstar", "Toronto Star"], @@ -83,13 +70,7 @@ class TestImportExport extends \JKingWeb\Arsse\Test\AbstractTest { ], ], 'arsse_subscriptions' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'folder' => "int", - 'feed' => "int", - 'title' => "str", - ], + 'columns' => ["id", "owner", "folder", "feed", "title"], 'rows' => [ [1, "john.doe@example.com", 2, 1, "NASA JPL"], [2, "john.doe@example.com", 5, 2, "Toronto Star"], @@ -100,11 +81,7 @@ class TestImportExport extends \JKingWeb\Arsse\Test\AbstractTest { ], ], 'arsse_tags' => [ - 'columns' => [ - 'id' => "int", - 'owner' => "str", - 'name' => "str", - ], + 'columns' => ["id", "owner", "name"], 'rows' => [ [1, "john.doe@example.com", "canada"], [2, "john.doe@example.com", "frequent"], @@ -115,11 +92,7 @@ class TestImportExport extends \JKingWeb\Arsse\Test\AbstractTest { ], ], 'arsse_tag_members' => [ - 'columns' => [ - 'tag' => "int", - 'subscription' => "int", - 'assigned' => "bool", - ], + 'columns' => ["tag", "subscription", "assigned"], 'rows' => [ [1, 2, 1], [1, 4, 1], diff --git a/tests/lib/AbstractTest.php b/tests/lib/AbstractTest.php index f498a543..5d7776ea 100644 --- a/tests/lib/AbstractTest.php +++ b/tests/lib/AbstractTest.php @@ -387,14 +387,33 @@ abstract class AbstractTest extends \PHPUnit\Framework\TestCase { return $value; } + /** Inserts into the database test data in the following format: + * + * ```php + * $data = [ + * 'some_table' => [ + * 'columns' => ["id", "name"], + * 'rows' => [ + * [1,"Dupond"], + * [2,"Dupont"], + * ] + * ], + * 'other_table' => [ + * ... + * ] + * ]; + * ``` + */ public function primeDatabase(Driver $drv, array $data): bool { $tr = $drv->begin(); foreach ($data as $table => $info) { $cols = array_map(function($v) { return '"'.str_replace('"', '""', $v).'"'; - }, array_keys($info['columns'])); + }, $info['columns']); $cols = implode(",", $cols); - $bindings = array_values($info['columns']); + $bindings = array_map(function($c) use ($table) { + return self::COL_DEFS[$table][$c]; + }, $info['columns']); $params = implode(",", array_fill(0, sizeof($info['columns']), "?")); $s = $drv->prepareArray("INSERT INTO $table($cols) values($params)", $bindings); foreach ($info['rows'] as $row) { @@ -433,16 +452,15 @@ abstract class AbstractTest extends \PHPUnit\Framework\TestCase { // serialize the rows of the actual output $cols = implode(",", array_map(function($v) { return '"'.str_replace('"', '""', $v).'"'; - }, array_keys($info['columns']))); + }, $info['columns'])); $data = $drv->prepare("SELECT $cols from $table")->run()->getAll(); - $types = $info['columns']; $act = []; $extra = []; foreach ($data as $r) { $row = []; foreach ($r as $c => $v) { // account for dates which might be off by one second - if ($types[$c] === "datetime") { + if (self::COL_DEFS[$table][$c] === "datetime") { if (array_search($v, $dates, true) === false) { $v = Date::transform(Date::sub("PT1S", $v), "sql"); if (array_search($v, $dates, true) === false) { @@ -490,23 +508,22 @@ abstract class AbstractTest extends \PHPUnit\Framework\TestCase { $out = []; foreach ($tableSpecs as $table => $columns) { // make sure the source has the table we want - $this->assertArrayHasKey($table, $source, "Source for expectations does not contain requested table $table."); + if (!isset($source[$table])) { + throw new Exception("Source for expectations does not contain requested table $table."); + } + // fill the output, particularly the correct number of (empty) rows + $rows = sizeof($source[$table]['rows']); $out[$table] = [ - 'columns' => [], - 'rows' => array_fill(0, sizeof($source[$table]['rows']), []), + 'columns' => $columns, + 'rows' => array_fill(0, $rows, []), ]; - // make sure the source has all the columns we want for the table - $cols = array_flip($columns); - $cols = array_intersect_key($cols, $source[$table]['columns']); - $this->assertSame(array_keys($cols), $columns, "Source for table $table does not contain all requested columns"); - // get a map of source value offsets and keys - $targets = array_flip(array_keys($source[$table]['columns'])); - foreach ($cols as $key => $order) { - // fill the column-spec - $out[$table]['columns'][$key] = $source[$table]['columns'][$key]; - foreach ($source[$table]['rows'] as $index => $row) { - // fill each row column-wise with re-ordered values - $out[$table]['rows'][$index][$order] = $row[$targets[$key]]; + // fill the rows with the requested data, column-wise + foreach ($columns as $c) { + if (($index = array_search($c, $source[$table]['columns'], true)) === false) { + throw new exception("Expected column $table.$c is not present in test data"); + } + for ($a = 0; $a < $rows; $a++) { + $out[$table]['rows'][$a][] = $source[$table]['rows'][$a][$index]; } } } From e9c6ddcfdfe28bec271e33b8e5cbbdd4b1420e2d Mon Sep 17 00:00:00 2001 From: "J. King" Date: Sun, 5 Jun 2022 18:17:26 -0400 Subject: [PATCH 10/10] Adjust date fix-up after column changes --- tests/lib/AbstractTest.php | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/tests/lib/AbstractTest.php b/tests/lib/AbstractTest.php index 5d7776ea..56e8a59c 100644 --- a/tests/lib/AbstractTest.php +++ b/tests/lib/AbstractTest.php @@ -428,14 +428,13 @@ abstract class AbstractTest extends \PHPUnit\Framework\TestCase { public function compareExpectations(Driver $drv, array $expected): void { foreach ($expected as $table => $info) { // serialize the rows of the expected output - $types = array_values($info['columns']); $exp = []; $dates = []; foreach ($info['rows'] as $r) { $row = []; foreach ($r as $c => $v) { // store any date values for later comparison - if ($types[$c] === "datetime") { + if (self::COL_DEFS[$table][$info['columns'][$c]] === "datetime") { $dates[] = $v; } // serialize to CSV, null being represented by no value