import { IndexEvent } from "../../webapp/app_event"; import { pageLinkRegex } from "../../webapp/constant"; import { syscall } from "../lib/syscall"; const wikilinkRegex = new RegExp(pageLinkRegex, "g"); export async function indexLinks({ name, text }: IndexEvent) { let backLinks: { key: string; value: string }[] = []; // [[Style Links]] console.log("Now indexing", name); for (let match of text.matchAll(wikilinkRegex)) { let toPage = match[1]; if (toPage.includes("@")) { toPage = toPage.split("@")[0]; } let pos = match.index!; backLinks.push({ key: `pl:${toPage}:${pos}`, value: name, }); } console.log("Found", backLinks.length, "wiki link(s)"); await syscall("indexer.batchSet", name, backLinks); } export async function deletePage() { let pageName = await syscall("editor.getCurrentPage"); console.log("Navigating to start page"); await syscall("editor.navigate", "start"); console.log("Deleting page from space"); await syscall("space.deletePage", pageName); } export async function renamePage() { const oldName = await syscall("editor.getCurrentPage"); console.log("Old name is", oldName); const newName = await syscall( "editor.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 syscall("editor.getText"); console.log("Writing new page to space"); await syscall("space.writePage", newName, text); console.log("Navigating to new page"); await syscall("editor.navigate", newName); console.log("Deleting page from space"); await syscall("space.deletePage", oldName); let pageToUpdateSet = new Set(); for (let pageToUpdate of pagesToUpdate) { pageToUpdateSet.add(pageToUpdate.page); } for (let pageToUpdate of pageToUpdateSet) { console.log("Now going to update links in", pageToUpdate); let { text } = await syscall("space.readPage", pageToUpdate); console.log("Received text", text); if (!text) { // Page likely does not exist, but at least we can skip it continue; } let newText = text.replaceAll(`[[${oldName}]]`, `[[${newName}]]`); if (text !== newText) { console.log("Changes made, saving..."); await syscall("space.writePage", pageToUpdate, newText); } } } type BackLink = { page: string; pos: number; }; async function getBackLinks(pageName: string): Promise { let allBackLinks = await syscall( "indexer.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 showBackLinks() { const pageName = await syscall("editor.getCurrentPage"); let backLinks = await getBackLinks(pageName); console.log("Backlinks", backLinks); } export async function reindexCommand() { await syscall("editor.flashNotification", "Reindexing..."); await syscall("system.invokeFunctionOnServer", "reindexSpace"); await syscall("editor.flashNotification", "Reindexing done"); } // Completion export async function pageComplete() { let prefix = await syscall("editor.matchBefore", "\\[\\[[\\w\\s]*"); if (!prefix) { return null; } let allPages = await syscall("space.listPages"); return { from: prefix.from + 2, options: allPages.map((pageMeta: any) => ({ label: pageMeta.name, type: "page", })), }; } // Server functions export async function reindexSpace() { console.log("Clearing page index..."); await syscall("indexer.clearPageIndex"); console.log("Listing all pages"); let pages = await syscall("space.listPages"); for (let { name } of pages) { console.log("Indexing", name); const pageObj = await syscall("space.readPage", name); await syscall("event.dispatch", "page:index", { name, text: pageObj.text, }); } } export async function clearPageIndex(page: string) { console.log("Clearing page index for page", page); await syscall("indexer.clearPageIndexForPage", page); }