playbook/antigravity-awesome-skills/skills/vercel-optimize/scripts/build-docs.mjs

75 lines
3.3 KiB
JavaScript
Executable File

#!/usr/bin/env node
// Regenerates references/{scanner-patterns,candidates}.md from lib/{scanners,gates}/*
// metadata. The .mjs files are the source of truth; check-docs-fresh.mjs blocks
// PRs where the regenerated output diverges from what's checked in.
import { writeFile } from 'node:fs/promises';
import { fileURLToPath } from 'node:url';
import { dirname, join } from 'node:path';
import { scanners } from '../lib/scanners/index.mjs';
import { gates, MAX_CODE_CANDIDATES, GATE_VERSION } from '../lib/gates/index.mjs';
const HERE = dirname(fileURLToPath(import.meta.url));
const REFS = join(HERE, '..', 'references');
const GENERATED_BANNER =
'<!-- THIS FILE IS GENERATED by scripts/build-docs.mjs. Do not edit by hand. -->\n' +
'<!-- To change scanner descriptions, edit lib/scanners/*.mjs metadata exports. -->\n' +
'<!-- To change gate thresholds, edit lib/gates/*.mjs metadata exports. -->\n\n';
async function main() {
await writeFile(join(REFS, 'scanner-patterns.md'), renderScanners());
await writeFile(join(REFS, 'candidates.md'), renderCandidates());
console.error('[build-docs] wrote scanner-patterns.md + candidates.md');
}
function renderScanners() {
const sorted = scanners.slice().sort((a, b) => a.metadata.id.localeCompare(b.metadata.id));
let out = GENERATED_BANNER + '# Scanner patterns\n\n';
out += 'AST/grep-style scanners run in parallel with metric-driven investigation. They find known anti-patterns. Findings on cold-path or unmappable files are dropped unless the scanner declares `trafficIndependent: true`.\n\n';
out += `Total scanners: ${sorted.length}.\n\n`;
out += '## Patterns\n\n';
for (const s of sorted) {
const m = s.metadata;
out += `### \`${m.id}\`${m.title}\n\n`;
out += `- **Severity**: ${m.severity}\n`;
out += `- **Billing dimension**: ${m.billingDimension}\n`;
out += `- **Traffic-independent**: ${m.trafficIndependent ? 'yes (cold-path findings survive the doctrine drop)' : 'no (cold-path findings get dropped)'}\n\n`;
out += `**Description.** ${m.description}\n\n`;
out += `**Fix.** ${m.fix}\n\n`;
if (m.citations?.length) {
out += `**Citations:**\n${m.citations.map((c) => `- \`${c}\``).join('\n')}\n\n`;
}
out += '---\n\n';
}
return trimTrailingBlankLine(out);
}
function renderCandidates() {
const sorted = gates.slice().sort((a, b) => a.metadata.id.localeCompare(b.metadata.id));
let out = GENERATED_BANNER + '# Candidate gates\n\n';
out += 'The deterministic threshold expressions that turn observability signals into investigation candidates. Pure JS, no LLM. Thresholds live in `lib/gates/*.mjs`.\n\n';
out += `Total gates: ${sorted.length}. Budget cap: \`MAX_CODE_CANDIDATES = ${MAX_CODE_CANDIDATES}\`. Gate version: \`${GATE_VERSION}\`.\n\n`;
out += '## Gates\n\n';
for (const g of sorted) {
const m = g.metadata;
out += `### \`${m.id}\`\n\n`;
out += `- **Threshold**: \`${m.threshold}\`\n`;
out += `- **Billing dimension**: ${m.billingDimension}\n`;
out += `- **Scope**: ${m.scope}\n`;
out += `- **Source citation**: \`${m.sourceCitation}\`\n\n`;
out += `${m.description}\n\n`;
out += '---\n\n';
}
return trimTrailingBlankLine(out);
}
function trimTrailingBlankLine(value) {
return value.replace(/\n{2,}$/, '\n');
}
main().catch((err) => {
console.error('[build-docs] FAILED:', err.message);
process.exit(1);
});