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:
parent
8527c83976
commit
f0bfe1fdff
3 changed files with 64 additions and 46 deletions
|
@ -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");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
|
@ -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],
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
];
|
];
|
||||||
|
|
Loading…
Reference in a new issue