Files
openclaw/test/scripts/render-maturity-docs.test.ts
Dallin Romney 61d4ff782e docs: clarify maturity scorecard scoring (#96594)
* docs: clarify maturity scorecard scoring

* chore: split qa profile workflow change

* docs: keep maturity coverage values stable

* test: keep maturity renderer fixture in core boundary
2026-06-24 17:32:08 -07:00

254 lines
7.6 KiB
TypeScript

import { spawnSync } from "node:child_process";
// Maturity docs renderer tests cover evidence-backed generated-doc checks.
import fs from "node:fs";
import path from "node:path";
import { afterEach, describe, expect, it } from "vitest";
import { parse as parseYaml } from "yaml";
import { createTempDirTracker } from "../helpers/temp-dir.js";
const repoRoot = path.resolve(__dirname, "../..");
const tempDirs = createTempDirTracker();
type TaxonomyFixture = {
surfaces?: TaxonomySurfaceFixture[];
};
type TaxonomySurfaceFixture = {
id?: string;
status?: string;
categories?: TaxonomyCategoryFixture[];
};
type TaxonomyCategoryFixture = {
id?: string;
name?: string;
features?: TaxonomyFeatureFixture[];
};
type TaxonomyFeatureFixture = {
coverageIds?: string[];
};
afterEach(() => {
tempDirs.cleanup();
});
function runCli(...args: string[]) {
return spawnSync(
process.execPath,
["--import", "tsx", "scripts/qa/render-maturity-docs.ts", ...args],
{
cwd: repoRoot,
encoding: "utf8",
},
);
}
function writeQaEvidence(params: {
dir: string;
entries: Array<{ id: string; status: "pass" | "fail" | "blocked" | "skipped" }>;
scorecard?: unknown;
}) {
const scorecard = params.scorecard ?? {
filters: { surface: null, category: null },
run: { evidenceEntryCount: params.entries.length },
categories: {
total: 0,
fulfilled: 0,
partial: 0,
missing: 0,
fulfillmentPercent: 0,
},
features: {
total: 0,
fulfilled: 0,
partial: 0,
missing: 0,
fulfillmentPercent: 0,
},
coverageIds: {
total: 0,
fulfilled: 0,
missing: 0,
fulfillmentPercent: 0,
},
categoryReports: [],
};
fs.mkdirSync(params.dir, { recursive: true });
fs.writeFileSync(
path.join(params.dir, "qa-evidence.json"),
`${JSON.stringify(
{
kind: "openclaw.qa.evidence-summary",
schemaVersion: 2,
generatedAt: "2026-06-23T00:00:00.000Z",
evidenceMode: "full",
profile: "all",
entries: params.entries.map((entry) => ({
test: {
kind: "qa-scenario",
id: entry.id,
title: entry.id,
source: { path: `qa/scenarios/${entry.id}.yaml` },
},
coverage: [{ id: "tools.evidence", role: "primary" }],
result: { status: entry.status },
})),
scorecard,
},
null,
2,
)}\n`,
"utf8",
);
}
function allProfileScorecardFixture() {
const taxonomy = parseYaml(
fs.readFileSync(path.join(repoRoot, "taxonomy.yaml"), "utf8"),
) as TaxonomyFixture;
const activeSurfaces = (taxonomy.surfaces ?? []).filter(
(surface) => surface.status !== "retired",
);
const categoryReports = activeSurfaces.flatMap((surface) =>
(surface.categories ?? []).map((category) => {
const coverageIds = [
...new Set((category.features ?? []).flatMap((feature) => feature.coverageIds ?? [])),
].sort();
return {
id: `${surface.id}.${category.id}`,
surfaceId: surface.id,
name: category.name,
status: "missing",
features: {
total: category.features.length,
fulfilled: 0,
partial: 0,
missing: category.features.length,
fulfillmentPercent: 0,
},
coverageIds: {
total: coverageIds.length,
fulfilled: 0,
missing: coverageIds.length,
fulfillmentPercent: 0,
secondaryOnly: 0,
},
missingCoverageIds: coverageIds,
};
}),
);
const featureCount = categoryReports.reduce((count, report) => count + report.features.total, 0);
const coverageIdCount = categoryReports.reduce(
(count, report) => count + report.coverageIds.total,
0,
);
return {
filters: { surface: null, category: null },
run: { evidenceEntryCount: 1 },
categories: {
total: categoryReports.length,
fulfilled: 0,
partial: 0,
missing: categoryReports.length,
fulfillmentPercent: 0,
},
features: {
total: featureCount,
fulfilled: 0,
partial: 0,
missing: featureCount,
fulfillmentPercent: 0,
},
coverageIds: {
total: coverageIdCount,
fulfilled: 0,
missing: coverageIdCount,
fulfillmentPercent: 0,
},
categoryReports,
};
}
describe("maturity docs renderer CLI", () => {
it("checks maturity inputs without requiring QA evidence artifacts", () => {
const result = runCli("--check");
expect(result.status).toBe(0);
expect(result.stderr).toBe("");
expect(result.stdout).toContain("maturity docs inputs are valid in docs");
expect(result.stdout).toContain("evidence-backed freshness check skipped");
});
it("still requires QA evidence artifacts when rendering generated docs", () => {
const outputDir = tempDirs.make("openclaw-maturity-docs-test-");
const result = runCli("--output-dir", outputDir);
expect(result.status).toBe(1);
expect(result.stdout).toBe("");
expect(result.stderr).toContain(
"maturity scorecard rendering requires all or release profile qa-evidence.json",
);
});
it("rejects scorecard evidence with failed or blocked entries", () => {
const outputDir = tempDirs.make("openclaw-maturity-docs-output-");
const evidenceDir = tempDirs.make("openclaw-maturity-docs-evidence-");
writeQaEvidence({
dir: evidenceDir,
entries: [
{ id: "passing-scenario", status: "pass" },
{ id: "failing-scenario", status: "fail" },
{ id: "blocked-scenario", status: "blocked" },
],
});
const result = runCli("--output-dir", outputDir, "--evidence-dir", evidenceDir);
expect(result.status).toBe(1);
expect(result.stdout).toBe("");
expect(result.stderr).toContain("maturity docs require passing QA evidence");
expect(result.stderr).toContain("failing-scenario (fail)");
expect(result.stderr).toContain("blocked-scenario (blocked)");
});
it("renders passing evidence without impossible failed or blocked result counts", () => {
const outputDir = tempDirs.make("openclaw-maturity-docs-output-");
const evidenceDir = tempDirs.make("openclaw-maturity-docs-evidence-");
writeQaEvidence({
dir: evidenceDir,
entries: [
{ id: "passing-scenario", status: "pass" },
{ id: "skipped-scenario", status: "skipped" },
],
});
const result = runCli("--output-dir", outputDir, "--evidence-dir", evidenceDir);
expect(result.status).toBe(0);
const scorecard = fs.readFileSync(path.join(outputDir, "maturity", "scorecard.md"), "utf8");
expect(scorecard).toContain("1 passed, 1 skipped");
expect(scorecard).not.toContain("0 failed");
expect(scorecard).not.toContain("0 blocked");
});
it("renders the maturity score from quality and completeness without coverage", () => {
const outputDir = tempDirs.make("openclaw-maturity-docs-output-");
const evidenceDir = tempDirs.make("openclaw-maturity-docs-evidence-");
writeQaEvidence({
dir: evidenceDir,
entries: [{ id: "passing-scenario", status: "pass" }],
scorecard: allProfileScorecardFixture(),
});
const result = runCli("--output-dir", outputDir, "--evidence-dir", evidenceDir);
expect(result.status).toBe(0);
const scorecard = fs.readFileSync(path.join(outputDir, "maturity", "scorecard.md"), "utf8");
expect(scorecard).toContain("<span>Maturity score</span>");
expect(scorecard).toContain('<span className="maturity-summary-value">67%</span>');
expect(scorecard).toContain("Coverage Experimental - 0%");
expect(scorecard).toContain("end-to-end coverage above 90%");
});
});