sourcecred/config/test.js
William Chargin 32f83ad676
ci: fast-fail on forked PRs with no credentials (#1522)
Summary:
PRs created from forks don’t have credentials when running CI. This
commit causes the `test-full` job (which requires credentials) to fail
fast with a helpful error message.

Test Plan:
Push distinct versions of this commit to a fork and to the main
repository, and open pull requests for each. Note that the tests pass
from the main repository, but fail with a nice message from the fork:

![Screenshot of expected fast-fail behavior][ss]

The “team member pushes to trusted branch” workflow has already been
successfully exercised for #1521.

[ss]: https://user-images.githubusercontent.com/4317806/71707839-b782ab00-2da1-11ea-8aa9-7d8720538a87.png

wchargin-branch: forked-pr-fail-fast
2020-01-07 21:03:23 -08:00

231 lines
6.0 KiB
JavaScript

// @flow
/*:: import type {Task} from "../src/tools/execDependencyGraph"; */
const tmp = require("tmp");
const execDependencyGraph = require("../src/tools/execDependencyGraph");
main();
function main() {
const options = parseArgs();
if (isForkedPrFullRun(options)) {
printForkedPrFullRunErrorMessage();
process.exitCode = 1;
return;
}
const printVerboseResults = options.mode === "FULL";
const runOptions = {printVerboseResults};
const tasks = makeTasks(options.mode, options.limitMemoryUsage);
execDependencyGraph(tasks, runOptions).then(({success}) => {
process.exitCode = success ? 0 : 1;
});
}
function parseArgs() {
const options = {mode: "BASIC", limitMemoryUsage: false};
const args = process.argv.slice(2);
for (const arg of args) {
if (arg === "--full") {
options.mode = "FULL";
} else if (arg === "--ci") {
options.limitMemoryUsage = true;
} else {
throw new Error("unknown argument: " + JSON.stringify(arg));
}
}
return options;
}
/**
* Check whether we're running full CI for a PR created on a fork. In
* this state, Circle CI omits secure environment variables (which is
* good and desired), but this means that we'll have to abort tests.
*/
function isForkedPrFullRun(options) {
if (options.mode !== "FULL") {
return false;
}
if (!process.env["CIRCLE_PR_NUMBER"]) {
// This environment variable is only set on forked PRs.
// https://circleci.com/docs/2.0/env-vars/#built-in-environment-variables
return false;
}
if (process.env["SOURCECRED_GITHUB_TOKEN"]) {
return false;
}
return true;
}
function printForkedPrFullRunErrorMessage() {
console.error(
[
"fatal: cannot run full test suite: missing credentials",
"Tests on forked PRs run without credentials by default. A core team ",
"member will sanity-check your PR and push its head commit to a branch ",
"on the main SourceCred repository, which will re-run these tests.",
].join("\n")
);
}
function makeTasks(
mode /*: "BASIC" | "FULL" */,
limitMemoryUsage /*: boolean */
) {
const backendOutput = tmp.dirSync({
unsafeCleanup: true,
prefix: "sourcecred-test-",
}).name;
console.log("tmpdir for backend output: " + backendOutput);
function withSourcecredBinEnv(
invocation /*: $ReadOnlyArray<string> */
) /*: string[] */ {
return ["env", "SOURCECRED_BIN=" + backendOutput, ...invocation];
}
function flowCommand(limitMemoryUsage /*: boolean */) {
const cmd = [
"yarn",
"run",
"--silent",
"flow",
"--quiet",
"--max-warnings=0",
];
// Use only one worker to try to avoid flow flakey failures
if (limitMemoryUsage) {
cmd.push("--flowconfig-name", ".flowconfig-ci");
}
return cmd;
}
const basicTasks = [
{
id: "ensure-flow-typing",
cmd: ["./scripts/ensure-flow.sh"],
deps: [],
},
{
// eslint-disable-next-line no-useless-concat
id: "check-stop" + "ships",
// eslint-disable-next-line no-useless-concat
cmd: ["./scripts/check-stop" + "ships.sh"],
deps: [],
},
{
id: "check-pretty",
cmd: ["yarn", "run", "--silent", "check-pretty"],
deps: [],
},
{
id: "lint",
cmd: ["yarn", "run", "--silent", "lint"],
deps: [],
},
{
id: "flow",
cmd: flowCommand(limitMemoryUsage),
deps: [],
},
{
id: "unit",
cmd: ["yarn", "run", "--silent", "unit", "--ci"],
deps: [],
},
{
id: "check-gnu-coreutils",
cmd: ["./scripts/check-gnu-coreutils.sh"],
deps: [],
},
{
id: "backend",
cmd: [
"yarn",
"run",
"--silent",
"backend",
"--output-path",
backendOutput,
],
deps: [],
},
{
id: {BASIC: "sharness", FULL: "sharness-full"}[mode],
cmd: withSourcecredBinEnv([
"yarn",
"run",
"--silent",
{BASIC: "sharness", FULL: "sharness-full"}[mode],
]),
deps: ["backend", "check-gnu-coreutils"],
},
];
const extraTasks = [
{
id: "fetchGithubRepoTest",
cmd: withSourcecredBinEnv([
"./src/plugins/github/fetchGithubRepoTest.sh",
"--no-build",
]),
deps: ["backend"],
},
{
id: "fetchGithubOrgTest",
cmd: withSourcecredBinEnv([
"./src/plugins/github/fetchGithubOrgTest.sh",
"--no-build",
]),
deps: ["backend"],
},
];
const tasks = (function() {
switch (mode) {
case "BASIC":
return basicTasks;
case "FULL":
return [].concat(basicTasks, extraTasks);
default:
/*:: (mode: empty); */ throw new Error(mode);
}
})();
if (limitMemoryUsage) {
// We've had issues with our tests flakily failing in CI, due to apparent
// memory issues.
//
// This block attempts to limit memory usage by having flow to run first,
// then stopping the flow server, then running unit tests, and only
// afterwards running all other tasks.
//
// The reasoning is that the flow server is fairly memory demanding and we
// can safely kill it after we've checked the types, and jest is also quite
// memory intensive. Hopefully by finishing these tasks first and releasing
// their resources, we won't have more memory exhaustion.
tasks.forEach((task) => {
switch (task.id) {
case "flow":
// Run flow first
return;
case "unit":
task.cmd.push("--maxWorkers=2");
// Run unit after we _stopped_ the flow server
// (to free up memory from flow)
task.deps.push("flow-stop");
return;
default:
// Run everything else after unit tests
// (unit is a memory hog)
task.deps.push("unit");
}
});
const flowStopTask /*: Task */ = {
id: "flow-stop",
cmd: ["yarn", "run", "--silent", "flow", "stop"],
deps: ["flow"],
};
tasks.push(flowStopTask);
}
return tasks;
}