silverbullet/plugs/search/search.ts

117 lines
3.0 KiB
TypeScript
Raw Normal View History

import { IndexTreeEvent, QueryProviderEvent } from "$sb/app_event.ts";
2022-10-14 21:11:33 +08:00
import { renderToText } from "$sb/lib/tree.ts";
import {
applyQuery,
evalQueryExpression,
liftAttributeFilter,
} from "$sb/lib/query.ts";
import { datastore, editor } from "$sb/syscalls.ts";
import { SimpleSearchEngine } from "./engine.ts";
import { FileMeta, KvKey } from "$sb/types.ts";
2023-08-30 03:17:29 +08:00
import { PromiseQueue } from "$sb/lib/async.ts";
2022-05-16 21:09:36 +08:00
const searchPrefix = "🔍 ";
const engine = new SimpleSearchEngine(datastore);
2023-08-30 03:17:29 +08:00
// Search indexing is prone to concurrency issues, so we queue all write operations
const promiseQueue = new PromiseQueue();
export function indexPage({ name, tree }: IndexTreeEvent) {
const text = renderToText(tree);
2023-08-30 03:17:29 +08:00
return promiseQueue.runInQueue(async () => {
// console.log("Now FTS indexing", name);
await engine.deleteDocument(name);
await engine.indexDocument({ id: name, text });
});
}
export async function clearIndex() {
const keysToDelete: KvKey[] = [];
for (const { key } of await datastore.query({ prefix: ["fts"] })) {
keysToDelete.push(key);
}
for (const { key } of await datastore.query({ prefix: ["fts_rev"] })) {
keysToDelete.push(key);
}
await datastore.batchDel(keysToDelete);
2022-05-16 21:09:36 +08:00
}
2023-08-30 03:17:29 +08:00
export function pageUnindex(pageName: string) {
return promiseQueue.runInQueue(() => {
return engine.deleteDocument(pageName);
});
2022-06-28 20:14:15 +08:00
}
2022-05-16 21:09:36 +08:00
export async function queryProvider({
query,
}: QueryProviderEvent): Promise<any[]> {
const phraseFilter = liftAttributeFilter(query.filter, "phrase");
2022-05-16 21:09:36 +08:00
if (!phraseFilter) {
throw Error("No 'phrase' filter specified, this is mandatory");
}
const phrase = evalQueryExpression(phraseFilter, {});
// console.log("Phrase", phrase);
let results: any[] = await engine.search(phrase);
// Patch the object to a format that users expect (translate id to name)
for (const r of results) {
r.name = r.id;
delete r.id;
}
2022-05-16 21:09:36 +08:00
results = applyQuery(query, results);
return results;
}
2022-05-17 17:53:17 +08:00
export async function searchCommand() {
2022-10-19 15:52:29 +08:00
const phrase = await editor.prompt("Search for: ");
2022-05-17 17:53:17 +08:00
if (phrase) {
2022-10-14 21:11:33 +08:00
await editor.navigate(`${searchPrefix}${phrase}`);
2022-05-17 17:53:17 +08:00
}
}
2022-10-19 15:52:29 +08:00
export async function readFileSearch(
2022-10-12 17:47:13 +08:00
name: string,
): Promise<{ data: Uint8Array; meta: FileMeta }> {
2022-10-19 15:52:29 +08:00
const phrase = name.substring(
searchPrefix.length,
name.length - ".md".length,
);
const results = await engine.search(phrase);
2022-10-12 17:47:13 +08:00
const text = `# Search results for "${phrase}"\n${
results
.map((r) => `* [[${r.id}]] (score ${r.score})`)
.join("\n")
2022-10-12 17:47:13 +08:00
}
`;
2022-10-19 15:52:29 +08:00
2022-05-17 17:53:17 +08:00
return {
data: new TextEncoder().encode(text),
2022-05-17 17:53:17 +08:00
meta: {
name,
2022-10-19 15:52:29 +08:00
contentType: "text/markdown",
size: text.length,
2022-05-17 17:53:17 +08:00
lastModified: 0,
perm: "ro",
},
};
}
export function writeFileSearch(
name: string,
): FileMeta {
// Never actually writing this
return getFileMetaSearch(name);
}
2022-10-19 15:52:29 +08:00
export function getFileMetaSearch(name: string): FileMeta {
2022-05-17 17:53:17 +08:00
return {
name,
2022-10-19 15:52:29 +08:00
contentType: "text/markdown",
size: -1,
2022-05-17 17:53:17 +08:00
lastModified: 0,
perm: "ro",
};
}