1
0
silverbullet/plugs/query/data.ts

127 lines
3.3 KiB
TypeScript
Raw Normal View History

// Index key space:
// data:page@pos
2022-10-14 13:11:33 +00:00
import type { IndexTreeEvent, QueryProviderEvent } from "$sb/app_event.ts";
import { index } from "$sb/silverbullet-syscall/mod.ts";
2022-04-25 08:33:38 +00:00
import {
2022-07-04 09:30:30 +00:00
addParentPointers,
2022-04-25 08:33:38 +00:00
collectNodesOfType,
findNodeOfType,
ParseTree,
replaceNodesMatching,
2022-10-14 13:11:33 +00:00
} from "$sb/lib/tree.ts";
import { applyQuery, removeQueries } from "$sb/lib/query.ts";
import * as YAML from "yaml";
export async function indexData({ name, tree }: IndexTreeEvent) {
2022-10-14 13:11:33 +00:00
const dataObjects: { key: string; value: any }[] = [];
removeQueries(tree);
collectNodesOfType(tree, "FencedCode").forEach((t) => {
2022-10-14 13:11:33 +00:00
const codeInfoNode = findNodeOfType(t, "CodeInfo");
if (!codeInfoNode) {
return;
}
if (codeInfoNode.children![0].text !== "data") {
return;
}
2022-10-14 13:11:33 +00:00
const codeTextNode = findNodeOfType(t, "CodeText");
if (!codeTextNode) {
// Honestly, this shouldn't happen
return;
}
2022-10-14 13:11:33 +00:00
const codeText = codeTextNode.children![0].text!;
try {
2022-10-14 13:11:33 +00:00
const docs = codeText.split("---").map((d) => YAML.parse(d));
// We support multiple YAML documents in one block
2022-10-14 13:11:33 +00:00
for (let i = 0; i < docs.length; i++) {
const doc = docs[i];
if (!doc) {
continue;
}
dataObjects.push({
2022-10-14 13:11:33 +00:00
key: `data:${name}@${i}`,
value: doc,
});
}
// console.log("Parsed data", parsedData);
} catch (e) {
console.error("Could not parse data", codeText, "error:", e);
return;
}
});
console.log("Found", dataObjects.length, "data objects");
2022-10-14 13:11:33 +00:00
await index.batchSet(name, dataObjects);
}
2022-05-06 16:55:04 +00:00
export function extractMeta(
parseTree: ParseTree,
removeKeys: string[] = [],
2022-05-06 16:55:04 +00:00
): any {
let data: any = {};
2022-07-04 09:30:30 +00:00
addParentPointers(parseTree);
replaceNodesMatching(parseTree, (t) => {
2022-07-04 09:30:30 +00:00
if (t.type === "Hashtag") {
// Check if if nested directly into a Paragraph
if (t.parent && t.parent.type === "Paragraph") {
let tagname = t.children![0].text;
if (!data.tags) {
data.tags = [];
}
if (!data.tags.includes(tagname)) {
data.tags.push(tagname);
}
}
return;
}
// Find a fenced code block
if (t.type !== "FencedCode") {
return;
}
2022-10-14 13:11:33 +00:00
const codeInfoNode = findNodeOfType(t, "CodeInfo");
if (!codeInfoNode) {
return;
}
if (codeInfoNode.children![0].text !== "meta") {
return;
}
2022-10-14 13:11:33 +00:00
const codeTextNode = findNodeOfType(t, "CodeText");
if (!codeTextNode) {
// Honestly, this shouldn't happen
return;
}
2022-10-14 13:11:33 +00:00
const codeText = codeTextNode.children![0].text!;
data = YAML.parse(codeText);
2022-05-06 16:55:04 +00:00
if (removeKeys.length > 0) {
2022-10-14 13:11:33 +00:00
const newData = { ...data };
for (const key of removeKeys) {
2022-05-06 16:55:04 +00:00
delete newData[key];
}
codeTextNode.children![0].text = YAML.stringify(newData).trim();
2022-08-08 11:09:19 +00:00
// If nothing is left, let's just delete this thing
if (Object.keys(newData).length === 0) {
return null;
}
2022-05-06 16:55:04 +00:00
}
return undefined;
});
return data;
}
export async function queryProvider({
query,
}: QueryProviderEvent): Promise<any[]> {
2022-10-14 13:11:33 +00:00
const allData: any[] = [];
for (const { key, page, value } of await index.queryPrefix("data:")) {
const [, pos] = key.split("@");
allData.push({
...value,
page: page,
pos: +pos,
});
}
return applyQuery(query, allData);
}