Files
openclaw/test/scripts/kova-ci-summary.test.ts
2026-06-21 19:07:02 +02:00

287 lines
8.6 KiB
TypeScript

// Kova Ci Summary tests cover kova ci summary script behavior.
import { spawnSync } from "node:child_process";
import { mkdtempSync, readFileSync, rmSync, writeFileSync } from "node:fs";
import { tmpdir } from "node:os";
import { join } from "node:path";
import { describe, expect, it } from "vitest";
function runSummary(report: unknown, extraArgs: string[] = []) {
const root = mkdtempSync(join(tmpdir(), "openclaw-kova-summary-"));
const reportPath = join(root, "report.json");
const outputPath = join(root, "summary.md");
writeFileSync(reportPath, `${JSON.stringify(report, null, 2)}\n`, "utf8");
const result = spawnSync(
process.execPath,
["scripts/kova-ci-summary.mjs", "--report", reportPath, "--output", outputPath, ...extraArgs],
{
cwd: process.cwd(),
encoding: "utf8",
},
);
let output = "";
try {
output = readFileSync(outputPath, "utf8");
} catch {}
rmSync(root, { force: true, recursive: true });
return { output, result };
}
describe("scripts/kova-ci-summary", () => {
it("prints help without treating --help as a valued option", () => {
const result = spawnSync(process.execPath, ["scripts/kova-ci-summary.mjs", "--help"], {
cwd: process.cwd(),
encoding: "utf8",
});
expect(result.status).toBe(0);
expect(result.stderr).toBe("");
expect(result.stdout).toContain("usage: node scripts/kova-ci-summary.mjs --report");
});
it.each([
["flag-shaped value", ["--report", "-h"]],
["option separator before help", ["--report", "--", "--help"]],
])("rejects %s before help handling", (_name, args) => {
const result = spawnSync(process.execPath, ["scripts/kova-ci-summary.mjs", ...args], {
cwd: process.cwd(),
encoding: "utf8",
});
expect(result.status).toBe(2);
expect(result.stdout).toBe("");
expect(result.stderr).toContain("error: --report requires a value");
});
it("rejects empty Kova reports instead of rendering unknown summaries", () => {
const empty = runSummary({});
expect(empty.result.status).toBe(1);
expect(empty.result.stderr).toContain("invalid Kova report: missing summary.statuses");
const noEvidence = runSummary({ summary: { statuses: { pass: 1 } } });
expect(noEvidence.result.status).toBe(1);
expect(noEvidence.result.stderr).toContain(
"invalid Kova report: missing records or performance groups",
);
});
it("rejects unknown flags instead of silently dropping report metadata", () => {
const result = runSummary(
{
records: [{ scenario: "gateway", state: "clean", status: "pass" }],
summary: { statuses: { pass: 1 } },
},
["--report-urlz", "https://example.test/report"],
).result;
expect(result.status).toBe(2);
expect(result.stderr).toContain("unknown argument: --report-urlz");
});
it("renders a Kova summary when status and evidence are present", () => {
const { output, result } = runSummary({
generatedAt: "2026-06-06T00:00:00.000Z",
performance: {
repeat: 1,
groups: [
{
metrics: {
cpuPercentMax: {
count: 1,
max: 12,
median: 12,
p95: 12,
title: "CPU max",
unit: "%",
},
resourcePeakGatewayRssMb: {
count: 1,
max: 256,
median: 256,
p95: 256,
title: "Gateway RSS",
unit: "MB",
},
timeToHealthReadyMs: {
count: 1,
max: 30,
median: 20,
p95: 30,
title: "Health ready",
unit: "ms",
},
},
scenario: "gateway",
state: "clean",
},
],
},
records: [{ scenario: "gateway", state: "clean", status: "pass" }],
runId: "run-1",
summary: { statuses: { pass: 1 } },
target: "main",
});
expect(result.status).toBe(0);
expect(output).toContain("- Statuses: pass: 1");
expect(output).toContain("| gateway | clean | Health ready | 20 ms | 30 ms | 30 ms |");
expect(output).toContain("| gateway | clean | Gateway RSS | 256 MB | 256 MB | 256 MB |");
expect(output).toContain("| gateway | clean | CPU max | 12 % | 12 % | 12 % |");
});
it("rejects performance summaries without resource metrics", () => {
const { result } = runSummary({
performance: {
repeat: 1,
groups: [
{
metrics: {
timeToHealthReadyMs: {
count: 1,
max: 30,
median: 20,
p95: 30,
title: "Health ready",
unit: "ms",
},
},
scenario: "gateway",
state: "clean",
},
],
},
records: [{ scenario: "gateway", state: "clean", status: "pass" }],
summary: { statuses: { pass: 1 } },
});
expect(result.status).toBe(1);
expect(result.stderr).toContain(
"invalid Kova report: missing sampled RSS metric in performance groups",
);
});
it("rejects performance summaries without CPU metrics", () => {
const { result } = runSummary({
performance: {
repeat: 1,
groups: [
{
metrics: {
resourcePeakGatewayRssMb: {
count: 1,
max: 256,
median: 256,
p95: 256,
title: "Gateway RSS",
unit: "MB",
},
timeToHealthReadyMs: {
count: 1,
max: 30,
median: 20,
p95: 30,
title: "Health ready",
unit: "ms",
},
},
scenario: "gateway",
state: "clean",
},
],
},
records: [{ scenario: "gateway", state: "clean", status: "pass" }],
summary: { statuses: { pass: 1 } },
});
expect(result.status).toBe(1);
expect(result.stderr).toContain(
"invalid Kova report: missing sampled CPU metric in performance groups",
);
});
it("rejects malformed resource metric counts instead of treating them as sampled", () => {
const { result } = runSummary({
performance: {
repeat: 1,
groups: [
{
metrics: {
cpuPercentMax: {
count: "Infinity",
max: 12,
median: 12,
p95: 12,
title: "CPU max",
unit: "%",
},
resourcePeakGatewayRssMb: {
count: true,
max: 256,
median: 256,
p95: 256,
title: "Gateway RSS",
unit: "MB",
},
},
scenario: "gateway",
state: "clean",
},
],
},
records: [{ scenario: "gateway", state: "clean", status: "pass" }],
summary: { statuses: { pass: 1 } },
});
expect(result.status).toBe(1);
expect(result.stderr).toContain(
"invalid Kova report: missing sampled RSS metric in performance groups",
);
});
it("omits key metric rows with invalid sample counts", () => {
const { output, result } = runSummary({
performance: {
repeat: 1,
groups: [
{
metrics: {
cpuPercentMax: {
count: 1,
max: 12,
median: 12,
p95: 12,
title: "CPU max",
unit: "%",
},
resourcePeakGatewayRssMb: {
count: 1,
max: 256,
median: 256,
p95: 256,
title: "Gateway RSS",
unit: "MB",
},
timeToHealthReadyMs: {
count: "0",
max: 30,
median: 20,
p95: 30,
title: "Health ready",
unit: "ms",
},
},
scenario: "gateway",
state: "clean",
},
],
},
records: [{ scenario: "gateway", state: "clean", status: "pass" }],
summary: { statuses: { pass: 1 } },
});
expect(result.status).toBe(0);
expect(output).not.toContain("Health ready");
expect(output).toContain("| gateway | clean | Gateway RSS | 256 MB | 256 MB | 256 MB |");
expect(output).toContain("| gateway | clean | CPU max | 12 % | 12 % | 12 % |");
});
});