mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-06 14:20:44 +00:00
ci: validate docs mdx before publish
This commit is contained in:
270
scripts/check-docs-mdx.mjs
Normal file
270
scripts/check-docs-mdx.mjs
Normal file
@@ -0,0 +1,270 @@
|
||||
#!/usr/bin/env node
|
||||
|
||||
import fs from "node:fs";
|
||||
import path from "node:path";
|
||||
import { compile } from "@mdx-js/mdx";
|
||||
|
||||
const MINTLIFY_LANGUAGE_CODES = new Set([
|
||||
"en",
|
||||
"cn",
|
||||
"zh",
|
||||
"zh-Hans",
|
||||
"zh-Hant",
|
||||
"es",
|
||||
"fr",
|
||||
"fr-CA",
|
||||
"fr-ca",
|
||||
"ja",
|
||||
"jp",
|
||||
"ja-jp",
|
||||
"pt",
|
||||
"pt-BR",
|
||||
"de",
|
||||
"ko",
|
||||
"it",
|
||||
"ru",
|
||||
"ro",
|
||||
"cs",
|
||||
"id",
|
||||
"ar",
|
||||
"tr",
|
||||
"hi",
|
||||
"sv",
|
||||
"no",
|
||||
"lv",
|
||||
"nl",
|
||||
"uk",
|
||||
"vi",
|
||||
"pl",
|
||||
"uz",
|
||||
"he",
|
||||
"ca",
|
||||
"fi",
|
||||
"hu",
|
||||
]);
|
||||
|
||||
function parseArgs(argv) {
|
||||
const roots = [];
|
||||
let jsonOut = "";
|
||||
let maxErrors = 50;
|
||||
|
||||
for (let index = 0; index < argv.length; index += 1) {
|
||||
const part = argv[index];
|
||||
if (part === "--json-out") {
|
||||
jsonOut = argv[index + 1] ?? "";
|
||||
index += 1;
|
||||
continue;
|
||||
}
|
||||
if (part === "--max-errors") {
|
||||
maxErrors = Number.parseInt(argv[index + 1] ?? "", 10);
|
||||
index += 1;
|
||||
continue;
|
||||
}
|
||||
if (part.startsWith("--")) {
|
||||
throw new Error(`unknown arg: ${part}`);
|
||||
}
|
||||
roots.push(part);
|
||||
}
|
||||
|
||||
return {
|
||||
roots: roots.length ? roots : ["docs"],
|
||||
jsonOut,
|
||||
maxErrors: Number.isFinite(maxErrors) && maxErrors > 0 ? maxErrors : 50,
|
||||
};
|
||||
}
|
||||
|
||||
function walkMarkdownFiles(entryPath, out = []) {
|
||||
const stat = fs.statSync(entryPath);
|
||||
if (stat.isFile()) {
|
||||
if (/\.mdx?$/i.test(entryPath)) {
|
||||
out.push(path.resolve(entryPath));
|
||||
}
|
||||
return out;
|
||||
}
|
||||
|
||||
for (const entry of fs.readdirSync(entryPath, { withFileTypes: true })) {
|
||||
if (entry.name === "node_modules" || entry.name === ".git") {
|
||||
continue;
|
||||
}
|
||||
walkMarkdownFiles(path.join(entryPath, entry.name), out);
|
||||
}
|
||||
return out;
|
||||
}
|
||||
|
||||
function stripFrontmatter(raw) {
|
||||
if (!raw.startsWith("---\n") && !raw.startsWith("---\r\n")) {
|
||||
return raw;
|
||||
}
|
||||
|
||||
const lines = raw.split(/\r?\n/u);
|
||||
for (let index = 1; index < lines.length; index += 1) {
|
||||
if (lines[index] === "---" || lines[index] === "...") {
|
||||
return lines.slice(index + 1).join("\n");
|
||||
}
|
||||
}
|
||||
return raw;
|
||||
}
|
||||
|
||||
function formatMdxError(filePath, error) {
|
||||
const place = error?.place ?? error?.position;
|
||||
const start = place?.start ?? place;
|
||||
const line = typeof start?.line === "number" ? start.line : undefined;
|
||||
const column = typeof start?.column === "number" ? start.column : undefined;
|
||||
return {
|
||||
type: "mdx",
|
||||
file: filePath,
|
||||
line,
|
||||
column,
|
||||
message: String(error?.reason ?? error?.message ?? error).split("\n")[0],
|
||||
};
|
||||
}
|
||||
|
||||
async function checkMdxFile(filePath) {
|
||||
const raw = fs.readFileSync(filePath, "utf8");
|
||||
const value = stripFrontmatter(raw);
|
||||
await compile(
|
||||
{ path: filePath, value },
|
||||
{
|
||||
development: false,
|
||||
jsx: false,
|
||||
},
|
||||
);
|
||||
}
|
||||
|
||||
function findDocsJsonPaths(roots) {
|
||||
const paths = new Set();
|
||||
for (const root of roots) {
|
||||
const absolute = path.resolve(root);
|
||||
if (!fs.existsSync(absolute)) {
|
||||
continue;
|
||||
}
|
||||
const stat = fs.statSync(absolute);
|
||||
if (stat.isFile() && path.basename(absolute) === "docs.json") {
|
||||
paths.add(absolute);
|
||||
continue;
|
||||
}
|
||||
if (stat.isDirectory()) {
|
||||
const docsJsonPath = path.join(absolute, "docs.json");
|
||||
if (fs.existsSync(docsJsonPath)) {
|
||||
paths.add(docsJsonPath);
|
||||
}
|
||||
}
|
||||
}
|
||||
return [...paths];
|
||||
}
|
||||
|
||||
function collectNavigationLanguages(value, out = []) {
|
||||
if (Array.isArray(value)) {
|
||||
for (const item of value) {
|
||||
collectNavigationLanguages(item, out);
|
||||
}
|
||||
return out;
|
||||
}
|
||||
if (!value || typeof value !== "object") {
|
||||
return out;
|
||||
}
|
||||
if (typeof value.language === "string") {
|
||||
out.push(value.language);
|
||||
}
|
||||
for (const child of Object.values(value)) {
|
||||
if (child && typeof child === "object") {
|
||||
collectNavigationLanguages(child, out);
|
||||
}
|
||||
}
|
||||
return out;
|
||||
}
|
||||
|
||||
function checkDocsJson(filePath) {
|
||||
const errors = [];
|
||||
let data;
|
||||
try {
|
||||
data = JSON.parse(fs.readFileSync(filePath, "utf8"));
|
||||
} catch (error) {
|
||||
return [
|
||||
{
|
||||
type: "docs-json",
|
||||
file: filePath,
|
||||
message: `Invalid JSON: ${String(error?.message ?? error)}`,
|
||||
},
|
||||
];
|
||||
}
|
||||
|
||||
const languages = collectNavigationLanguages(data?.navigation);
|
||||
for (const language of languages) {
|
||||
if (!MINTLIFY_LANGUAGE_CODES.has(language)) {
|
||||
errors.push({
|
||||
type: "docs-json",
|
||||
file: filePath,
|
||||
message: `Unsupported Mintlify navigation language: ${language}`,
|
||||
});
|
||||
}
|
||||
}
|
||||
return errors;
|
||||
}
|
||||
|
||||
function relativize(root, filePath) {
|
||||
const relative = path.relative(root, filePath);
|
||||
return relative && !relative.startsWith("..") ? relative : filePath;
|
||||
}
|
||||
|
||||
async function main() {
|
||||
const startedAt = Date.now();
|
||||
const args = parseArgs(process.argv.slice(2));
|
||||
const cwd = process.cwd();
|
||||
const roots = args.roots.map((root) => path.resolve(root));
|
||||
const files = [
|
||||
...new Set(
|
||||
roots.flatMap((root) => {
|
||||
if (!fs.existsSync(root)) {
|
||||
throw new Error(`path does not exist: ${root}`);
|
||||
}
|
||||
return walkMarkdownFiles(root);
|
||||
}),
|
||||
),
|
||||
].toSorted((left, right) => left.localeCompare(right));
|
||||
|
||||
const errors = [];
|
||||
for (const docsJsonPath of findDocsJsonPaths(args.roots)) {
|
||||
errors.push(...checkDocsJson(docsJsonPath));
|
||||
}
|
||||
|
||||
for (const file of files) {
|
||||
try {
|
||||
await checkMdxFile(file);
|
||||
} catch (error) {
|
||||
errors.push(formatMdxError(file, error));
|
||||
if (errors.length >= args.maxErrors) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
const report = {
|
||||
files: files.length,
|
||||
errors: errors.map((error) => Object.assign({}, error, { file: relativize(cwd, error.file) })),
|
||||
ms: Date.now() - startedAt,
|
||||
};
|
||||
|
||||
if (args.jsonOut) {
|
||||
fs.mkdirSync(path.dirname(path.resolve(args.jsonOut)), { recursive: true });
|
||||
fs.writeFileSync(args.jsonOut, `${JSON.stringify(report, null, 2)}\n`);
|
||||
}
|
||||
|
||||
if (report.errors.length === 0) {
|
||||
console.log(`Docs MDX check passed (${report.files} files, ${report.ms}ms).`);
|
||||
return;
|
||||
}
|
||||
|
||||
console.error(`Docs MDX check failed (${report.errors.length} error(s), ${report.files} files).`);
|
||||
for (const error of report.errors) {
|
||||
const location =
|
||||
error.line && error.column ? `${error.file}:${error.line}:${error.column}` : error.file;
|
||||
console.error(`- ${location}: ${error.message}`);
|
||||
}
|
||||
process.exitCode = 1;
|
||||
}
|
||||
|
||||
main().catch((error) => {
|
||||
console.error(error?.stack ?? error);
|
||||
process.exit(1);
|
||||
});
|
||||
Reference in New Issue
Block a user