Files
openclaw/scripts/lib/vitest-shard-timings.mjs
2026-05-02 08:09:14 +01:00

127 lines
4.1 KiB
JavaScript

import { createHash } from "node:crypto";
import fs from "node:fs";
import path from "node:path";
const TIMINGS_FILE_ENV_KEY = "OPENCLAW_TEST_PROJECTS_TIMINGS_PATH";
const TIMINGS_DISABLE_ENV_KEY = "OPENCLAW_TEST_PROJECTS_TIMINGS";
const SHARD_NAME_ENV_KEY = "OPENCLAW_VITEST_SHARD_NAME";
function sanitizeTimingLabel(value) {
return String(value)
.trim()
.replace(/[^a-zA-Z0-9_.-]+/g, "-")
.replace(/^-+|-+$/g, "");
}
function hashIncludePatterns(includePatterns) {
return createHash("sha1").update(JSON.stringify(includePatterns)).digest("hex").slice(0, 12);
}
function shouldUseShardTimings(env = process.env) {
return env[TIMINGS_DISABLE_ENV_KEY] !== "0";
}
function resolveShardTimingsPath(cwd = process.cwd(), env = process.env) {
return env[TIMINGS_FILE_ENV_KEY] || path.join(cwd, ".artifacts", "vitest-shard-timings.json");
}
export function resolveShardTimingKey(spec) {
if (!Array.isArray(spec.includePatterns) || spec.includePatterns.length === 0) {
return spec.config;
}
const shardName = sanitizeTimingLabel(spec.env?.[SHARD_NAME_ENV_KEY] ?? "");
if (shardName) {
return `${spec.config}#${shardName}`;
}
return `${spec.config}#include-${spec.includePatterns.length}-${hashIncludePatterns(
spec.includePatterns,
)}`;
}
export function createShardTimingSample(spec, durationMs) {
if (spec.watchMode || !Number.isFinite(durationMs) || durationMs <= 0) {
return null;
}
const includePatternCount = Array.isArray(spec.includePatterns) ? spec.includePatterns.length : 0;
return {
baseConfig: spec.config,
config: resolveShardTimingKey(spec),
durationMs,
includePatternCount,
};
}
export function readShardTimings(cwd = process.cwd(), env = process.env) {
if (!shouldUseShardTimings(env)) {
return new Map();
}
try {
const raw = fs.readFileSync(resolveShardTimingsPath(cwd, env), "utf8");
const parsed = JSON.parse(raw);
const configs = parsed && typeof parsed === "object" ? parsed.configs : null;
if (!configs || typeof configs !== "object") {
return new Map();
}
return new Map(
Object.entries(configs)
.map(([config, value]) => {
const durationMs = Number(value?.averageMs ?? value?.durationMs);
return Number.isFinite(durationMs) && durationMs > 0 ? [config, durationMs] : null;
})
.filter(Boolean),
);
} catch {
return new Map();
}
}
export function writeShardTimings(samples, cwd = process.cwd(), env = process.env) {
if (!shouldUseShardTimings(env) || samples.length === 0) {
return;
}
const outputPath = resolveShardTimingsPath(cwd, env);
let current = { version: 1, configs: {} };
try {
current = JSON.parse(fs.readFileSync(outputPath, "utf8"));
} catch {
// First run, or a corrupt local artifact. Rewrite below.
}
const configs =
current && typeof current === "object" && current.configs && typeof current.configs === "object"
? { ...current.configs }
: {};
const updatedAt = new Date().toISOString();
for (const sample of samples) {
if (!sample.config || !Number.isFinite(sample.durationMs) || sample.durationMs <= 0) {
continue;
}
const previous = configs[sample.config];
const previousAverage = Number(previous?.averageMs ?? previous?.durationMs);
const sampleCount = Math.max(0, Number(previous?.sampleCount) || 0) + 1;
const averageMs =
Number.isFinite(previousAverage) && previousAverage > 0
? Math.round(previousAverage * 0.7 + sample.durationMs * 0.3)
: Math.round(sample.durationMs);
configs[sample.config] = {
averageMs,
lastMs: Math.round(sample.durationMs),
sampleCount,
updatedAt,
...(sample.baseConfig && sample.baseConfig !== sample.config
? { baseConfig: sample.baseConfig }
: {}),
...(sample.includePatternCount ? { includePatternCount: sample.includePatternCount } : {}),
};
}
fs.mkdirSync(path.dirname(outputPath), { recursive: true });
const tempPath = `${outputPath}.${process.pid}.tmp`;
fs.writeFileSync(tempPath, `${JSON.stringify({ version: 1, configs }, null, 2)}\n`, "utf8");
fs.renameSync(tempPath, outputPath);
}