254 lines
7.0 KiB
TypeScript
254 lines
7.0 KiB
TypeScript
import { IndexEvent, IndexTreeEvent } from "../../webapp/app_event";
|
|
import {
|
|
batchSet,
|
|
clearPageIndex as clearPageIndexSyscall,
|
|
clearPageIndexForPage,
|
|
scanPrefixGlobal,
|
|
set
|
|
} from "plugos-silverbullet-syscall/index";
|
|
import {
|
|
flashNotification,
|
|
getCurrentPage,
|
|
getText,
|
|
matchBefore,
|
|
navigate,
|
|
prompt
|
|
} from "plugos-silverbullet-syscall/editor";
|
|
|
|
import { dispatch } from "plugos-syscall/event";
|
|
import { deletePage as deletePageSyscall, listPages, readPage, writePage } from "plugos-silverbullet-syscall/space";
|
|
import { invokeFunction } from "plugos-silverbullet-syscall/system";
|
|
import { parseMarkdown } from "plugos-silverbullet-syscall/markdown";
|
|
import {
|
|
addParentPointers,
|
|
collectNodesMatching,
|
|
ParseTree,
|
|
renderToText,
|
|
replaceNodesMatching
|
|
} from "../../common/tree";
|
|
import { applyQuery, QueryProviderEvent } from "../query/engine";
|
|
import { PageMeta } from "../../common/types";
|
|
import { extractMeta } from "../query/data";
|
|
import { jsonToMDTable } from "../query/util";
|
|
|
|
// Key space:
|
|
// pl:toPage:pos => pageName
|
|
// meta => metaJson
|
|
|
|
export async function indexLinks({ name, tree }: IndexTreeEvent) {
|
|
let backLinks: { key: string; value: string }[] = [];
|
|
// [[Style Links]]
|
|
console.log("Now indexing", name);
|
|
let pageMeta = extractMeta(tree);
|
|
if (Object.keys(pageMeta).length > 0) {
|
|
await set(name, "meta:", pageMeta);
|
|
}
|
|
|
|
collectNodesMatching(tree, (n) => n.type === "WikiLinkPage").forEach((n) => {
|
|
let toPage = n.children![0].text!;
|
|
if (toPage.includes("@")) {
|
|
toPage = toPage.split("@")[0];
|
|
}
|
|
backLinks.push({
|
|
key: `pl:${toPage}:${n.from}`,
|
|
value: name,
|
|
});
|
|
});
|
|
console.log("Found", backLinks.length, "wiki link(s)");
|
|
await batchSet(name, backLinks);
|
|
}
|
|
|
|
export async function pageQueryProvider({
|
|
query,
|
|
}: QueryProviderEvent): Promise<string> {
|
|
let allPages = await listPages();
|
|
if (query.select) {
|
|
let allPageMap: Map<string, any> = new Map(
|
|
allPages.map((pm) => [pm.name, pm])
|
|
);
|
|
for (let { page, value } of await scanPrefixGlobal("meta:")) {
|
|
let p = allPageMap.get(page);
|
|
if (p) {
|
|
for (let [k, v] of Object.entries(value)) {
|
|
p[k] = v;
|
|
}
|
|
}
|
|
}
|
|
allPages = [...allPageMap.values()];
|
|
return jsonToMDTable(applyQuery(query, allPages), (k, v) =>
|
|
k === "name" ? `[[${v}]]` : v
|
|
);
|
|
} else {
|
|
return applyQuery(query, allPages)
|
|
.map((pageMeta: PageMeta) => `* [[${pageMeta.name}]]`)
|
|
.join("\n");
|
|
}
|
|
}
|
|
|
|
export async function linkQueryProvider({
|
|
query,
|
|
pageName,
|
|
}: QueryProviderEvent): Promise<string> {
|
|
let uniqueLinks = new Set<string>();
|
|
for (let { value: name } of await scanPrefixGlobal(`pl:${pageName}:`)) {
|
|
uniqueLinks.add(name);
|
|
}
|
|
let markdownLinks = applyQuery(
|
|
query,
|
|
[...uniqueLinks].map((l) => ({ name: l }))
|
|
).map((pageMeta) => `* [[${pageMeta.name}]]`);
|
|
return markdownLinks.join("\n");
|
|
}
|
|
|
|
export async function deletePage() {
|
|
let pageName = await getCurrentPage();
|
|
console.log("Navigating to start page");
|
|
await navigate("start");
|
|
console.log("Deleting page from space");
|
|
await deletePageSyscall(pageName);
|
|
}
|
|
|
|
export async function renamePage() {
|
|
const oldName = await getCurrentPage();
|
|
console.log("Old name is", oldName);
|
|
const newName = await prompt(`Rename ${oldName} to:`, oldName);
|
|
if (!newName) {
|
|
return;
|
|
}
|
|
console.log("New name", newName);
|
|
|
|
let pagesToUpdate = await getBackLinks(oldName);
|
|
console.log("All pages containing backlinks", pagesToUpdate);
|
|
|
|
let text = await getText();
|
|
console.log("Writing new page to space");
|
|
await writePage(newName, text);
|
|
console.log("Navigating to new page");
|
|
await navigate(newName);
|
|
console.log("Deleting page from space");
|
|
await deletePageSyscall(oldName);
|
|
|
|
let pageToUpdateSet = new Set<string>();
|
|
for (let pageToUpdate of pagesToUpdate) {
|
|
pageToUpdateSet.add(pageToUpdate.page);
|
|
}
|
|
|
|
for (let pageToUpdate of pageToUpdateSet) {
|
|
if (pageToUpdate === oldName) {
|
|
continue;
|
|
}
|
|
console.log("Now going to update links in", pageToUpdate);
|
|
let { text } = await readPage(pageToUpdate);
|
|
// console.log("Received text", text);
|
|
if (!text) {
|
|
// Page likely does not exist, but at least we can skip it
|
|
continue;
|
|
}
|
|
let mdTree = await parseMarkdown(text);
|
|
addParentPointers(mdTree);
|
|
replaceNodesMatching(mdTree, (n): ParseTree | undefined | null => {
|
|
if (n.type === "WikiLinkPage") {
|
|
let pageName = n.children![0].text!;
|
|
if (pageName === oldName) {
|
|
n.children![0].text = newName;
|
|
return n;
|
|
}
|
|
// page name with @pos position
|
|
if (pageName.startsWith(`${oldName}@`)) {
|
|
let [, pos] = pageName.split("@");
|
|
n.children![0].text = `${newName}@${pos}`;
|
|
return n;
|
|
}
|
|
}
|
|
return;
|
|
});
|
|
// let newText = text.replaceAll(`[[${oldName}]]`, `[[${newName}]]`);
|
|
let newText = renderToText(mdTree);
|
|
if (text !== newText) {
|
|
console.log("Changes made, saving...");
|
|
await writePage(pageToUpdate, newText);
|
|
}
|
|
}
|
|
}
|
|
|
|
type BackLink = {
|
|
page: string;
|
|
pos: number;
|
|
};
|
|
|
|
async function getBackLinks(pageName: string): Promise<BackLink[]> {
|
|
let allBackLinks = await scanPrefixGlobal(`pl:${pageName}:`);
|
|
let pagesToUpdate: BackLink[] = [];
|
|
for (let { key, value } of allBackLinks) {
|
|
let keyParts = key.split(":");
|
|
pagesToUpdate.push({
|
|
page: value,
|
|
pos: +keyParts[keyParts.length - 1],
|
|
});
|
|
}
|
|
return pagesToUpdate;
|
|
}
|
|
|
|
export async function reindexCommand() {
|
|
await flashNotification("Reindexing...");
|
|
await invokeFunction("server", "reindexSpace");
|
|
await flashNotification("Reindexing done");
|
|
}
|
|
|
|
// Completion
|
|
export async function pageComplete() {
|
|
let prefix = await matchBefore("\\[\\[[\\w\\s]*");
|
|
if (!prefix) {
|
|
return null;
|
|
}
|
|
let allPages = await listPages();
|
|
return {
|
|
from: prefix.from + 2,
|
|
options: allPages.map((pageMeta) => ({
|
|
label: pageMeta.name,
|
|
type: "page",
|
|
})),
|
|
};
|
|
}
|
|
|
|
// Server functions
|
|
export async function reindexSpace() {
|
|
console.log("Clearing page index...");
|
|
await clearPageIndexSyscall();
|
|
console.log("Listing all pages");
|
|
let pages = await listPages();
|
|
for (let { name } of pages) {
|
|
console.log("Indexing", name);
|
|
const { text } = await readPage(name);
|
|
let parsed = await parseMarkdown(text);
|
|
await dispatch("page:index", {
|
|
name,
|
|
tree: parsed,
|
|
});
|
|
}
|
|
}
|
|
|
|
export async function clearPageIndex(page: string) {
|
|
console.log("Clearing page index for page", page);
|
|
await clearPageIndexForPage(page);
|
|
}
|
|
|
|
export async function parseIndexTextRepublish({ name, text }: IndexEvent) {
|
|
await dispatch("page:index", {
|
|
name,
|
|
tree: await parseMarkdown(text),
|
|
});
|
|
}
|
|
|
|
export async function parseServerPageCommand() {
|
|
console.log(await invokeFunction("server", "parsePage", await getText()));
|
|
}
|
|
|
|
export async function parsePageCommand() {
|
|
parsePage(await getText());
|
|
}
|
|
|
|
export async function parsePage(text: string) {
|
|
console.log("AST", JSON.stringify(await parseMarkdown(text), null, 2));
|
|
}
|