2023-08-28 23:12:15 +08:00
|
|
|
import { editor, markdown, mq, space, sync } from "$sb/syscalls.ts";
|
2022-12-15 03:04:20 +08:00
|
|
|
import {
|
2023-10-03 20:16:33 +08:00
|
|
|
addParentPointers,
|
|
|
|
findParentMatching,
|
|
|
|
nodeAtPos,
|
2023-08-05 00:56:55 +08:00
|
|
|
ParseTree,
|
2022-12-15 03:04:20 +08:00
|
|
|
removeParentPointers,
|
|
|
|
renderToText,
|
|
|
|
traverseTree,
|
|
|
|
} from "$sb/lib/tree.ts";
|
|
|
|
import { renderDirectives } from "./directives.ts";
|
2022-11-24 19:04:00 +08:00
|
|
|
import { extractFrontmatter } from "$sb/lib/frontmatter.ts";
|
2023-07-30 14:56:44 +08:00
|
|
|
import { isFederationPath } from "$sb/lib/resolve.ts";
|
2023-10-03 20:16:33 +08:00
|
|
|
import { MQMessage, PageMeta } from "$sb/types.ts";
|
2023-08-30 03:17:29 +08:00
|
|
|
import { sleep } from "$sb/lib/async.ts";
|
2023-08-12 02:37:13 +08:00
|
|
|
|
|
|
|
const directiveUpdateQueueName = "directiveUpdateQueue";
|
2022-10-28 22:17:40 +08:00
|
|
|
|
2023-07-14 19:44:30 +08:00
|
|
|
export async function updateDirectivesOnPageCommand() {
|
2022-11-24 23:08:51 +08:00
|
|
|
// If `arg` is a string, it's triggered automatically via an event, not explicitly via a command
|
2023-07-30 14:56:44 +08:00
|
|
|
const currentPage = await editor.getCurrentPage();
|
2023-10-10 02:39:03 +08:00
|
|
|
let pageMeta: PageMeta | undefined;
|
|
|
|
try {
|
|
|
|
pageMeta = await space.getPageMeta(currentPage);
|
|
|
|
} catch {
|
|
|
|
console.info("Page not found, not updating directives");
|
|
|
|
return;
|
|
|
|
}
|
2022-11-19 22:22:43 +08:00
|
|
|
const text = await editor.getText();
|
|
|
|
const tree = await markdown.parseMarkdown(text);
|
2023-05-24 02:53:53 +08:00
|
|
|
const metaData = await extractFrontmatter(tree, ["$disableDirectives"]);
|
2023-07-30 14:56:44 +08:00
|
|
|
|
|
|
|
if (isFederationPath(currentPage)) {
|
|
|
|
console.info("Current page is a federation page, not updating directives.");
|
2023-07-30 17:30:01 +08:00
|
|
|
return;
|
2023-07-30 14:56:44 +08:00
|
|
|
}
|
|
|
|
|
2022-11-19 22:22:43 +08:00
|
|
|
if (metaData.$disableDirectives) {
|
2023-07-30 14:56:44 +08:00
|
|
|
console.info("Directives disabled in page meta, not updating them.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(await sync.hasInitialSyncCompleted())) {
|
|
|
|
console.info(
|
|
|
|
"Initial sync hasn't completed yet, not updating directives.",
|
|
|
|
);
|
2022-11-19 22:22:43 +08:00
|
|
|
return;
|
2022-10-28 22:17:40 +08:00
|
|
|
}
|
2022-11-19 22:22:43 +08:00
|
|
|
|
2023-07-28 19:54:31 +08:00
|
|
|
await editor.save();
|
|
|
|
|
2023-08-05 00:56:55 +08:00
|
|
|
const replacements = await findReplacements(tree, text, pageMeta);
|
2022-11-19 22:22:43 +08:00
|
|
|
|
|
|
|
// Iterate again and replace the bodies. Iterating again (not using previous positions)
|
|
|
|
// because text may have changed in the mean time (directive processing may take some time)
|
|
|
|
// Hypothetically in the mean time directives in text may have been changed/swapped, in which
|
|
|
|
// case this will break. This would be a rare edge case, however.
|
2022-11-23 17:05:01 +08:00
|
|
|
for (const replacement of replacements) {
|
|
|
|
// Fetch the text every time, because dispatch() will have been made changes
|
|
|
|
const text = await editor.getText();
|
|
|
|
// Determine the current position
|
|
|
|
const index = text.indexOf(replacement.fullMatch);
|
|
|
|
|
|
|
|
// This may happen if the query itself, or the user is editing inside the directive block (WHY!?)
|
|
|
|
if (index === -1) {
|
2022-12-15 03:04:20 +08:00
|
|
|
console.warn(
|
|
|
|
"Text I got",
|
|
|
|
text,
|
|
|
|
);
|
2022-11-23 17:05:01 +08:00
|
|
|
console.warn(
|
|
|
|
"Could not find directive in text, skipping",
|
|
|
|
replacement.fullMatch,
|
|
|
|
);
|
|
|
|
continue;
|
|
|
|
}
|
2022-11-24 23:08:51 +08:00
|
|
|
const from = index, to = index + replacement.fullMatch.length;
|
2022-12-15 03:04:20 +08:00
|
|
|
const newText = await replacement.textPromise;
|
|
|
|
if (text.substring(from, to) === newText) {
|
2022-11-24 23:08:51 +08:00
|
|
|
// No change, skip
|
|
|
|
continue;
|
|
|
|
}
|
2022-11-23 17:05:01 +08:00
|
|
|
await editor.dispatch({
|
|
|
|
changes: {
|
2022-11-24 23:08:51 +08:00
|
|
|
from,
|
|
|
|
to,
|
2022-12-15 03:04:20 +08:00
|
|
|
insert: newText,
|
2022-11-23 17:05:01 +08:00
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
2022-10-28 22:17:40 +08:00
|
|
|
}
|
|
|
|
|
2023-08-05 03:25:14 +08:00
|
|
|
export async function updateDirectivesInSpaceCommand() {
|
|
|
|
await editor.flashNotification(
|
|
|
|
"Updating directives in entire space, this can take a while...",
|
|
|
|
);
|
2023-08-12 02:37:13 +08:00
|
|
|
await updateDirectivesInSpace();
|
2023-08-11 01:00:28 +08:00
|
|
|
|
2023-08-12 02:37:13 +08:00
|
|
|
// And notify the user
|
|
|
|
await editor.flashNotification("Updating of all directives completed!");
|
2023-08-11 01:00:28 +08:00
|
|
|
}
|
|
|
|
|
2023-08-28 23:12:15 +08:00
|
|
|
export async function processUpdateQueue(messages: MQMessage[]) {
|
2023-08-11 01:00:28 +08:00
|
|
|
for (const message of messages) {
|
|
|
|
const pageName: string = message.body;
|
|
|
|
console.log("Updating directives in page", pageName);
|
|
|
|
await updateDirectivesForPage(pageName);
|
2023-08-12 02:37:13 +08:00
|
|
|
await mq.ack(directiveUpdateQueueName, message.id);
|
2023-08-11 01:00:28 +08:00
|
|
|
}
|
2023-08-05 03:25:14 +08:00
|
|
|
}
|
|
|
|
|
2023-08-05 00:56:55 +08:00
|
|
|
async function findReplacements(
|
|
|
|
tree: ParseTree,
|
2022-12-15 03:04:20 +08:00
|
|
|
text: string,
|
2023-08-05 00:56:55 +08:00
|
|
|
pageMeta: PageMeta,
|
2022-12-15 03:04:20 +08:00
|
|
|
) {
|
|
|
|
// Collect all directives and their body replacements
|
|
|
|
const replacements: { fullMatch: string; textPromise: Promise<string> }[] =
|
|
|
|
[];
|
|
|
|
|
2023-08-05 00:56:55 +08:00
|
|
|
// Convenience array to wait for all promises to resolve
|
2022-12-15 03:04:20 +08:00
|
|
|
const allPromises: Promise<string>[] = [];
|
|
|
|
|
2023-08-05 00:56:55 +08:00
|
|
|
removeParentPointers(tree);
|
|
|
|
|
2022-12-15 03:04:20 +08:00
|
|
|
traverseTree(tree, (tree) => {
|
|
|
|
if (tree.type !== "Directive") {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
const fullMatch = text.substring(tree.from!, tree.to!);
|
|
|
|
try {
|
2023-08-05 00:56:55 +08:00
|
|
|
const promise = renderDirectives(pageMeta, tree);
|
2022-12-15 03:04:20 +08:00
|
|
|
replacements.push({
|
|
|
|
textPromise: promise,
|
|
|
|
fullMatch,
|
|
|
|
});
|
|
|
|
allPromises.push(promise);
|
|
|
|
} catch (e: any) {
|
|
|
|
replacements.push({
|
|
|
|
fullMatch,
|
|
|
|
textPromise: Promise.resolve(
|
|
|
|
`${renderToText(tree.children![0])}\n**ERROR:** ${e.message}\n${
|
|
|
|
renderToText(tree.children![tree.children!.length - 1])
|
|
|
|
}`,
|
|
|
|
),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
});
|
|
|
|
|
|
|
|
// Wait for all to have processed
|
|
|
|
await Promise.all(allPromises);
|
|
|
|
|
2023-08-05 00:56:55 +08:00
|
|
|
return replacements;
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function updateDirectivesInSpace() {
|
2023-08-12 02:37:13 +08:00
|
|
|
const pages = await space.listPages();
|
|
|
|
await mq.batchSend(directiveUpdateQueueName, pages.map((page) => page.name));
|
|
|
|
|
|
|
|
// Now let's wait for the processing to finish
|
|
|
|
let queueStats = await mq.getQueueStats(directiveUpdateQueueName);
|
|
|
|
while (queueStats.queued > 0 || queueStats.processing > 0) {
|
|
|
|
sleep(1000);
|
|
|
|
queueStats = await mq.getQueueStats(directiveUpdateQueueName);
|
2023-08-05 00:56:55 +08:00
|
|
|
}
|
2023-08-12 02:37:13 +08:00
|
|
|
|
|
|
|
console.log("Done updating directives in space!");
|
2023-08-05 00:56:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
async function updateDirectivesForPage(
|
|
|
|
pageName: string,
|
|
|
|
) {
|
|
|
|
const pageMeta = await space.getPageMeta(pageName);
|
|
|
|
const currentText = await space.readPage(pageName);
|
2023-08-11 01:00:28 +08:00
|
|
|
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);
|
2023-08-05 00:56:55 +08:00
|
|
|
if (newText !== currentText) {
|
2023-08-12 02:37:13 +08:00
|
|
|
console.info("Content of page changed, saving", pageName);
|
2023-08-05 00:56:55 +08:00
|
|
|
await space.writePage(pageName, newText);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function updateDirectives(
|
|
|
|
pageMeta: PageMeta,
|
2023-08-11 01:00:28 +08:00
|
|
|
tree: ParseTree,
|
2023-08-05 00:56:55 +08:00
|
|
|
text: string,
|
|
|
|
) {
|
|
|
|
const replacements = await findReplacements(tree, text, pageMeta);
|
|
|
|
|
2022-12-15 03:04:20 +08:00
|
|
|
// Iterate again and replace the bodies.
|
|
|
|
for (const replacement of replacements) {
|
2023-08-05 00:56:55 +08:00
|
|
|
text = text.replace(
|
|
|
|
replacement.fullMatch,
|
|
|
|
await replacement.textPromise,
|
|
|
|
);
|
2022-12-15 03:04:20 +08:00
|
|
|
}
|
|
|
|
return text;
|
2022-10-28 22:17:40 +08:00
|
|
|
}
|
2023-10-03 20:16:33 +08:00
|
|
|
|
2023-10-11 16:27:12 +08:00
|
|
|
export async function convertToLive() {
|
2023-10-03 20:16:33 +08:00
|
|
|
const text = await editor.getText();
|
|
|
|
const pos = await editor.getCursor();
|
|
|
|
const tree = await markdown.parseMarkdown(text);
|
|
|
|
addParentPointers(tree);
|
|
|
|
const currentNode = nodeAtPos(tree, pos);
|
|
|
|
const directive = findParentMatching(
|
|
|
|
currentNode!,
|
|
|
|
(node) => node.type === "Directive",
|
|
|
|
);
|
|
|
|
if (!directive) {
|
|
|
|
await editor.flashNotification(
|
|
|
|
"No directive found at cursor position",
|
|
|
|
"error",
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
2023-10-11 16:27:12 +08:00
|
|
|
console.log("Got this directive", directive);
|
|
|
|
const startNode = directive.children![0];
|
|
|
|
const startNodeText = renderToText(startNode);
|
|
|
|
if (startNodeText.includes("#query")) {
|
|
|
|
const queryText = renderToText(startNode.children![1]);
|
|
|
|
await editor.dispatch({
|
|
|
|
changes: {
|
|
|
|
from: directive.from,
|
|
|
|
to: directive.to,
|
|
|
|
insert: "```query\n" + queryText + "\n```",
|
|
|
|
},
|
|
|
|
});
|
|
|
|
} else if (
|
|
|
|
startNodeText.includes("#use") || startNodeText.includes("#include")
|
|
|
|
) {
|
|
|
|
const pageRefMatch = /\[\[([^\]]+)\]\]\s*([^\-]+)?/.exec(startNodeText);
|
|
|
|
if (!pageRefMatch) {
|
|
|
|
await editor.flashNotification(
|
|
|
|
"No page reference found in directive",
|
|
|
|
"error",
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const val = pageRefMatch[2];
|
|
|
|
await editor.dispatch({
|
|
|
|
changes: {
|
|
|
|
from: directive.from,
|
|
|
|
to: directive.to,
|
|
|
|
insert: '```template\npage: "[[' + pageRefMatch[1] + ']]"\n' +
|
|
|
|
(val ? `val: ${val}\n` : "") + "```",
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
2023-10-03 20:16:33 +08:00
|
|
|
}
|