mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-06 06:40:44 +00:00
127 lines
4.1 KiB
JavaScript
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);
|
|
}
|