silverbullet/plug-api/lib/frontmatter.ts

147 lines
4.0 KiB
TypeScript
Raw Normal View History

import { YAML } from "$sb/plugos-syscall/mod.ts";
2022-11-24 19:04:00 +08:00
import {
addParentPointers,
collectNodesOfType,
2022-11-24 19:04:00 +08:00
ParseTree,
renderToText,
replaceNodesMatchingAsync,
traverseTreeAsync,
2022-11-24 19:04:00 +08:00
} from "$sb/lib/tree.ts";
2023-11-06 16:14:16 +08:00
export type FrontMatter = { tags: string[] } & Record<string, any>;
export type FrontmatterExtractOptions = {
removeKeys?: string[];
removeTags?: string[] | true;
removeFrontmatterSection?: boolean;
};
// Extracts front matter from a markdown document
2022-11-24 19:04:00 +08:00
// optionally removes certain keys from the front matter
export async function extractFrontmatter(
2022-11-24 19:04:00 +08:00
tree: ParseTree,
options: FrontmatterExtractOptions = {},
2023-11-06 16:14:16 +08:00
): Promise<FrontMatter> {
let data: FrontMatter = {
tags: [],
};
2022-11-24 19:04:00 +08:00
addParentPointers(tree);
let paragraphCounter = 0;
2022-11-24 19:04:00 +08:00
await replaceNodesMatchingAsync(tree, async (t) => {
2023-11-06 16:14:16 +08:00
// Find tags in the first paragraph to attach to the page
if (t.type === "Paragraph") {
paragraphCounter++;
// Only attach hashtags in the first paragraph to the page
if (paragraphCounter !== 1) {
return;
}
collectNodesOfType(t, "Hashtag").forEach((h) => {
const tagname = h.children![0].text!.substring(1);
2023-11-06 16:14:16 +08:00
if (!data.tags.includes(tagname)) {
2022-11-24 19:04:00 +08:00
data.tags.push(tagname);
}
if (
options.removeTags === true || options.removeTags?.includes(tagname)
) {
// Ugly hack to remove the hashtag
h.children![0].text = "";
}
});
2022-11-24 19:04:00 +08:00
}
// Find FrontMatter and parse it
if (t.type === "FrontMatter") {
2022-11-24 23:08:51 +08:00
const yamlNode = t.children![1].children![0];
const yamlText = renderToText(yamlNode);
2022-11-24 19:04:00 +08:00
try {
const parsedData: any = await YAML.parse(yamlText);
2022-11-24 19:04:00 +08:00
const newData = { ...parsedData };
data = { ...data, ...parsedData };
2023-11-06 16:14:16 +08:00
// Make sure we have a tags array
if (!data.tags) {
data.tags = [];
}
// Normalize tags to an array and support a "tag1, tag2" notation
if (typeof data.tags === "string") {
data.tags = (data.tags as string).split(/,\s*/);
}
if (options.removeKeys && options.removeKeys.length > 0) {
2022-11-24 19:04:00 +08:00
let removedOne = false;
for (const key of options.removeKeys) {
2022-11-24 19:04:00 +08:00
if (key in newData) {
delete newData[key];
removedOne = true;
}
}
if (removedOne) {
yamlNode.text = await YAML.stringify(newData);
2022-11-24 19:04:00 +08:00
}
}
// If nothing is left, let's just delete this whole block
if (
Object.keys(newData).length === 0 || options.removeFrontmatterSection
) {
2022-11-24 19:04:00 +08:00
return null;
}
} catch (e: any) {
2023-08-02 03:35:19 +08:00
console.warn("Could not parse frontmatter", e.message);
2022-11-24 19:04:00 +08:00
}
}
return undefined;
});
if (data.name) {
data.displayName = data.name;
delete data.name;
}
return data;
}
// Updates the front matter of a markdown document and returns the text as a rendered string
export async function prepareFrontmatterDispatch(
2022-11-24 19:04:00 +08:00
tree: ParseTree,
data: Record<string, any>,
): Promise<any> {
2022-11-24 19:04:00 +08:00
let dispatchData: any = null;
await traverseTreeAsync(tree, async (t) => {
2022-11-24 19:04:00 +08:00
// Find FrontMatter and parse it
if (t.type === "FrontMatter") {
const bodyNode = t.children![1].children![0];
const yamlText = renderToText(bodyNode);
try {
const parsedYaml = await YAML.parse(yamlText) as any;
2022-11-24 19:04:00 +08:00
const newData = { ...parsedYaml, ...data };
// Patch inline
dispatchData = {
changes: {
from: bodyNode.from,
to: bodyNode.to,
insert: await YAML.stringify(newData),
2022-11-24 19:04:00 +08:00
},
};
} catch (e: any) {
console.error("Error parsing YAML", e);
}
return true;
}
return false;
});
if (!dispatchData) {
// If we didn't find frontmatter, let's add it
dispatchData = {
changes: {
from: 0,
to: 0,
insert: "---\n" + await YAML.stringify(data) +
2022-11-24 19:04:00 +08:00
"---\n",
},
};
}
return dispatchData;
}