Use MQ for updating directives in entire space

pull/503/head
Zef Hemel 2023-08-10 19:00:28 +02:00
parent 97a84e8538
commit d4f7833f0d
6 changed files with 64 additions and 23 deletions

View File

@ -160,7 +160,11 @@ export class DexieMQ {
await this.processing.bulkDelete(ids.map((id) => [queue, id])); await this.processing.bulkDelete(ids.map((id) => [queue, id]));
} }
async requeueTimeouts(timeout: number, maxRetries?: number) { async requeueTimeouts(
timeout: number,
maxRetries?: number,
disableDLQ?: boolean,
) {
const now = Date.now(); const now = Date.now();
const messages = await this.processing.where("ts").below(now - timeout) const messages = await this.processing.where("ts").below(now - timeout)
.toArray(); .toArray();
@ -175,18 +179,26 @@ export class DexieMQ {
for (const m of messages) { for (const m of messages) {
const retries = (m.retries || 0) + 1; const retries = (m.retries || 0) + 1;
if (maxRetries && retries > maxRetries) { if (maxRetries && retries > maxRetries) {
console.warn( if (disableDLQ) {
"[mq]", console.warn(
"Message exceeded max retries, moving to DLQ", "[mq]",
m, "Message exceeded max retries, flushing message",
); m,
dlqMessages.push({ );
queue: m.queue, } else {
id: m.id, console.warn(
body: m.body, "[mq]",
ts: Date.now(), "Message exceeded max retries, moving to DLQ",
retries, m,
}); );
dlqMessages.push({
queue: m.queue,
id: m.id,
body: m.body,
ts: Date.now(),
retries,
});
}
} else { } else {
console.info("[mq]", "Message ack timed out, requeueing", m); console.info("[mq]", "Message ack timed out, requeueing", m);
requeuedMessages.push({ requeuedMessages.push({

View File

@ -91,9 +91,6 @@ export async function reindexCommand() {
const pages = await space.listPages(); const pages = await space.listPages();
await mq.batchSend("indexQueue", pages.map((page) => page.name)); await mq.batchSend("indexQueue", pages.map((page) => page.name));
// console.log("Indexing queued!");
// await editor.flashNotification("Reindexing done");
} }
// Completion // Completion

View File

@ -9,6 +9,8 @@ import { renderDirectives } from "./directives.ts";
import { extractFrontmatter } from "$sb/lib/frontmatter.ts"; import { extractFrontmatter } from "$sb/lib/frontmatter.ts";
import { PageMeta } from "../../web/types.ts"; import { PageMeta } from "../../web/types.ts";
import { isFederationPath } from "$sb/lib/resolve.ts"; import { isFederationPath } from "$sb/lib/resolve.ts";
import { mq } from "$sb/plugos-syscall/mod.ts";
import { Message } from "$sb/mq.ts";
export async function updateDirectivesOnPageCommand() { export async function updateDirectivesOnPageCommand() {
// If `arg` is a string, it's triggered automatically via an event, not explicitly via a command // If `arg` is a string, it's triggered automatically via an event, not explicitly via a command
@ -81,8 +83,19 @@ export async function updateDirectivesInSpaceCommand() {
await editor.flashNotification( await editor.flashNotification(
"Updating directives in entire space, this can take a while...", "Updating directives in entire space, this can take a while...",
); );
await updateDirectivesInSpace(); // await updateDirectivesInSpace();
await editor.flashNotification("Done!"); const pages = await space.listPages();
await mq.batchSend("directiveUpdateQueue", pages.map((page) => page.name));
}
export async function processUpdateQueue(messages: Message[]) {
for (const message of messages) {
const pageName: string = message.body;
console.log("Updating directives in page", pageName);
await updateDirectivesForPage(pageName);
await mq.ack("directiveUpdateQueue", message.id);
}
} }
async function findReplacements( async function findReplacements(
@ -152,7 +165,20 @@ async function updateDirectivesForPage(
) { ) {
const pageMeta = await space.getPageMeta(pageName); const pageMeta = await space.getPageMeta(pageName);
const currentText = await space.readPage(pageName); const currentText = await space.readPage(pageName);
const newText = await updateDirectives(pageMeta, currentText); const tree = await markdown.parseMarkdown(currentText);
const metaData = await extractFrontmatter(tree, ["$disableDirectives"]);
if (isFederationPath(pageName)) {
console.info("Current page is a federation page, not updating directives.");
return;
}
if (metaData.$disableDirectives) {
console.info("Directives disabled in page meta, not updating them.");
return;
}
const newText = await updateDirectives(pageMeta, tree, currentText);
if (newText !== currentText) { if (newText !== currentText) {
console.info("Content of page changed, saving."); console.info("Content of page changed, saving.");
await space.writePage(pageName, newText); await space.writePage(pageName, newText);
@ -161,9 +187,9 @@ async function updateDirectivesForPage(
export async function updateDirectives( export async function updateDirectives(
pageMeta: PageMeta, pageMeta: PageMeta,
tree: ParseTree,
text: string, text: string,
) { ) {
const tree = await markdown.parseMarkdown(text);
const replacements = await findReplacements(tree, text, pageMeta); const replacements = await findReplacements(tree, text, pageMeta);
// Iterate again and replace the bodies. // Iterate again and replace the bodies.

View File

@ -15,6 +15,11 @@ functions:
path: ./command.ts:updateDirectivesInSpaceCommand path: ./command.ts:updateDirectivesInSpaceCommand
command: command:
name: "Directives: Update Entire Space" name: "Directives: Update Entire Space"
processUpdateQueue:
path: ./command.ts:processUpdateQueue
mqSubscriptions:
- queue: directiveUpdateQueue
batchSize: 3
indexData: indexData:
path: ./data.ts:indexData path: ./data.ts:indexData
events: events:

View File

@ -77,7 +77,8 @@ export async function templateDirectiveRenderer(
newBody = templateFn(parsedArgs, buildHandebarOptions(pageMeta)); newBody = templateFn(parsedArgs, buildHandebarOptions(pageMeta));
// Recursively render directives // Recursively render directives
newBody = await updateDirectives(pageMeta, newBody); const tree = await markdown.parseMarkdown(newBody);
newBody = await updateDirectives(pageMeta, tree, newBody);
} }
return newBody.trim(); return newBody.trim();
} }

View File

@ -99,8 +99,8 @@ export class Client {
this.mq = new DexieMQ(`${this.dbPrefix}_mq`, indexedDB, IDBKeyRange); this.mq = new DexieMQ(`${this.dbPrefix}_mq`, indexedDB, IDBKeyRange);
setInterval(() => { setInterval(() => {
// Timeout after 5s // Timeout after 5s, retries 3 times, otherwise drops the message (no DLQ)
this.mq.requeueTimeouts(5000, 3).catch(console.error); this.mq.requeueTimeouts(5000, 3, true).catch(console.error);
}, 20000); // Look to requeue every 20s }, 20000); // Look to requeue every 20s
// Event hook // Event hook