2024-02-29 22:23:05 +08:00
|
|
|
import type { LintDiagnostic, LintEvent } from "../../plug-api/types.ts";
|
|
|
|
import {
|
|
|
|
findNodeOfType,
|
|
|
|
renderToText,
|
|
|
|
traverseTreeAsync,
|
|
|
|
} from "$sb/lib/tree.ts";
|
2024-01-21 02:16:07 +08:00
|
|
|
import { FrontmatterConfig } from "./types.ts";
|
|
|
|
import { extractFrontmatter } from "$sb/lib/frontmatter.ts";
|
2024-02-03 02:19:07 +08:00
|
|
|
import { template, YAML } from "$sb/syscalls.ts";
|
2024-01-21 02:16:07 +08:00
|
|
|
|
|
|
|
export async function lintTemplateFrontmatter(
|
|
|
|
{ tree }: LintEvent,
|
|
|
|
): Promise<LintDiagnostic[]> {
|
|
|
|
const diagnostics: LintDiagnostic[] = [];
|
|
|
|
const frontmatter = await extractFrontmatter(tree);
|
|
|
|
|
|
|
|
// Just looking this up again for the purposes of error reporting
|
|
|
|
const frontmatterNode = findNodeOfType(tree, "FrontMatterCode")!;
|
|
|
|
if (!frontmatter.tags?.includes("template")) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
// Just parse to make sure it's valid
|
|
|
|
FrontmatterConfig.parse(frontmatter);
|
|
|
|
} catch (e: any) {
|
|
|
|
if (e.message.startsWith("[")) { // We got a zod error
|
|
|
|
const zodErrors = JSON.parse(e.message);
|
|
|
|
for (const zodError of zodErrors) {
|
|
|
|
console.log("Zod validation error", zodError);
|
|
|
|
diagnostics.push({
|
|
|
|
from: frontmatterNode.from!,
|
|
|
|
to: frontmatterNode.to!,
|
|
|
|
message: `Attribute ${zodError.path.join(".")}: ${zodError.message}`,
|
|
|
|
severity: "error",
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
diagnostics.push({
|
|
|
|
from: frontmatterNode.from!,
|
|
|
|
to: frontmatterNode.to!,
|
|
|
|
message: e.message,
|
|
|
|
severity: "error",
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return diagnostics;
|
|
|
|
}
|
2024-02-03 02:19:07 +08:00
|
|
|
|
|
|
|
export async function lintTemplateBlocks(
|
|
|
|
{ tree }: LintEvent,
|
|
|
|
): Promise<LintDiagnostic[]> {
|
|
|
|
const frontmatter = await extractFrontmatter(tree);
|
|
|
|
const diagnostics: LintDiagnostic[] = [];
|
|
|
|
|
|
|
|
if (frontmatter.tags?.includes("template")) {
|
|
|
|
// Parse the whole page as a template to check for errors if this is a template
|
|
|
|
try {
|
|
|
|
await template.parseTemplate(renderToText(tree));
|
|
|
|
} catch (e: any) {
|
|
|
|
diagnostics.push({
|
|
|
|
from: tree.from!,
|
|
|
|
to: tree.to!,
|
|
|
|
message: e.message,
|
|
|
|
severity: "error",
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
await traverseTreeAsync(tree, async (node) => {
|
|
|
|
if (node.type === "FencedCode") {
|
|
|
|
const codeInfo = findNodeOfType(node, "CodeInfo")!;
|
|
|
|
if (!codeInfo) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
const codeLang = codeInfo.children![0].text!;
|
|
|
|
if (codeLang !== "template") {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
const codeText = findNodeOfType(node, "CodeText");
|
|
|
|
if (!codeText) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
const bodyText = renderToText(codeText);
|
|
|
|
// See if it parses as YAML, then issue a warning
|
|
|
|
try {
|
|
|
|
const parsedYaml = await YAML.parse(bodyText);
|
|
|
|
if (
|
|
|
|
typeof parsedYaml === "object" &&
|
|
|
|
(parsedYaml.template || parsedYaml.page || parsedYaml.raw)
|
|
|
|
) {
|
|
|
|
diagnostics.push({
|
|
|
|
from: codeText.from!,
|
|
|
|
to: codeText.to!,
|
|
|
|
message:
|
|
|
|
"Legacy template syntax detected, please replace ```template with ```include to fix.",
|
|
|
|
severity: "warning",
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} catch {
|
|
|
|
// Ignore
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ok, now parse it as a template and report any parse errors
|
|
|
|
try {
|
|
|
|
await template.parseTemplate(bodyText);
|
|
|
|
} catch (e: any) {
|
|
|
|
diagnostics.push({
|
|
|
|
from: codeText.from!,
|
|
|
|
to: codeText.to!,
|
|
|
|
message: e.message,
|
|
|
|
severity: "error",
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
|
|
|
|
return diagnostics;
|
|
|
|
}
|