@aws-cdk/integ-runner
Version:
CDK Integration Testing Tool
100 lines • 14.4 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.runIntegrationTests = runIntegrationTests;
exports.runIntegrationTestsInParallel = runIntegrationTestsInParallel;
const common_1 = require("./common");
const logger = require("../logger");
const utils_1 = require("../utils");
/**
* Run Integration tests.
*/
async function runIntegrationTests(options) {
logger.highlight('\nRunning integration tests for failed tests...\n');
logger.print('Running in parallel across %sregions: %s', options.profiles ? `profiles ${options.profiles.join(', ')} and ` : '', options.regions.join(', '));
const totalTests = options.tests.length;
const responses = await runIntegrationTestsInParallel(options);
logger.highlight('\nTest Results: \n');
(0, common_1.printSummary)(totalTests, responses.failedTests.length);
return {
success: responses.failedTests.length === 0,
metrics: responses.metrics,
};
}
/**
* Returns a list of AccountWorkers based on the list of regions and profiles
* given to the CLI.
*/
function getAccountWorkers(regions, profiles) {
const workers = [];
function pushWorker(profile) {
for (const region of regions) {
workers.push({
region,
profile,
});
}
}
if (profiles && profiles.length > 0) {
for (const profile of profiles ?? []) {
pushWorker(profile);
}
}
else {
pushWorker();
}
return workers;
}
/**
* Runs a set of integration tests in parallel across a list of AWS regions.
* Only a single test can be run at a time in a given region. Once a region
* is done running a test, the next test will be pulled from the queue
*/
async function runIntegrationTestsInParallel(options) {
const queue = options.tests;
const results = {
metrics: [],
failedTests: [],
};
const accountWorkers = getAccountWorkers(options.regions, options.profiles);
async function runTest(worker) {
const start = Date.now();
const tests = {};
do {
const test = queue.pop();
if (!test)
break;
const testStart = Date.now();
logger.highlight(`Running test ${test.fileName} in ${worker.profile ? worker.profile + '/' : ''}${worker.region}`);
const response = await options.pool.exec('integTestWorker', [{
watch: options.watch,
region: worker.region,
profile: worker.profile,
tests: [test],
clean: options.clean,
dryRun: options.dryRun,
verbosity: options.verbosity,
updateWorkflow: options.updateWorkflow,
engine: options.engine,
}], {
on: common_1.printResults,
});
results.failedTests.push(...(0, utils_1.flatten)(response));
tests[test.fileName] = (Date.now() - testStart) / 1000;
} while (queue.length > 0);
const metrics = {
region: worker.region,
profile: worker.profile,
duration: (Date.now() - start) / 1000,
tests,
};
if (Object.keys(tests).length > 0) {
results.metrics.push(metrics);
}
}
const workers = accountWorkers.map((worker) => runTest(worker));
// Workers are their own concurrency limits
// eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism
await Promise.all(workers);
return results;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"integ-test-worker.js","sourceRoot":"","sources":["integ-test-worker.ts"],"names":[],"mappings":"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","sourcesContent":["import type * as workerpool from 'workerpool';\nimport type { IntegBatchResponse, IntegTestOptions, IntegRunnerMetrics } from './common';\nimport { printResults, printSummary } from './common';\nimport * as logger from '../logger';\nimport type { IntegTestInfo } from '../runner/integration-tests';\nimport { flatten } from '../utils';\n\n/**\n * Options for an integration test batch\n */\nexport interface IntegTestBatchRequest extends IntegTestOptions {\n  /**\n   * The AWS region to run this batch in\n   */\n  readonly region: string;\n\n  /**\n   * The AWS profile to use when running this test\n   */\n  readonly profile?: string;\n}\n\n/**\n * Options for running all integration tests\n */\nexport interface IntegTestRunOptions extends IntegTestOptions {\n  /**\n   * The regions to run the integration tests across.\n   * This allows the runner to run integration tests in parallel\n   */\n  readonly regions: string[];\n\n  /**\n   * List of AWS profiles. This will be used in conjunction with `regions`\n   * to run tests in parallel across accounts + regions\n   */\n  readonly profiles?: string[];\n\n  /**\n   * The workerpool to use\n   */\n  readonly pool: workerpool.WorkerPool;\n}\n\n/**\n * Run Integration tests.\n */\nexport async function runIntegrationTests(options: IntegTestRunOptions): Promise<{ success: boolean; metrics: IntegRunnerMetrics[] }> {\n  logger.highlight('\\nRunning integration tests for failed tests...\\n');\n  logger.print(\n    'Running in parallel across %sregions: %s',\n    options.profiles ? `profiles ${options.profiles.join(', ')} and `: '',\n    options.regions.join(', '));\n  const totalTests = options.tests.length;\n\n  const responses = await runIntegrationTestsInParallel(options);\n  logger.highlight('\\nTest Results: \\n');\n  printSummary(totalTests, responses.failedTests.length);\n  return {\n    success: responses.failedTests.length === 0,\n    metrics: responses.metrics,\n  };\n}\n\n/**\n * Represents a worker for a single account + region\n */\ninterface AccountWorker {\n  /**\n   * The region the worker should run in\n   */\n  readonly region: string;\n\n  /**\n   * The AWS profile that the worker should use\n   * This will be passed as the '--profile' option to the CDK CLI\n   *\n   * @default - default profile\n   */\n  readonly profile?: string;\n}\n\n/**\n * Returns a list of AccountWorkers based on the list of regions and profiles\n * given to the CLI.\n */\nfunction getAccountWorkers(regions: string[], profiles?: string[]): AccountWorker[] {\n  const workers: AccountWorker[] = [];\n  function pushWorker(profile?: string) {\n    for (const region of regions) {\n      workers.push({\n        region,\n        profile,\n      });\n    }\n  }\n  if (profiles && profiles.length > 0) {\n    for (const profile of profiles ?? []) {\n      pushWorker(profile);\n    }\n  } else {\n    pushWorker();\n  }\n  return workers;\n}\n\n/**\n * Runs a set of integration tests in parallel across a list of AWS regions.\n * Only a single test can be run at a time in a given region. Once a region\n * is done running a test, the next test will be pulled from the queue\n */\nexport async function runIntegrationTestsInParallel(\n  options: IntegTestRunOptions,\n): Promise<IntegBatchResponse> {\n  const queue = options.tests;\n  const results: IntegBatchResponse = {\n    metrics: [],\n    failedTests: [],\n  };\n  const accountWorkers: AccountWorker[] = getAccountWorkers(options.regions, options.profiles);\n\n  async function runTest(worker: AccountWorker): Promise<void> {\n    const start = Date.now();\n    const tests: { [testName: string]: number } = {};\n    do {\n      const test = queue.pop();\n      if (!test) break;\n      const testStart = Date.now();\n      logger.highlight(`Running test ${test.fileName} in ${worker.profile ? worker.profile + '/' : ''}${worker.region}`);\n      const response: IntegTestInfo[][] = await options.pool.exec('integTestWorker', [{\n        watch: options.watch,\n        region: worker.region,\n        profile: worker.profile,\n        tests: [test],\n        clean: options.clean,\n        dryRun: options.dryRun,\n        verbosity: options.verbosity,\n        updateWorkflow: options.updateWorkflow,\n        engine: options.engine,\n      }], {\n        on: printResults,\n      });\n\n      results.failedTests.push(...flatten(response));\n      tests[test.fileName] = (Date.now() - testStart) / 1000;\n    } while (queue.length > 0);\n    const metrics: IntegRunnerMetrics = {\n      region: worker.region,\n      profile: worker.profile,\n      duration: (Date.now() - start) / 1000,\n      tests,\n    };\n    if (Object.keys(tests).length > 0) {\n      results.metrics.push(metrics);\n    }\n  }\n\n  const workers = accountWorkers.map((worker) => runTest(worker));\n  // Workers are their own concurrency limits\n  // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n  await Promise.all(workers);\n  return results;\n}\n"]}