Files
openclaw/scripts/format-docs.mjs
Vincent Koc add00d747b build(docs): finish PowerShell-safe docs formatting (#93512)
Co-authored-by: openclaw-clownfish[bot] <280122609+openclaw-clownfish[bot]@users.noreply.github.com>
Co-authored-by: yil337 <220073147+yil337@users.noreply.github.com>
2026-06-16 14:19:01 +08:00

269 lines
7.9 KiB
JavaScript

#!/usr/bin/env node
// Formats docs Markdown/MDX and repairs Mintlify accordion indentation.
import { spawnSync } from "node:child_process";
import fs from "node:fs";
import os from "node:os";
import path from "node:path";
import { pathToFileURL } from "node:url";
import { repairMintlifyAccordionIndentation } from "./lib/mintlify-accordion.mjs";
import { buildCmdExeCommandLine } from "./windows-cmd-helpers.mjs";
const ROOT = path.resolve(import.meta.dirname, "..");
const CHECK = process.argv.includes("--check");
const DOCS_FORMAT_MAX_BUFFER_BYTES = 1024 * 1024 * 16;
export const DOCS_FORMAT_MAX_COMMAND_LINE_BYTES = 24 * 1024;
const FAILURE_OUTPUT_TAIL_BYTES = 16 * 1024;
function outputText(value) {
if (typeof value === "string") {
return value;
}
if (Buffer.isBuffer(value)) {
return value.toString("utf8");
}
return "";
}
function outputTail(value) {
const text = outputText(value).trim();
if (!text) {
return "";
}
const bytes = Buffer.from(text, "utf8");
if (bytes.byteLength <= FAILURE_OUTPUT_TAIL_BYTES) {
return text;
}
return bytes.subarray(bytes.byteLength - FAILURE_OUTPUT_TAIL_BYTES).toString("utf8");
}
function commandFailureMessage(label, result, invocation) {
const details = [];
if (invocation) {
details.push(`command: ${invocation.command}`);
if (invocation.args.length > 0) {
const previewArgs = invocation.args.slice(0, 12).join(" ");
const suffix = invocation.args.length > 12 ? ` ... (${invocation.args.length} args)` : "";
details.push(`args: ${previewArgs}${suffix}`);
}
}
if (result.error?.message) {
details.push(result.error.message);
}
if (result.status !== null && result.status !== undefined && result.status !== 0) {
details.push(`exit status: ${result.status}`);
}
if (result.signal) {
details.push(`signal: ${result.signal}`);
}
const stderrTail = outputTail(result.stderr);
if (stderrTail) {
details.push(`stderr tail:\n${stderrTail}`);
}
const stdoutTail = outputTail(result.stdout);
if (stdoutTail) {
details.push(`stdout tail:\n${stdoutTail}`);
}
return `${label} failed${details.length > 0 ? `:\n${details.join("\n")}` : ""}`;
}
export function docsFiles(root = ROOT, deps = {}) {
const spawnSyncImpl = deps.spawnSync ?? spawnSync;
const result = spawnSyncImpl("git", ["ls-files", "docs/**/*.md", "docs/**/*.mdx", "README.md"], {
cwd: root,
encoding: "utf8",
maxBuffer: DOCS_FORMAT_MAX_BUFFER_BYTES,
});
if (result.error || result.status !== 0) {
throw new Error(
commandFailureMessage("git ls-files", result, {
command: "git",
args: ["ls-files", "docs/**/*.md", "docs/**/*.mdx", "README.md"],
}),
);
}
return outputText(result.stdout)
.split("\n")
.filter(Boolean)
.filter((relativePath) => (deps.existsSync ?? fs.existsSync)(path.join(root, relativePath)));
}
function commandLineBytes(args) {
return args.reduce((total, arg) => total + Buffer.byteLength(arg, "utf8") + 3, 0);
}
export function chunkFilesForCommand(
files,
prefixArgs,
maxBytes = DOCS_FORMAT_MAX_COMMAND_LINE_BYTES,
) {
const chunks = [];
let chunk = [];
let chunkBytes = commandLineBytes(prefixArgs);
for (const file of files) {
const fileBytes = Buffer.byteLength(file, "utf8") + 3;
if (chunk.length > 0 && chunkBytes + fileBytes > maxBytes) {
chunks.push(chunk);
chunk = [];
chunkBytes = commandLineBytes(prefixArgs);
}
chunk.push(file);
chunkBytes += fileBytes;
}
if (chunk.length > 0) {
chunks.push(chunk);
}
return chunks;
}
export function resolveOxfmtInvocation(args, params = {}) {
const repoRoot = params.repoRoot ?? ROOT;
const platform = params.platform ?? process.platform;
const existsSync = params.existsSync ?? fs.existsSync;
const shimName = platform === "win32" ? "oxfmt.cmd" : "oxfmt";
const shimPath = path.join(repoRoot, "node_modules", ".bin", shimName);
if (existsSync(shimPath)) {
if (platform === "win32") {
const comSpec = params.comSpec ?? process.env.ComSpec ?? "cmd.exe";
return {
command: comSpec,
args: ["/d", "/s", "/c", buildCmdExeCommandLine(shimPath, args)],
shell: false,
windowsVerbatimArguments: true,
};
}
return {
command: shimPath,
args,
shell: false,
};
}
return {
command: params.nodeExecPath ?? process.execPath,
args: [path.join(repoRoot, "node_modules", "oxfmt", "bin", "oxfmt"), ...args],
shell: false,
};
}
export function runOxfmt(files, params = {}, deps = {}) {
if (files.length === 0) {
return;
}
const repoRoot = params.repoRoot ?? ROOT;
const spawnSyncImpl = deps.spawnSync ?? spawnSync;
const prefixArgs = ["--write", "--threads=1", "--config", path.join(repoRoot, ".oxfmtrc.jsonc")];
for (const chunk of chunkFilesForCommand(
files,
prefixArgs,
params.maxCommandLineBytes ?? DOCS_FORMAT_MAX_COMMAND_LINE_BYTES,
)) {
const invocation = resolveOxfmtInvocation([...prefixArgs, ...chunk], {
comSpec: params.comSpec,
existsSync: deps.existsSync,
nodeExecPath: params.nodeExecPath,
platform: params.platform,
repoRoot,
});
const result = spawnSyncImpl(invocation.command, invocation.args, {
cwd: repoRoot,
encoding: "utf8",
maxBuffer: DOCS_FORMAT_MAX_BUFFER_BYTES,
shell: invocation.shell,
windowsVerbatimArguments: invocation.windowsVerbatimArguments,
});
if (result.error || result.status !== 0) {
throw new Error(commandFailureMessage("oxfmt", result, invocation));
}
}
}
export function repairFiles(root, files) {
const changed = [];
for (const relativePath of files) {
const absolutePath = path.join(root, relativePath);
const raw = fs.readFileSync(absolutePath, "utf8");
const formatted = repairMintlifyAccordionIndentation(raw);
if (formatted === raw) {
continue;
}
fs.writeFileSync(absolutePath, formatted);
changed.push(relativePath);
}
return changed;
}
function copyDocsToTemp(root, files) {
const tempRoot = fs.mkdtempSync(path.join(os.tmpdir(), "openclaw-docs-format-"));
for (const relativePath of files) {
const source = path.join(root, relativePath);
const target = path.join(tempRoot, relativePath);
fs.mkdirSync(path.dirname(target), { recursive: true });
fs.copyFileSync(source, target);
}
return tempRoot;
}
export function formatDocs(params = {}, deps = {}) {
const root = params.root ?? ROOT;
const check = params.check ?? false;
const changed = [];
const files = docsFiles(root, deps);
if (check) {
const tempRoot = copyDocsToTemp(root, files);
try {
runOxfmt(
files.map((relativePath) => path.join(tempRoot, relativePath)),
{ ...params, repoRoot: root },
deps,
);
repairFiles(tempRoot, files);
for (const relativePath of files) {
const raw = fs.readFileSync(path.join(root, relativePath), "utf8");
const formatted = fs.readFileSync(path.join(tempRoot, relativePath), "utf8");
if (formatted !== raw) {
changed.push(relativePath);
}
}
} finally {
fs.rmSync(tempRoot, { recursive: true, force: true });
}
} else {
runOxfmt(files, { ...params, repoRoot: root }, deps);
changed.push(...repairFiles(root, files));
}
return {
changed,
fileCount: files.length,
};
}
function main() {
const { changed, fileCount } = formatDocs({ check: CHECK, root: ROOT });
if (CHECK && changed.length > 0) {
console.error(`Format issues found in ${changed.length} docs file(s):`);
for (const relativePath of changed) {
console.error(`- ${relativePath}`);
}
process.exit(1);
}
if (changed.length > 0) {
console.log(`Formatted ${changed.length} docs file(s).`);
} else {
console.log(`Docs formatting clean (${fileCount} files).`);
}
}
if (import.meta.url === pathToFileURL(process.argv[1] ?? "").href) {
main();
}