silverbullet/plugs/core/page.ts

373 lines
10 KiB
TypeScript
Raw Normal View History

2022-10-14 21:11:33 +08:00
import type {
CompleteEvent,
2022-10-14 21:11:33 +08:00
IndexEvent,
IndexTreeEvent,
QueryProviderEvent,
} from "$sb/app_event.ts";
2022-04-01 23:07:08 +08:00
import {
2022-10-14 21:11:33 +08:00
editor,
index,
markdown,
space,
} from "$sb/silverbullet-syscall/mod.ts";
2022-10-16 01:02:56 +08:00
import { events } from "$sb/plugos-syscall/mod.ts";
import {
addParentPointers,
findNodeOfType,
2022-04-12 02:34:09 +08:00
ParseTree,
renderToText,
2022-04-25 16:33:38 +08:00
replaceNodesMatching,
traverseTree,
2022-10-14 21:11:33 +08:00
} from "$sb/lib/tree.ts";
import { applyQuery } from "$sb/lib/query.ts";
2022-11-24 19:04:00 +08:00
import { extractFrontmatter } from "$sb/lib/frontmatter.ts";
import { invokeFunction } from "$sb/silverbullet-syscall/system.ts";
import { isValidPageName } from "$sb/lib/page.ts";
2023-07-25 01:54:31 +08:00
import { extractAttributes } from "$sb/lib/attribute.ts";
// Key space:
// l:toPage:pos => {name: pageName, inDirective: true}
// meta: => metaJson
2022-02-28 21:35:51 +08:00
const backlinkPrefix = `l:`;
export type BacklinkEntry = {
name: string;
alias?: string;
inDirective?: boolean;
};
export async function indexLinks({ name, tree }: IndexTreeEvent) {
const backLinks: { key: string; value: BacklinkEntry }[] = [];
2022-03-14 17:07:38 +08:00
// [[Style Links]]
// console.log("Now indexing links for", name);
const pageMeta = await extractFrontmatter(tree);
2023-07-26 23:12:56 +08:00
const toplevelAttributes = await extractAttributes(tree, false);
2023-07-25 01:54:31 +08:00
if (
Object.keys(pageMeta).length > 0 ||
Object.keys(toplevelAttributes).length > 0
) {
for (const [k, v] of Object.entries(toplevelAttributes)) {
pageMeta[k] = v;
}
// Don't index meta data starting with $
2022-10-14 21:11:33 +08:00
for (const key in pageMeta) {
if (key.startsWith("$")) {
delete pageMeta[key];
}
}
2023-07-25 23:33:07 +08:00
// console.log("Extracted page meta data", pageMeta);
2022-10-14 21:11:33 +08:00
await index.set(name, "meta:", pageMeta);
}
let directiveDepth = 0;
traverseTree(tree, (n): boolean => {
if (n.type === "DirectiveStart") {
directiveDepth++;
return true;
2022-03-28 21:25:05 +08:00
}
if (n.type === "DirectiveStop") {
directiveDepth--;
return true;
}
if (n.type === "WikiLink") {
const wikiLinkPage = findNodeOfType(n, "WikiLinkPage")!;
const wikiLinkAlias = findNodeOfType(n, "WikiLinkAlias");
let toPage = wikiLinkPage.children![0].text!;
if (toPage.includes("@")) {
toPage = toPage.split("@")[0];
}
const blEntry: BacklinkEntry = { name };
if (directiveDepth > 0) {
blEntry.inDirective = true;
}
if (wikiLinkAlias) {
blEntry.alias = wikiLinkAlias.children![0].text!;
}
backLinks.push({
key: `${backlinkPrefix}${toPage}:${wikiLinkPage.from}`,
value: blEntry,
});
return true;
}
return false;
});
// console.log("Found", backLinks.length, "wiki link(s)");
2022-10-14 21:11:33 +08:00
await index.batchSet(name, backLinks);
2022-02-28 21:35:51 +08:00
}
export async function pageQueryProvider({
query,
}: QueryProviderEvent): Promise<any[]> {
return applyQuery(query, await space.listPages());
}
export async function linkQueryProvider({
query,
pageName,
}: QueryProviderEvent): Promise<any[]> {
2022-10-14 21:11:33 +08:00
const links: any[] = [];
for (
const { value: blEntry, key } of await index.queryPrefix(
`${backlinkPrefix}${pageName}:`,
)
2022-10-14 21:11:33 +08:00
) {
const [, , pos] = key.split(":"); // Key: l:page:pos
if (!blEntry.inDirective) {
blEntry.inDirective = false;
}
links.push({ ...blEntry, pos });
}
2022-08-09 21:37:47 +08:00
return applyQuery(query, links);
}
2022-02-28 21:35:51 +08:00
export async function deletePage() {
2022-10-14 21:11:33 +08:00
const pageName = await editor.getCurrentPage();
if (
!await editor.confirm(`Are you sure you would like to delete ${pageName}?`)
) {
return;
}
2022-06-28 20:14:15 +08:00
console.log("Navigating to index page");
2022-10-14 21:11:33 +08:00
await editor.navigate("");
2022-02-28 21:35:51 +08:00
console.log("Deleting page from space");
2022-10-14 21:11:33 +08:00
await space.deletePage(pageName);
2022-02-28 21:35:51 +08:00
}
2023-06-14 15:20:15 +08:00
export async function copyPage() {
const oldName = await editor.getCurrentPage();
const newName = await editor.prompt(`New page title:`, `${oldName} (copy)`);
if (!newName) {
return;
}
try {
// This throws an error if the page does not exist, which we expect to be the case
await space.getPageMeta(newName);
// So when we get to this point, we error out
throw new Error(
`Page ${newName} already exists, cannot rename to existing page.`,
2023-06-14 15:20:15 +08:00
);
} catch (e: any) {
if (e.message === "Not found") {
// Expected not found error, so we can continue
} else {
await editor.flashNotification(e.message, "error");
throw e;
}
}
const text = await editor.getText();
console.log("Writing new page to space");
await space.writePage(newName, text);
console.log("Navigating to new page");
await editor.navigate(newName);
}
2022-12-13 16:49:31 +08:00
export async function renamePage(cmdDef: any) {
console.log("Got a target name", cmdDef.page);
2022-10-14 21:11:33 +08:00
const oldName = await editor.getCurrentPage();
const cursor = await editor.getCursor();
2022-03-03 17:35:32 +08:00
console.log("Old name is", oldName);
2022-12-13 16:49:31 +08:00
const newName = cmdDef.page ||
await editor.prompt(`Rename ${oldName} to:`, oldName);
2022-02-28 21:35:51 +08:00
if (!newName) {
return;
}
if (!isValidPageName(newName)) {
return editor.flashNotification(
2023-07-07 17:44:05 +08:00
"Invalid page name: page names cannot end with a file extension nor start with a '.'",
"error",
);
}
console.log("New name", newName);
if (newName.trim() === oldName.trim()) {
// Nothing to do here
console.log("Name unchanged, exiting");
return;
}
try {
// This throws an error if the page does not exist, which we expect to be the case
await space.getPageMeta(newName);
// So when we get to this point, we error out
throw new Error(
`Page ${newName} already exists, cannot rename to existing page.`,
);
} catch (e: any) {
if (e.message === "Not found") {
// Expected not found error, so we can continue
} else {
await editor.flashNotification(e.message, "error");
throw e;
}
}
2022-02-28 21:35:51 +08:00
2022-10-14 21:11:33 +08:00
const pagesToUpdate = await getBackLinks(oldName);
2022-02-28 21:35:51 +08:00
console.log("All pages containing backlinks", pagesToUpdate);
2022-10-14 21:11:33 +08:00
const text = await editor.getText();
2022-02-28 21:35:51 +08:00
console.log("Writing new page to space");
const newPageMeta = await space.writePage(newName, text);
2022-02-28 21:35:51 +08:00
console.log("Navigating to new page");
2022-10-14 21:11:33 +08:00
await editor.navigate(newName, cursor, true);
// Handling the edge case of a changing page name just in casing on a case insensitive FS
const oldPageMeta = await space.getPageMeta(oldName);
if (oldPageMeta.lastModified !== newPageMeta.lastModified) {
// If they're the same, let's assume it's the same file (case insensitive FS) and not delete, otherwise...
console.log("Deleting page from space");
await space.deletePage(oldName);
}
2022-02-28 21:35:51 +08:00
2022-10-14 21:11:33 +08:00
const pageToUpdateSet = new Set<string>();
for (const pageToUpdate of pagesToUpdate) {
2022-02-28 21:35:51 +08:00
pageToUpdateSet.add(pageToUpdate.page);
}
let updatedReferences = 0;
2022-10-14 21:11:33 +08:00
for (const pageToUpdate of pageToUpdateSet) {
2022-04-10 17:04:07 +08:00
if (pageToUpdate === oldName) {
continue;
}
2022-02-28 21:35:51 +08:00
console.log("Now going to update links in", pageToUpdate);
2022-10-14 21:11:33 +08:00
const text = await space.readPage(pageToUpdate);
// console.log("Received text", text);
2022-02-28 21:35:51 +08:00
if (!text) {
// Page likely does not exist, but at least we can skip it
continue;
}
2022-10-14 21:11:33 +08:00
const mdTree = await markdown.parseMarkdown(text);
addParentPointers(mdTree);
// The links in the page are going to be relative pointers to the old name
2022-04-12 02:34:09 +08:00
replaceNodesMatching(mdTree, (n): ParseTree | undefined | null => {
if (n.type === "WikiLinkPage") {
2022-10-14 21:11:33 +08:00
const pageName = n.children![0].text!;
if (pageName === oldName) {
n.children![0].text = newName;
updatedReferences++;
return n;
}
// page name with @pos position
if (pageName.startsWith(`${oldName}@`)) {
2022-10-14 21:11:33 +08:00
const [, pos] = pageName.split("@");
n.children![0].text = `${newName}@${pos}`;
updatedReferences++;
return n;
}
}
return;
});
// let newText = text.replaceAll(`[[${oldName}]]`, `[[${newName}]]`);
2022-10-14 21:11:33 +08:00
const newText = renderToText(mdTree);
2022-02-28 21:35:51 +08:00
if (text !== newText) {
console.log("Changes made, saving...");
2022-10-14 21:11:33 +08:00
await space.writePage(pageToUpdate, newText);
2022-02-28 21:35:51 +08:00
}
}
await editor.flashNotification(
`Renamed page, and updated ${updatedReferences} references`,
);
}
export async function newPageCommand() {
const allPages = await space.listPages();
let pageName = `Untitled`;
let i = 1;
while (allPages.find((p) => p.name === pageName)) {
pageName = `Untitled ${i}`;
i++;
}
await editor.navigate(pageName);
2022-02-28 21:35:51 +08:00
}
type BackLink = {
page: string;
pos: number;
};
async function getBackLinks(pageName: string): Promise<BackLink[]> {
const allBackLinks = await index.queryPrefix(
`${backlinkPrefix}${pageName}:`,
);
2022-10-14 21:11:33 +08:00
const pagesToUpdate: BackLink[] = [];
for (const { key, value: { name } } of allBackLinks) {
2022-10-14 21:11:33 +08:00
const keyParts = key.split(":");
2022-02-28 21:35:51 +08:00
pagesToUpdate.push({
page: name,
2022-02-28 21:35:51 +08:00
pos: +keyParts[keyParts.length - 1],
});
}
return pagesToUpdate;
}
2022-03-28 21:25:05 +08:00
export async function reindexCommand() {
2022-10-14 21:11:33 +08:00
await editor.flashNotification("Reindexing...");
await reindexSpace();
2022-10-14 21:11:33 +08:00
await editor.flashNotification("Reindexing done");
2022-03-28 21:25:05 +08:00
}
2022-03-29 18:13:46 +08:00
// Completion
export async function pageComplete(completeEvent: CompleteEvent) {
const match = /\[\[([^\]@:\{}]*)$/.exec(completeEvent.linePrefix);
if (!match) {
2022-03-29 18:13:46 +08:00
return null;
}
2022-10-14 21:11:33 +08:00
const allPages = await space.listPages();
2022-03-29 18:13:46 +08:00
return {
from: completeEvent.pos - match[1].length,
options: allPages.map((pageMeta) => {
return {
label: pageMeta.name,
boost: pageMeta.lastModified,
type: "page",
};
}),
2022-03-29 18:13:46 +08:00
};
}
2022-03-28 21:25:05 +08:00
export async function reindexSpace() {
console.log("Clearing page index...");
2022-10-14 21:11:33 +08:00
await index.clearPageIndex();
// Executed this way to not have to embed the search plug code here
await invokeFunction("client", "search.clearIndex");
2022-03-28 21:25:05 +08:00
console.log("Listing all pages");
2022-10-14 21:11:33 +08:00
const pages = await space.listPages();
let counter = 0;
2022-10-14 21:11:33 +08:00
for (const { name } of pages) {
counter++;
console.log(`Indexing page ${counter}/${pages.length}: ${name}`);
2022-10-14 21:11:33 +08:00
const text = await space.readPage(name);
const parsed = await markdown.parseMarkdown(text);
await events.dispatchEvent("page:index", {
2022-03-28 21:25:05 +08:00
name,
tree: parsed,
2022-03-28 21:25:05 +08:00
});
}
console.log("Indexing completed!");
2022-03-28 21:25:05 +08:00
}
export async function clearPageIndex(page: string) {
2022-10-21 22:56:46 +08:00
// console.log("Clearing page index for page", page);
2022-10-14 21:11:33 +08:00
await index.clearPageIndexForPage(page);
2022-02-28 21:35:51 +08:00
}
2022-04-09 20:28:41 +08:00
export async function parseIndexTextRepublish({ name, text }: IndexEvent) {
// console.log("Reindexing", name);
2022-10-14 21:11:33 +08:00
await events.dispatchEvent("page:index", {
name,
2022-10-14 21:11:33 +08:00
tree: await markdown.parseMarkdown(text),
});
}