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

Simplify editionLatest Database method

Also adjust label querying to take hidden marks into account
This commit is contained in:
J. King 2020-12-20 17:34:32 -05:00
parent 8527c83976
commit f0bfe1fdff
3 changed files with 64 additions and 46 deletions

View file

@ -79,7 +79,11 @@ class Database {
/** Returns the bare name of the calling context's calling method, when __FUNCTION__ is not appropriate */ /** Returns the bare name of the calling context's calling method, when __FUNCTION__ is not appropriate */
protected function caller(): string { protected function caller(): string {
return debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS, 3)[2]['function']; $trace = debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS, 4);
if ($trace[2]['function'] === "articleQuery") {
return $trace[3]['function'];
}
return $trace[2]['function'];
} }
/** Returns the current (actual) schema version of the database; compared against self::SCHEMA_VERSION to know when an upgrade is required */ /** Returns the current (actual) schema version of the database; compared against self::SCHEMA_VERSION to know when an upgrade is required */
@ -748,7 +752,7 @@ class Database {
i.url as favicon, i.url as favicon,
t.top as top_folder, t.top as top_folder,
coalesce(s.title, f.title) as title, coalesce(s.title, f.title) as title,
coalesce((articles - hidden - marked + hidden_marked), articles) as unread coalesce((articles - hidden - marked), articles) as unread
FROM arsse_subscriptions as s FROM arsse_subscriptions as s
left join topmost as t on t.f_id = s.folder left join topmost as t on t.f_id = s.folder
join arsse_feeds as f on f.id = s.feed join arsse_feeds as f on f.id = s.feed
@ -757,9 +761,8 @@ class Database {
left join ( left join (
select select
subscription, subscription,
sum(cast((\"read\" = 1 and hidden = 0) as integer)) as marked, sum(hidden) as hidden,
sum(cast((\"read\" = 0 and hidden = 1) as integer)) as hidden, sum(cast((\"read\" = 1 and hidden = 0) as integer)) as marked
sum(cast((\"read\" = 1 and hidden = 1) as integer)) as hidden_marked
from arsse_marks group by subscription from arsse_marks group by subscription
) as mark_stats on mark_stats.subscription = s.id" ) as mark_stats on mark_stats.subscription = s.id"
); );
@ -1206,12 +1209,11 @@ class Database {
* Each record includes the following keys: * Each record includes the following keys:
* *
* - "owner": The user for whom to apply the filters * - "owner": The user for whom to apply the filters
* - "sub": The subscription ID which ties the user to the feed
* - "keep": The "keep" rule; any articles which fail to match this rule are hidden * - "keep": The "keep" rule; any articles which fail to match this rule are hidden
* - "block": The block rule; any article which matches this rule are hidden * - "block": The block rule; any article which matches this rule are hidden
*/ */
public function feedRulesGet(int $feedID): Db\Result { public function feedRulesGet(int $feedID): Db\Result {
return $this->db->prepare("SELECT owner, id as sub, keep_rule as keep, block_rule as block from arsse_subscriptions where feed = ? and (coalesce(keep_rule, '') || coalesce(block_rule, '')) <> ''", "int")->run($feedID); return $this->db->prepare("SELECT owner, keep_rule as keep, block_rule as block from arsse_subscriptions where feed = ? and (coalesce(keep_rule, '') || coalesce(block_rule, '')) <> ''", "int")->run($feedID);
} }
/** Retrieves various identifiers for the latest $count articles in the given newsfeed. The identifiers are: /** Retrieves various identifiers for the latest $count articles in the given newsfeed. The identifiers are:
@ -1310,6 +1312,7 @@ class Database {
return [ return [
'id' => "arsse_articles.id", 'id' => "arsse_articles.id",
'edition' => "latest_editions.edition", 'edition' => "latest_editions.edition",
'latest_edition' => "max(latest_editions.edition)",
'url' => "arsse_articles.url", 'url' => "arsse_articles.url",
'title' => "arsse_articles.title", 'title' => "arsse_articles.title",
'author' => "arsse_articles.author", 'author' => "arsse_articles.author",
@ -1385,6 +1388,7 @@ class Database {
} }
$outColumns = implode(",", $outColumns); $outColumns = implode(",", $outColumns);
} }
assert(strlen($outColumns) > 0, new \Exception("No input columns matched whitelist"));
// define the basic query, to which we add lots of stuff where necessary // define the basic query, to which we add lots of stuff where necessary
$q = new Query( $q = new Query(
"SELECT "SELECT
@ -1895,14 +1899,8 @@ class Database {
/** Returns the numeric identifier of the most recent edition of an article matching the given context */ /** Returns the numeric identifier of the most recent edition of an article matching the given context */
public function editionLatest(string $user, Context $context = null): int { public function editionLatest(string $user, Context $context = null): int {
$context = $context ?? new Context; $context = $context ?? new Context;
$q = new Query("SELECT max(arsse_editions.id) from arsse_editions left join arsse_articles on article = arsse_articles.id join arsse_subscriptions on arsse_articles.feed = arsse_subscriptions.feed and arsse_subscriptions.owner = ?", "str", $user); $q = $this->articleQuery($user, $context, ["latest_edition"]);
if ($context->subscription()) { return (int) $this->db->prepare((string) $q, $q->getTypes())->run($q->getValues())->getValue();
// if a subscription is specified, make sure it exists
$this->subscriptionValidateId($user, $context->subscription);
// a simple WHERE clause is required here
$q->setWhere("arsse_subscriptions.id = ?", "int", $context->subscription);
}
return (int) $this->db->prepare($q->getQuery(), $q->getTypes())->run($q->getValues())->getValue();
} }
/** Returns a map between all the given edition identifiers and their associated article identifiers */ /** Returns a map between all the given edition identifiers and their associated article identifiers */
@ -1945,14 +1943,19 @@ class Database {
return $this->db->prepare( return $this->db->prepare(
"SELECT * FROM ( "SELECT * FROM (
SELECT SELECT
id,name,coalesce(articles,0) as articles,coalesce(marked,0) as \"read\" id,
name,
coalesce(articles - coalesce(hidden, 0), 0) as articles,
coalesce(marked, 0) as \"read\"
from arsse_labels from arsse_labels
left join ( left join (
SELECT label, sum(assigned) as articles from arsse_label_members group by label SELECT label, sum(assigned) as articles from arsse_label_members group by label
) as label_stats on label_stats.label = arsse_labels.id ) as label_stats on label_stats.label = arsse_labels.id
left join ( left join (
SELECT SELECT
label, sum(\"read\") as marked label,
sum(hidden) as hidden,
sum(cast((\"read\" = 1 and hidden = 0) as integer)) as marked
from arsse_marks from arsse_marks
join arsse_subscriptions on arsse_subscriptions.id = arsse_marks.subscription join arsse_subscriptions on arsse_subscriptions.id = arsse_marks.subscription
join arsse_label_members on arsse_label_members.article = arsse_marks.article join arsse_label_members on arsse_label_members.article = arsse_marks.article
@ -2007,14 +2010,19 @@ class Database {
$type = $byName ? "str" : "int"; $type = $byName ? "str" : "int";
$out = $this->db->prepare( $out = $this->db->prepare(
"SELECT "SELECT
id,name,coalesce(articles,0) as articles,coalesce(marked,0) as \"read\" id,
name,
coalesce(articles - coalesce(hidden, 0), 0) as articles,
coalesce(marked, 0) as \"read\"
FROM arsse_labels FROM arsse_labels
left join ( left join (
SELECT label, sum(assigned) as articles from arsse_label_members group by label SELECT label, sum(assigned) as articles from arsse_label_members group by label
) as label_stats on label_stats.label = arsse_labels.id ) as label_stats on label_stats.label = arsse_labels.id
left join ( left join (
SELECT SELECT
label, sum(\"read\") as marked label,
sum(hidden) as hidden,
sum(cast((\"read\" = 1 and hidden = 0) as integer)) as marked
from arsse_marks from arsse_marks
join arsse_subscriptions on arsse_subscriptions.id = arsse_marks.subscription join arsse_subscriptions on arsse_subscriptions.id = arsse_marks.subscription
join arsse_label_members on arsse_label_members.article = arsse_marks.article join arsse_label_members on arsse_label_members.article = arsse_marks.article
@ -2069,19 +2077,25 @@ class Database {
* @param boolean $byName Whether to interpret the $id parameter as the label's name (true) or identifier (false) * @param boolean $byName Whether to interpret the $id parameter as the label's name (true) or identifier (false)
*/ */
public function labelArticlesGet(string $user, $id, bool $byName = false): array { public function labelArticlesGet(string $user, $id, bool $byName = false): array {
// just do a syntactic check on the label ID $c = (new Context)->hidden(false);
$this->labelValidateId($user, $id, $byName, false); if ($byName) {
$field = !$byName ? "id" : "name"; $c->labelName($id);
$type = !$byName ? "int" : "str"; } else {
$out = $this->db->prepare("SELECT article from arsse_label_members join arsse_labels on label = id where assigned = 1 and $field = ? and owner = ? order by article", $type, "str")->run($id, $user)->getAll(); $c->label($id);
}
try {
$q = $this->articleQuery($user, $c);
$out = $this->db->prepare((string) $q, $q->getTypes())->run($q->getValues())->getAll();
} catch (Db\ExceptionInput $e) {
if ($e->getCode() === 10235) {
throw new Db\ExceptionInput("subjectMissing", ["action" => __FUNCTION__, "field" => "label", 'id' => $id]);
}
throw $e;
}
if (!$out) { if (!$out) {
// if no results were returned, do a full validation on the label ID
$this->labelValidateId($user, $id, $byName, true, true);
// if the validation passes, return the empty result
return $out; return $out;
} else { } else {
// flatten the result to return just the article IDs in a simple array return array_column($out, "id");
return array_column($out, "article");
} }
} }

View file

@ -251,7 +251,7 @@ trait SeriesArticle {
[12, 3,0,1,'2017-01-01 00:00:00','ack',0], [12, 3,0,1,'2017-01-01 00:00:00','ack',0],
[12, 4,1,1,'2017-01-01 00:00:00','ach',0], [12, 4,1,1,'2017-01-01 00:00:00','ach',0],
[1, 2,0,0,'2010-01-01 00:00:00','Some Note',0], [1, 2,0,0,'2010-01-01 00:00:00','Some Note',0],
[3, 5,0,0,'2000-01-01 00:00:00','',1], [3, 6,0,0,'2000-01-01 00:00:00','',1],
[6, 1,0,1,'2010-01-01 00:00:00','',1], [6, 1,0,1,'2010-01-01 00:00:00','',1],
[6, 2,1,0,'2010-01-01 00:00:00','',1], [6, 2,1,0,'2010-01-01 00:00:00','',1],
], ],
@ -447,8 +447,8 @@ trait SeriesArticle {
'Starred and Read in subscription' => [(new Context)->starred(true)->unread(false)->subscription(5), []], 'Starred and Read in subscription' => [(new Context)->starred(true)->unread(false)->subscription(5), []],
'Annotated' => [(new Context)->annotated(true), [2]], 'Annotated' => [(new Context)->annotated(true), [2]],
'Not annotated' => [(new Context)->annotated(false), [1,3,4,5,6,7,8,19,20]], 'Not annotated' => [(new Context)->annotated(false), [1,3,4,5,6,7,8,19,20]],
'Hidden' => [(new Context)->hidden(true), [5]], 'Hidden' => [(new Context)->hidden(true), [6]],
'Not hidden' => [(new Context)->hidden(false), [1,2,3,4,6,7,8,19,20]], 'Not hidden' => [(new Context)->hidden(false), [1,2,3,4,5,7,8,19,20]],
'Labelled' => [(new Context)->labelled(true), [1,5,8,19,20]], 'Labelled' => [(new Context)->labelled(true), [1,5,8,19,20]],
'Not labelled' => [(new Context)->labelled(false), [2,3,4,6,7]], 'Not labelled' => [(new Context)->labelled(false), [2,3,4,6,7]],
'Not after edition 999' => [(new Context)->subscription(5)->latestEdition(999), [19]], 'Not after edition 999' => [(new Context)->subscription(5)->latestEdition(999), [19]],
@ -985,6 +985,7 @@ trait SeriesArticle {
public function testFetchLatestEdition(): void { public function testFetchLatestEdition(): void {
$this->assertSame(1001, Arsse::$db->editionLatest($this->user)); $this->assertSame(1001, Arsse::$db->editionLatest($this->user));
$this->assertSame(4, Arsse::$db->editionLatest($this->user, (new Context)->subscription(12))); $this->assertSame(4, Arsse::$db->editionLatest($this->user, (new Context)->subscription(12)));
$this->assertSame(5, Arsse::$db->editionLatest("john.doe@example.com", (new Context)->subscription(3)->hidden(false)));
} }
public function testFetchLatestEditionOfMissingSubscription(): void { public function testFetchLatestEditionOfMissingSubscription(): void {

View file

@ -194,20 +194,22 @@ trait SeriesLabel {
'read' => "bool", 'read' => "bool",
'starred' => "bool", 'starred' => "bool",
'modified' => "datetime", 'modified' => "datetime",
'hidden' => "bool",
], ],
'rows' => [ 'rows' => [
[1, 1,1,1,'2000-01-01 00:00:00'], [1, 1,1,1,'2000-01-01 00:00:00',0],
[5, 19,1,0,'2000-01-01 00:00:00'], [5, 19,1,0,'2000-01-01 00:00:00',0],
[5, 20,0,1,'2010-01-01 00:00:00'], [5, 20,0,1,'2010-01-01 00:00:00',0],
[7, 20,1,0,'2010-01-01 00:00:00'], [7, 20,1,0,'2010-01-01 00:00:00',0],
[8, 102,1,0,'2000-01-02 02:00:00'], [8, 102,1,0,'2000-01-02 02:00:00',0],
[9, 103,0,1,'2000-01-03 03:00:00'], [9, 103,0,1,'2000-01-03 03:00:00',0],
[9, 104,1,1,'2000-01-04 04:00:00'], [9, 104,1,1,'2000-01-04 04:00:00',0],
[10,105,0,0,'2000-01-05 05:00:00'], [10,105,0,0,'2000-01-05 05:00:00',0],
[11, 19,0,0,'2017-01-01 00:00:00'], [11, 19,0,0,'2017-01-01 00:00:00',0],
[11, 20,1,0,'2017-01-01 00:00:00'], [11, 20,1,0,'2017-01-01 00:00:00',0],
[12, 3,0,1,'2017-01-01 00:00:00'], [12, 3,0,1,'2017-01-01 00:00:00',0],
[12, 4,1,1,'2017-01-01 00:00:00'], [12, 4,1,1,'2017-01-01 00:00:00',0],
[4, 8,0,0,'2000-01-02 02:00:00',1]
], ],
], ],
'arsse_labels' => [ 'arsse_labels' => [
@ -237,6 +239,7 @@ trait SeriesLabel {
[2,20,5,1], [2,20,5,1],
[1, 5,3,0], [1, 5,3,0],
[2, 5,3,1], [2, 5,3,1],
[2, 8,4,1],
], ],
], ],
]; ];