silverbullet/plugs/query/data.ts

131 lines
3.4 KiB
TypeScript
Raw Normal View History

// Index key space:
// data:page@pos
import type { IndexTreeEvent } from "../../web/app_event.ts";
2022-04-25 16:33:38 +08:00
import {
batchSet,
queryPrefix,
} from "../../syscall/silverbullet-syscall/index.ts";
2022-04-25 16:33:38 +08:00
import {
2022-07-04 17:30:30 +08:00
addParentPointers,
2022-04-25 16:33:38 +08:00
collectNodesOfType,
findNodeOfType,
ParseTree,
replaceNodesMatching,
} from "../../common/tree.ts";
import type { QueryProviderEvent } from "./engine.ts";
import { applyQuery } from "./engine.ts";
import { removeQueries } from "./util.ts";
import * as YAML from "yaml";
export async function indexData({ name, tree }: IndexTreeEvent) {
let dataObjects: { key: string; value: Object }[] = [];
removeQueries(tree);
collectNodesOfType(tree, "FencedCode").forEach((t) => {
let codeInfoNode = findNodeOfType(t, "CodeInfo");
if (!codeInfoNode) {
return;
}
if (codeInfoNode.children![0].text !== "data") {
return;
}
let codeTextNode = findNodeOfType(t, "CodeText");
if (!codeTextNode) {
// Honestly, this shouldn't happen
return;
}
let codeText = codeTextNode.children![0].text!;
try {
// We support multiple YAML documents in one block
for (let doc of parseAllDocuments(codeText)) {
if (!doc.contents) {
continue;
}
console.log(doc.contents.toJSON());
dataObjects.push({
key: `data:${name}@${t.from! + doc.range[0]}`,
value: doc.contents.toJSON(),
});
}
// console.log("Parsed data", parsedData);
} catch (e) {
console.error("Could not parse data", codeText, "error:", e);
return;
}
});
console.log("Found", dataObjects.length, "data objects");
await batchSet(name, dataObjects);
}
2022-05-07 00:55:04 +08:00
export function extractMeta(
parseTree: ParseTree,
removeKeys: string[] = [],
2022-05-07 00:55:04 +08:00
): any {
let data: any = {};
2022-07-04 17:30:30 +08:00
addParentPointers(parseTree);
replaceNodesMatching(parseTree, (t) => {
2022-07-04 17:30:30 +08: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;
}
let codeInfoNode = findNodeOfType(t, "CodeInfo");
if (!codeInfoNode) {
return;
}
if (codeInfoNode.children![0].text !== "meta") {
return;
}
let codeTextNode = findNodeOfType(t, "CodeText");
if (!codeTextNode) {
// Honestly, this shouldn't happen
return;
}
let codeText = codeTextNode.children![0].text!;
data = YAML.parse(codeText);
2022-05-07 00:55:04 +08:00
if (removeKeys.length > 0) {
let newData = { ...data };
for (let key of removeKeys) {
delete newData[key];
}
codeTextNode.children![0].text = YAML.stringify(newData).trim();
2022-08-08 19:09:19 +08:00
// If nothing is left, let's just delete this thing
if (Object.keys(newData).length === 0) {
return null;
}
2022-05-07 00:55:04 +08:00
}
return undefined;
});
return data;
}
export async function queryProvider({
query,
}: QueryProviderEvent): Promise<any[]> {
let allData: any[] = [];
for (let { key, page, value } of await queryPrefix("data:")) {
let [, pos] = key.split("@");
allData.push({
...value,
page: page,
pos: +pos,
});
}
return applyQuery(query, allData);
}