mirror of
https://github.com/openclaw/openclaw.git
synced 2026-03-27 18:01:46 +00:00
* test: stabilize ci and local vitest workers * test: introduce planner-backed test runner * test: address planner review follow-ups * test: derive planner budgets from host capabilities * test: restore planner filter helper import * test: align planner explain output with execution * test: keep low profile as serial alias * test: restrict explicit planner file targets * test: clean planner exits and pnpm launch * test: tighten wrapper flag validation * ci: gate heavy fanout on check * test: key shard assignments by unit identity * ci(bun): shard vitest lanes further * test: restore ci overlap and stabilize planner tests * test: relax planner output worker assertions * test: reset plugin runtime state in optional tools suite * ci: split macos node and swift jobs * test: honor no-isolate top-level concurrency budgets * ci: fix macos swift format lint * test: cap max-profile top-level concurrency * ci: shard macos node checks * ci: use four macos node shards * test: normalize explain targets before classification
153 lines
4.3 KiB
JavaScript
153 lines
4.3 KiB
JavaScript
import {
|
|
createExecutionArtifacts,
|
|
executePlan,
|
|
formatExplanation,
|
|
formatPlanOutput,
|
|
} from "./test-planner/executor.mjs";
|
|
import { buildExecutionPlan, explainExecutionTarget } from "./test-planner/planner.mjs";
|
|
|
|
const parseCliArgs = (args) => {
|
|
const wrapper = {
|
|
plan: false,
|
|
explain: null,
|
|
mode: null,
|
|
profile: null,
|
|
surfaces: [],
|
|
files: [],
|
|
passthroughArgs: [],
|
|
showHelp: false,
|
|
};
|
|
let passthroughMode = false;
|
|
for (let index = 0; index < args.length; index += 1) {
|
|
const arg = args[index];
|
|
if (passthroughMode) {
|
|
wrapper.passthroughArgs.push(arg);
|
|
continue;
|
|
}
|
|
if (arg === "--") {
|
|
passthroughMode = true;
|
|
continue;
|
|
}
|
|
if (arg === "--plan") {
|
|
wrapper.plan = true;
|
|
continue;
|
|
}
|
|
if (arg === "--help") {
|
|
wrapper.showHelp = true;
|
|
continue;
|
|
}
|
|
if (arg === "--mode") {
|
|
const nextValue = args[index + 1] ?? null;
|
|
if (nextValue === "ci" || nextValue === "local") {
|
|
wrapper.mode = nextValue;
|
|
index += 1;
|
|
continue;
|
|
}
|
|
}
|
|
if (arg === "--profile") {
|
|
const nextValue = args[index + 1] ?? "";
|
|
if (!nextValue || nextValue === "--" || nextValue.startsWith("-")) {
|
|
throw new Error(`Invalid --profile value: ${String(nextValue || "<missing>")}`);
|
|
}
|
|
wrapper.profile = nextValue;
|
|
index += 1;
|
|
continue;
|
|
}
|
|
if (arg === "--surface") {
|
|
const nextValue = args[index + 1] ?? "";
|
|
if (!nextValue || nextValue === "--" || nextValue.startsWith("-")) {
|
|
throw new Error(`Invalid --surface value: ${String(nextValue || "<missing>")}`);
|
|
}
|
|
wrapper.surfaces.push(nextValue);
|
|
index += 1;
|
|
continue;
|
|
}
|
|
if (arg === "--files") {
|
|
const nextValue = args[index + 1] ?? "";
|
|
if (!nextValue || nextValue === "--" || nextValue.startsWith("-")) {
|
|
throw new Error(`Invalid --files value: ${String(nextValue || "<missing>")}`);
|
|
}
|
|
wrapper.files.push(nextValue);
|
|
index += 1;
|
|
continue;
|
|
}
|
|
if (arg === "--explain") {
|
|
const nextValue = args[index + 1] ?? "";
|
|
if (!nextValue || nextValue === "--" || nextValue.startsWith("-")) {
|
|
throw new Error(`Invalid --explain value: ${String(nextValue || "<missing>")}`);
|
|
}
|
|
wrapper.explain = nextValue;
|
|
index += 1;
|
|
continue;
|
|
}
|
|
wrapper.passthroughArgs.push(arg);
|
|
}
|
|
return wrapper;
|
|
};
|
|
|
|
const exitWithCleanup = (artifacts, code) => {
|
|
artifacts?.cleanupTempArtifacts?.();
|
|
process.exit(code);
|
|
};
|
|
|
|
let rawCli;
|
|
try {
|
|
rawCli = parseCliArgs(process.argv.slice(2));
|
|
} catch (error) {
|
|
console.error(`[test-parallel] ${error instanceof Error ? error.message : String(error)}`);
|
|
process.exit(2);
|
|
}
|
|
if (rawCli.showHelp) {
|
|
console.log(
|
|
[
|
|
"Usage: node scripts/test-parallel.mjs [wrapper flags] [-- vitest args]",
|
|
"",
|
|
"Wrapper flags:",
|
|
" --plan Print the resolved execution plan",
|
|
" --explain <file> Explain how a file is classified and run",
|
|
" --surface <name> Select a surface (repeatable or comma-separated)",
|
|
" --files <pattern> Add targeted files/patterns (repeatable)",
|
|
" --mode <ci|local> Override runtime mode",
|
|
" --profile <name> Override execution intent (normal|max|serial)",
|
|
].join("\n"),
|
|
);
|
|
process.exit(0);
|
|
}
|
|
|
|
const request = {
|
|
mode: rawCli.mode,
|
|
profile: rawCli.profile,
|
|
surfaces: rawCli.surfaces,
|
|
fileFilters: rawCli.files,
|
|
passthroughArgs: rawCli.passthroughArgs,
|
|
};
|
|
|
|
if (rawCli.explain) {
|
|
const explanation = explainExecutionTarget(
|
|
{ ...request, passthroughArgs: [], fileFilters: [rawCli.explain] },
|
|
{ env: process.env },
|
|
);
|
|
console.log(formatExplanation(explanation));
|
|
process.exit(0);
|
|
}
|
|
|
|
const artifacts = createExecutionArtifacts(process.env);
|
|
let plan;
|
|
try {
|
|
plan = buildExecutionPlan(request, {
|
|
env: process.env,
|
|
writeTempJsonArtifact: artifacts.writeTempJsonArtifact,
|
|
});
|
|
} catch (error) {
|
|
console.error(`[test-parallel] ${error instanceof Error ? error.message : String(error)}`);
|
|
exitWithCleanup(artifacts, 2);
|
|
}
|
|
|
|
if (process.env.OPENCLAW_TEST_LIST_LANES === "1" || rawCli.plan) {
|
|
console.log(formatPlanOutput(plan));
|
|
exitWithCleanup(artifacts, 0);
|
|
}
|
|
|
|
const exitCode = await executePlan(plan, { env: process.env, artifacts });
|
|
process.exit(exitCode);
|