2023-04-15 00:47:00 +00:00
|
|
|
#!/usr/bin/env node --redirect-warnings=/dev/null
|
|
|
|
|
2024-02-25 16:03:33 +00:00
|
|
|
const fs = require("fs");
|
|
|
|
const path = require("path");
|
|
|
|
const { spawnSync } = require("child_process");
|
2023-04-15 00:47:00 +00:00
|
|
|
|
2024-02-25 16:03:33 +00:00
|
|
|
const FAILING_SEED_REGEX = /failing seed: (\d+)/gi;
|
|
|
|
const CARGO_TEST_ARGS = ["--release", "--lib", "--package", "collab"];
|
2023-04-17 22:34:23 +00:00
|
|
|
|
|
|
|
if (require.main === module) {
|
|
|
|
if (process.argv.length < 4) {
|
2024-02-25 16:03:33 +00:00
|
|
|
process.stderr.write(
|
|
|
|
"usage: script/randomized-test-minimize <input-plan> <output-plan> [start-index]\n",
|
|
|
|
);
|
|
|
|
process.exit(1);
|
2023-04-15 00:47:00 +00:00
|
|
|
}
|
|
|
|
|
2023-04-17 22:34:23 +00:00
|
|
|
minimizeTestPlan(
|
|
|
|
process.argv[2],
|
|
|
|
process.argv[3],
|
2024-02-25 16:03:33 +00:00
|
|
|
parseInt(process.argv[4]) || 0,
|
2023-04-17 22:34:23 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-02-25 16:03:33 +00:00
|
|
|
function minimizeTestPlan(inputPlanPath, outputPlanPath, startIndex = 0) {
|
|
|
|
const tempPlanPath = inputPlanPath + ".try";
|
2023-04-17 22:34:23 +00:00
|
|
|
|
2024-02-25 16:03:33 +00:00
|
|
|
fs.copyFileSync(inputPlanPath, outputPlanPath);
|
|
|
|
let testPlan = JSON.parse(fs.readFileSync(outputPlanPath, "utf8"));
|
2023-04-17 22:34:23 +00:00
|
|
|
|
2024-02-25 16:03:33 +00:00
|
|
|
process.stderr.write("minimizing failing test plan...\n");
|
2023-04-17 22:34:23 +00:00
|
|
|
for (let ix = startIndex; ix < testPlan.length; ix++) {
|
|
|
|
// Skip 'MutateClients' entries, since they themselves are not single operations.
|
|
|
|
if (testPlan[ix].MutateClients) {
|
2024-02-25 16:03:33 +00:00
|
|
|
continue;
|
2023-04-17 22:34:23 +00:00
|
|
|
}
|
2023-04-15 00:47:00 +00:00
|
|
|
|
2023-04-17 22:34:23 +00:00
|
|
|
// Remove a row from the test plan
|
2024-02-25 16:03:33 +00:00
|
|
|
const newTestPlan = testPlan.slice();
|
|
|
|
newTestPlan.splice(ix, 1);
|
|
|
|
fs.writeFileSync(tempPlanPath, serializeTestPlan(newTestPlan), "utf8");
|
2023-04-17 22:34:23 +00:00
|
|
|
|
2024-02-25 16:03:33 +00:00
|
|
|
process.stderr.write(
|
|
|
|
`${ix}/${testPlan.length}: ${JSON.stringify(testPlan[ix])}`,
|
|
|
|
);
|
2023-04-17 22:34:23 +00:00
|
|
|
const failingSeed = runTests({
|
2024-02-25 16:03:33 +00:00
|
|
|
SEED: "0",
|
2023-04-17 22:34:23 +00:00
|
|
|
LOAD_PLAN: tempPlanPath,
|
|
|
|
SAVE_PLAN: tempPlanPath,
|
2024-02-25 16:03:33 +00:00
|
|
|
ITERATIONS: "500",
|
|
|
|
});
|
2023-04-17 22:34:23 +00:00
|
|
|
|
|
|
|
// If the test failed, keep the test plan with the removed row. Reload the test
|
|
|
|
// plan from the JSON file, since the test itself will remove any operations
|
|
|
|
// which are no longer valid before saving the test plan.
|
|
|
|
if (failingSeed != null) {
|
2024-02-25 16:03:33 +00:00
|
|
|
process.stderr.write(` - remove. failing seed: ${failingSeed}.\n`);
|
|
|
|
fs.copyFileSync(tempPlanPath, outputPlanPath);
|
|
|
|
testPlan = JSON.parse(fs.readFileSync(outputPlanPath, "utf8"));
|
|
|
|
ix--;
|
2023-04-17 22:34:23 +00:00
|
|
|
} else {
|
2024-02-25 16:03:33 +00:00
|
|
|
process.stderr.write(` - keep.\n`);
|
2023-04-17 22:34:23 +00:00
|
|
|
}
|
|
|
|
}
|
2023-04-15 00:47:00 +00:00
|
|
|
|
2024-02-25 16:03:33 +00:00
|
|
|
fs.unlinkSync(tempPlanPath);
|
2023-04-17 22:34:23 +00:00
|
|
|
|
|
|
|
// Re-run the final minimized plan to get the correct failing seed.
|
|
|
|
// This is a workaround for the fact that the execution order can
|
|
|
|
// slightly change when replaying a test plan after it has been
|
|
|
|
// saved and loaded.
|
|
|
|
const failingSeed = runTests({
|
2024-02-25 16:03:33 +00:00
|
|
|
SEED: "0",
|
|
|
|
ITERATIONS: "5000",
|
2023-04-17 22:34:23 +00:00
|
|
|
LOAD_PLAN: outputPlanPath,
|
2024-02-25 16:03:33 +00:00
|
|
|
});
|
2023-04-17 22:34:23 +00:00
|
|
|
|
2024-02-25 16:03:33 +00:00
|
|
|
process.stderr.write(`final test plan: ${outputPlanPath}\n`);
|
|
|
|
process.stderr.write(`final seed: ${failingSeed}\n`);
|
|
|
|
return failingSeed;
|
2023-04-17 22:34:23 +00:00
|
|
|
}
|
2023-04-15 00:47:00 +00:00
|
|
|
|
2023-04-17 22:34:23 +00:00
|
|
|
function buildTests() {
|
2024-02-25 16:03:33 +00:00
|
|
|
const { status } = spawnSync(
|
|
|
|
"cargo",
|
|
|
|
["test", "--no-run", ...CARGO_TEST_ARGS],
|
|
|
|
{
|
|
|
|
stdio: "inherit",
|
|
|
|
encoding: "utf8",
|
|
|
|
env: {
|
|
|
|
...process.env,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
);
|
2023-04-17 22:34:23 +00:00
|
|
|
if (status !== 0) {
|
2024-02-25 16:03:33 +00:00
|
|
|
throw new Error("build failed");
|
2023-04-15 00:47:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-17 22:34:23 +00:00
|
|
|
function runTests(env) {
|
2024-02-25 16:03:33 +00:00
|
|
|
const { status, stdout } = spawnSync(
|
|
|
|
"cargo",
|
|
|
|
["test", ...CARGO_TEST_ARGS, "random_project_collaboration"],
|
|
|
|
{
|
|
|
|
stdio: "pipe",
|
|
|
|
encoding: "utf8",
|
|
|
|
env: {
|
|
|
|
...process.env,
|
|
|
|
...env,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
);
|
2023-04-15 00:47:00 +00:00
|
|
|
|
|
|
|
if (status !== 0) {
|
2024-02-25 16:03:33 +00:00
|
|
|
FAILING_SEED_REGEX.lastIndex = 0;
|
|
|
|
const match = FAILING_SEED_REGEX.exec(stdout);
|
2023-04-15 00:47:00 +00:00
|
|
|
if (!match) {
|
2024-02-25 16:03:33 +00:00
|
|
|
process.stderr.write("test failed, but no failing seed found:\n");
|
|
|
|
process.stderr.write(stdout);
|
|
|
|
process.stderr.write("\n");
|
|
|
|
process.exit(1);
|
2023-04-15 00:47:00 +00:00
|
|
|
}
|
2024-02-25 16:03:33 +00:00
|
|
|
return match[1];
|
2023-04-15 00:47:00 +00:00
|
|
|
} else {
|
2024-02-25 16:03:33 +00:00
|
|
|
return null;
|
2023-04-15 00:47:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function serializeTestPlan(plan) {
|
2024-02-25 16:03:33 +00:00
|
|
|
return "[\n" + plan.map((row) => JSON.stringify(row)).join(",\n") + "\n]\n";
|
2023-04-15 00:47:00 +00:00
|
|
|
}
|
2023-04-17 22:34:23 +00:00
|
|
|
|
2024-02-25 16:03:33 +00:00
|
|
|
exports.buildTests = buildTests;
|
|
|
|
exports.runTests = runTests;
|
|
|
|
exports.minimizeTestPlan = minimizeTestPlan;
|