Files
openclaw/scripts/test-parallel.mjs
Tak Hoffman 5b68e52894 ci: collapse preflight manifest routing (#54773)
* ci: collapse preflight manifest routing

* ci: fix preflight workflow outputs

* ci: restore compat workflow tasks

* ci: match macos shards to windows

* ci: collapse macos swift jobs

* ci: skip empty submodule setup

* ci: drop submodule setup from node env
2026-03-25 22:38:30 -05:00

169 lines
4.7 KiB
JavaScript

import {
createExecutionArtifacts,
executePlan,
formatExplanation,
formatPlanOutput,
} from "./test-planner/executor.mjs";
import {
buildCIExecutionManifest,
buildExecutionPlan,
explainExecutionTarget,
} from "./test-planner/planner.mjs";
const parseCliArgs = (args) => {
const wrapper = {
ciManifest: false,
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 === "--ci-manifest") {
wrapper.ciManifest = 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",
" --ci-manifest Print the planner-backed CI execution manifest as JSON",
" --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);
}
if (rawCli.ciManifest) {
const manifest = buildCIExecutionManifest(undefined, { env: process.env });
console.log(`${JSON.stringify(manifest, null, 2)}\n`);
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);