@aws-cdk/integ-runner 2.50.0 → 2.51.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -38,7 +38,7 @@ integ-runner [ARGS] [TEST...]
38
38
  This will look for all files that match the naming convention of `/integ.*.js$/`. Each of these files will be expected
39
39
  to be a self contained CDK app. The runner will execute the following for each file (app):
40
40
 
41
- 1. Check if a snapshot file exists (i.e. `/*.integ.snapshot$/`)
41
+ 1. Check if a snapshot file exists (i.e. `/*.snapshot$/`)
42
42
  2. If the snapshot does not exist
43
43
  2a. Synth the integ app which will produce the `integ.json` file
44
44
  3. Read the `integ.json` file which contains instructions on what the runner should do.
@@ -68,7 +68,11 @@ to be a self contained CDK app. The runner will execute the following for each f
68
68
  Read the list of tests from this file
69
69
  - `--disable-update-workflow` (default=`false`)
70
70
  If this is set to `true` then the [update workflow](#update-workflow) will be disabled
71
-
71
+ - `--app`
72
+ The custom CLI command that will be used to run the test files. You can include {filePath} to specify where in the command the test file path should be inserted. Example: --app="python3.8 {filePath}".
73
+ - `--test-regex`
74
+ Detect integration test files matching this JavaScript regex pattern. If used multiple times, all files matching any one of the patterns are detected.
75
+
72
76
  Example:
73
77
 
74
78
  ```bash
@@ -81,7 +85,7 @@ If you are providing a list of tests to execute, either as CLI arguments or from
81
85
  For example, if there is a test `aws-iam/test/integ.policy.js` and the current working directory is `aws-iam` you would provide `integ.policy.js`
82
86
 
83
87
  ```bash
84
- yarn integ integ.policy.js
88
+ integ-runner integ.policy.js
85
89
  ```
86
90
 
87
91
  ### Common Workflow
@@ -117,7 +121,7 @@ Snapshot Results:
117
121
 
118
122
  Tests: 1 failed, 9 total
119
123
  Error: Some snapshot tests failed!
120
- To re-run failed tests run: yarn integ-runner --update-on-failed
124
+ To re-run failed tests run: integ-runner --update-on-failed
121
125
  at main (packages/@aws-cdk/integ-runner/lib/cli.js:90:15)
122
126
  error Command failed with exit code 1.
123
127
  ```
@@ -165,6 +169,8 @@ Test Results:
165
169
  Tests: 1 passed, 1 total
166
170
  ```
167
171
 
172
+ Nested stack templates are also compared as part of the snapshot. However asset hashes are ignored by default. To enable diff for asset hashes, set `diffAssets: true` of `IntegTestProps`.
173
+
168
174
  #### Update Workflow
169
175
 
170
176
  By default, integration tests are run with the "update workflow" enabled. This can be disabled by using the `--disable-update-workflow` command line option.
@@ -191,7 +197,7 @@ If you are adding a new test which creates a new snapshot then you should run th
191
197
  For example, if you are working on a new test `integ.new-test.js` then you would run:
192
198
 
193
199
  ```bash
194
- yarn integ --update-on-failed --disable-update-workflow integ.new-test.js
200
+ integ-runner --update-on-failed --disable-update-workflow integ.new-test.js
195
201
  ```
196
202
 
197
203
  This is because for a new test we do not need to test the update workflow (there is nothing to update).
@@ -204,3 +210,25 @@ See [@aws-cdk/cloud-assembly-schema/lib/integ-tests/schema.ts](../cloud-assembly
204
210
 
205
211
  See the `@aws-cdk/integ-tests` module for information on how to define
206
212
  integration tests for the runner to exercise.
213
+
214
+ ### Config file
215
+
216
+ All options can be configured via the `integ.config.json` configuration file in the current working directory.
217
+
218
+ ```json
219
+ {
220
+ "maxWorkers": 10,
221
+ "parallelRegions": [
222
+ "eu-west-1",
223
+ "ap-southeast-2"
224
+ ]
225
+ }
226
+ ```
227
+
228
+ Available options can be listed by running the following command:
229
+
230
+ ```sh
231
+ integ-runner --help
232
+ ```
233
+
234
+ To use a different config file, provide the `--config` command-line option.
@@ -1,6 +1,6 @@
1
1
  The @aws-cdk/integ-runner package includes the following third-party software/licensing:
2
2
 
3
- ** ajv@8.11.0 - https://www.npmjs.com/package/ajv/v/8.11.0 | MIT
3
+ ** ajv@8.11.2 - https://www.npmjs.com/package/ajv/v/8.11.2 | MIT
4
4
  The MIT License (MIT)
5
5
 
6
6
  Copyright (c) 2015-2021 Evgeny Poberezkin
@@ -156,7 +156,7 @@ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH RE
156
156
 
157
157
  ----------------
158
158
 
159
- ** aws-sdk@2.1240.0 - https://www.npmjs.com/package/aws-sdk/v/2.1240.0 | Apache-2.0
159
+ ** aws-sdk@2.1255.0 - https://www.npmjs.com/package/aws-sdk/v/2.1255.0 | Apache-2.0
160
160
  AWS SDK for JavaScript
161
161
  Copyright 2012-2017 Amazon.com, Inc. or its affiliates. All Rights Reserved.
162
162
 
@@ -2151,7 +2151,7 @@ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLI
2151
2151
 
2152
2152
  ----------------
2153
2153
 
2154
- ** table@6.8.0 - https://www.npmjs.com/package/table/v/6.8.0 | BSD-3-Clause
2154
+ ** table@6.8.1 - https://www.npmjs.com/package/table/v/6.8.1 | BSD-3-Clause
2155
2155
  Copyright (c) 2018, Gajus Kuizinas (http://gajus.com/)
2156
2156
  All rights reserved.
2157
2157
 
@@ -2263,7 +2263,7 @@ OTHER DEALINGS IN THE SOFTWARE.
2263
2263
 
2264
2264
  ----------------
2265
2265
 
2266
- ** workerpool@6.3.0 - https://www.npmjs.com/package/workerpool/v/6.3.0 | Apache-2.0
2266
+ ** workerpool@6.3.1 - https://www.npmjs.com/package/workerpool/v/6.3.1 | Apache-2.0
2267
2267
  Apache License
2268
2268
  Version 2.0, January 2004
2269
2269
  http://www.apache.org/licenses/
package/lib/cli.d.ts CHANGED
@@ -1 +1,22 @@
1
- export declare function cli(): void;
1
+ export declare function parseCliArgs(args?: string[]): {
2
+ tests: string[] | undefined;
3
+ app: string | undefined;
4
+ testRegex: string[] | undefined;
5
+ testRegions: string[];
6
+ profiles: string[] | undefined;
7
+ runUpdateOnFailed: boolean;
8
+ fromFile: string | undefined;
9
+ exclude: boolean;
10
+ maxWorkers: number;
11
+ list: boolean;
12
+ directory: string;
13
+ inspectFailures: boolean;
14
+ verbosity: number;
15
+ verbose: boolean;
16
+ clean: boolean;
17
+ force: boolean;
18
+ dryRun: boolean;
19
+ disableUpdateWorkflow: boolean;
20
+ };
21
+ export declare function main(args: string[]): Promise<void>;
22
+ export declare function cli(args?: string[]): void;
package/lib/cli.js CHANGED
@@ -1,8 +1,8 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.cli = void 0;
3
+ exports.cli = exports.main = exports.parseCliArgs = void 0;
4
4
  // Exercise all integ stacks and if they deploy, update the expected synth files
5
- const fs_1 = require("fs");
5
+ const fs = require("fs");
6
6
  const path = require("path");
7
7
  const chalk = require("chalk");
8
8
  const workerpool = require("workerpool");
@@ -13,10 +13,16 @@ const workers_1 = require("./workers");
13
13
  // https://github.com/evanw/esbuild/issues/1492
14
14
  // eslint-disable-next-line @typescript-eslint/no-require-imports
15
15
  const yargs = require('yargs');
16
- async function main() {
17
- var _a, _b;
16
+ function parseCliArgs(args = []) {
17
+ var _a;
18
18
  const argv = yargs
19
19
  .usage('Usage: integ-runner [TEST...]')
20
+ .option('config', {
21
+ config: true,
22
+ configParser: integration_tests_1.IntegrationTests.configFromFile,
23
+ default: 'integ.config.json',
24
+ desc: 'Load options from a JSON config file. Options provided as CLI arguments take precedent.',
25
+ })
20
26
  .option('list', { type: 'boolean', default: false, desc: 'List tests instead of running them' })
21
27
  .option('clean', { type: 'boolean', default: true, desc: 'Skips stack clean up after test is completed (use --no-clean to negate)' })
22
28
  .option('verbose', { type: 'boolean', default: false, alias: 'v', count: true, desc: 'Verbose logs and metrics on integration tests durations (specify multiple times to increase verbosity)' })
@@ -31,50 +37,83 @@ async function main() {
31
37
  .options('from-file', { type: 'string', desc: 'Read TEST names from a file (one TEST per line)' })
32
38
  .option('inspect-failures', { type: 'boolean', desc: 'Keep the integ test cloud assembly if a failure occurs for inspection', default: false })
33
39
  .option('disable-update-workflow', { type: 'boolean', default: false, desc: 'If this is "true" then the stack update workflow will be disabled' })
40
+ .option('app', { type: 'string', default: undefined, desc: 'The custom CLI command that will be used to run the test files. You can include {filePath} to specify where in the command the test file path should be inserted. Example: --app="python3.8 {filePath}".' })
41
+ .option('test-regex', { type: 'array', desc: 'Detect integration test files matching this JavaScript regex pattern. If used multiple times, all files matching any one of the patterns are detected.', default: [] })
34
42
  .strict()
35
- .argv;
36
- const pool = workerpool.pool(path.join(__dirname, '../lib/workers/extract/index.js'), {
37
- maxWorkers: argv['max-workers'],
38
- });
39
- // list of integration tests that will be executed
40
- const testsToRun = [];
41
- const destructiveChanges = [];
42
- const testsFromArgs = [];
43
+ .parse(args);
44
+ const tests = argv._;
43
45
  const parallelRegions = arrayFromYargs(argv['parallel-regions']);
44
46
  const testRegions = parallelRegions !== null && parallelRegions !== void 0 ? parallelRegions : ['us-east-1', 'us-east-2', 'us-west-2'];
45
47
  const profiles = arrayFromYargs(argv.profiles);
46
- const runUpdateOnFailed = (_a = argv['update-on-failed']) !== null && _a !== void 0 ? _a : false;
47
48
  const fromFile = argv['from-file'];
48
- const exclude = argv.exclude;
49
- let failedSnapshots = [];
50
- if (argv['max-workers'] < testRegions.length * (profiles !== null && profiles !== void 0 ? profiles : [1]).length) {
51
- logger.warning('You are attempting to run %s tests in parallel, but only have %s workers. Not all of your profiles+regions will be utilized', argv.profiles * argv['parallel-regions'], argv['max-workers']);
49
+ const maxWorkers = argv['max-workers'];
50
+ const verbosity = argv.verbose;
51
+ const verbose = verbosity >= 1;
52
+ const numTests = testRegions.length * (profiles !== null && profiles !== void 0 ? profiles : [1]).length;
53
+ if (maxWorkers < numTests) {
54
+ logger.warning('You are attempting to run %s tests in parallel, but only have %s workers. Not all of your profiles+regions will be utilized', numTests, maxWorkers);
55
+ }
56
+ if (tests.length > 0 && fromFile) {
57
+ throw new Error('A list of tests cannot be provided if "--from-file" is provided');
58
+ }
59
+ const requestedTests = fromFile
60
+ ? (fs.readFileSync(fromFile, { encoding: 'utf8' })).split('\n').filter(x => x)
61
+ : (tests.length > 0 ? tests : undefined); // 'undefined' means no request
62
+ return {
63
+ tests: requestedTests,
64
+ app: argv.app,
65
+ testRegex: arrayFromYargs(argv['test-regex']),
66
+ testRegions,
67
+ profiles,
68
+ runUpdateOnFailed: ((_a = argv['update-on-failed']) !== null && _a !== void 0 ? _a : false),
69
+ fromFile,
70
+ exclude: argv.exclude,
71
+ maxWorkers,
72
+ list: argv.list,
73
+ directory: argv.directory,
74
+ inspectFailures: argv['inspect-failures'],
75
+ verbosity,
76
+ verbose,
77
+ clean: argv.clean,
78
+ force: argv.force,
79
+ dryRun: argv['dry-run'],
80
+ disableUpdateWorkflow: argv['disable-update-workflow'],
81
+ };
82
+ }
83
+ exports.parseCliArgs = parseCliArgs;
84
+ async function main(args) {
85
+ var _a;
86
+ const options = parseCliArgs(args);
87
+ const testsFromArgs = await new integration_tests_1.IntegrationTests(path.resolve(options.directory)).fromCliArgs({
88
+ app: options.app,
89
+ testRegex: options.testRegex,
90
+ tests: options.tests,
91
+ exclude: options.exclude,
92
+ });
93
+ // List only prints the discoverd tests
94
+ if (options.list) {
95
+ process.stdout.write(testsFromArgs.map(t => t.discoveryRelativeFileName).join('\n') + '\n');
96
+ return;
52
97
  }
98
+ const pool = workerpool.pool(path.join(__dirname, '../lib/workers/extract/index.js'), {
99
+ maxWorkers: options.maxWorkers,
100
+ });
101
+ const testsToRun = [];
102
+ const destructiveChanges = [];
103
+ let failedSnapshots = [];
53
104
  let testsSucceeded = false;
54
105
  try {
55
- if (argv.list) {
56
- const tests = await new integration_tests_1.IntegrationTests(argv.directory).fromCliArgs();
57
- process.stdout.write(tests.map(t => t.discoveryRelativeFileName).join('\n') + '\n');
58
- return;
59
- }
60
- if (argv._.length > 0 && fromFile) {
61
- throw new Error('A list of tests cannot be provided if "--from-file" is provided');
62
- }
63
- const requestedTests = fromFile
64
- ? (await fs_1.promises.readFile(fromFile, { encoding: 'utf8' })).split('\n').filter(x => x)
65
- : (argv._.length > 0 ? argv._ : undefined); // 'undefined' means no request
66
- testsFromArgs.push(...(await new integration_tests_1.IntegrationTests(path.resolve(argv.directory)).fromCliArgs(requestedTests, exclude)));
67
106
  // always run snapshot tests, but if '--force' is passed then
68
107
  // run integration tests on all failed tests, not just those that
69
108
  // failed snapshot tests
70
109
  failedSnapshots = await workers_1.runSnapshotTests(pool, testsFromArgs, {
71
- retain: argv['inspect-failures'],
72
- verbose: Boolean(argv.verbose),
110
+ retain: options.inspectFailures,
111
+ verbose: options.verbose,
73
112
  });
74
113
  for (const failure of failedSnapshots) {
75
- destructiveChanges.push(...(_b = failure.destructiveChanges) !== null && _b !== void 0 ? _b : []);
114
+ destructiveChanges.push(...(_a = failure.destructiveChanges) !== null && _a !== void 0 ? _a : []);
76
115
  }
77
- if (!argv.force) {
116
+ if (!options.force) {
78
117
  testsToRun.push(...failedSnapshots);
79
118
  }
80
119
  else {
@@ -83,22 +122,22 @@ async function main() {
83
122
  testsToRun.push(...mergeTests(testsFromArgs.map(t => t.info), failedSnapshots));
84
123
  }
85
124
  // run integration tests if `--update-on-failed` OR `--force` is used
86
- if (runUpdateOnFailed || argv.force) {
125
+ if (options.runUpdateOnFailed || options.force) {
87
126
  const { success, metrics } = await workers_1.runIntegrationTests({
88
127
  pool,
89
128
  tests: testsToRun,
90
- regions: testRegions,
91
- profiles,
92
- clean: argv.clean,
93
- dryRun: argv['dry-run'],
94
- verbosity: argv.verbose,
95
- updateWorkflow: !argv['disable-update-workflow'],
129
+ regions: options.testRegions,
130
+ profiles: options.profiles,
131
+ clean: options.clean,
132
+ dryRun: options.dryRun,
133
+ verbosity: options.verbosity,
134
+ updateWorkflow: !options.disableUpdateWorkflow,
96
135
  });
97
136
  testsSucceeded = success;
98
- if (argv.clean === false) {
137
+ if (options.clean === false) {
99
138
  logger.warning('Not cleaning up stacks since "--no-clean" was used');
100
139
  }
101
- if (Boolean(argv.verbose)) {
140
+ if (Boolean(options.verbose)) {
102
141
  printMetrics(metrics);
103
142
  }
104
143
  if (!success) {
@@ -115,14 +154,15 @@ async function main() {
115
154
  }
116
155
  if (failedSnapshots.length > 0) {
117
156
  let message = '';
118
- if (!runUpdateOnFailed) {
119
- message = 'To re-run failed tests run: yarn integ-runner --update-on-failed';
157
+ if (!options.runUpdateOnFailed) {
158
+ message = 'To re-run failed tests run: integ-runner --update-on-failed';
120
159
  }
121
160
  if (!testsSucceeded) {
122
161
  throw new Error(`Some tests failed!\n${message}`);
123
162
  }
124
163
  }
125
164
  }
165
+ exports.main = main;
126
166
  function printDestructiveChanges(changes) {
127
167
  if (changes.length > 0) {
128
168
  logger.warning('!!! This test contains %s !!!', chalk.bold('destructive changes'));
@@ -167,11 +207,11 @@ function mergeTests(testFromArgs, failedSnapshotTests) {
167
207
  final.push(...testFromArgs.filter(test => !failedTestNames.has(test.fileName)));
168
208
  return final;
169
209
  }
170
- function cli() {
171
- main().then().catch(err => {
210
+ function cli(args = process.argv.slice(2)) {
211
+ main(args).then().catch(err => {
172
212
  logger.error(err);
173
213
  process.exitCode = 1;
174
214
  });
175
215
  }
176
216
  exports.cli = cli;
177
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli.js","sourceRoot":"","sources":["cli.ts"],"names":[],"mappings":"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","sourcesContent":["// Exercise all integ stacks and if they deploy, update the expected synth files\nimport { promises as fs } from 'fs';\nimport * as path from 'path';\nimport * as chalk from 'chalk';\nimport * as workerpool from 'workerpool';\nimport * as logger from './logger';\nimport { IntegrationTests, IntegTestInfo, IntegTest } from './runner/integration-tests';\nimport { runSnapshotTests, runIntegrationTests, IntegRunnerMetrics, IntegTestWorkerConfig, DestructiveChange } from './workers';\n\n// https://github.com/yargs/yargs/issues/1929\n// https://github.com/evanw/esbuild/issues/1492\n// eslint-disable-next-line @typescript-eslint/no-require-imports\nconst yargs = require('yargs');\n\n\nasync function main() {\n  const argv = yargs\n    .usage('Usage: integ-runner [TEST...]')\n    .option('list', { type: 'boolean', default: false, desc: 'List tests instead of running them' })\n    .option('clean', { type: 'boolean', default: true, desc: 'Skips stack clean up after test is completed (use --no-clean to negate)' })\n    .option('verbose', { type: 'boolean', default: false, alias: 'v', count: true, desc: 'Verbose logs and metrics on integration tests durations (specify multiple times to increase verbosity)' })\n    .option('dry-run', { type: 'boolean', default: false, desc: 'do not actually deploy the stack. just update the snapshot (not recommended!)' })\n    .option('update-on-failed', { type: 'boolean', default: false, desc: 'rerun integration tests and update snapshots for failed tests.' })\n    .option('force', { type: 'boolean', default: false, desc: 'Rerun all integration tests even if tests are passing' })\n    .option('parallel-regions', { type: 'array', desc: 'Tests are run in parallel across these regions. To prevent tests from running in parallel, provide only a single region', default: [] })\n    .options('directory', { type: 'string', default: 'test', desc: 'starting directory to discover integration tests. Tests will be discovered recursively from this directory' })\n    .options('profiles', { type: 'array', desc: 'list of AWS profiles to use. Tests will be run in parallel across each profile+regions', default: [] })\n    .options('max-workers', { type: 'number', desc: 'The max number of workerpool workers to use when running integration tests in parallel', default: 16 })\n    .options('exclude', { type: 'boolean', desc: 'Run all tests in the directory, except the specified TESTs', default: false })\n    .options('from-file', { type: 'string', desc: 'Read TEST names from a file (one TEST per line)' })\n    .option('inspect-failures', { type: 'boolean', desc: 'Keep the integ test cloud assembly if a failure occurs for inspection', default: false })\n    .option('disable-update-workflow', { type: 'boolean', default: false, desc: 'If this is \"true\" then the stack update workflow will be disabled' })\n    .strict()\n    .argv;\n\n  const pool = workerpool.pool(path.join(__dirname, '../lib/workers/extract/index.js'), {\n    maxWorkers: argv['max-workers'],\n  });\n\n  // list of integration tests that will be executed\n  const testsToRun: IntegTestWorkerConfig[] = [];\n  const destructiveChanges: DestructiveChange[] = [];\n  const testsFromArgs: IntegTest[] = [];\n  const parallelRegions = arrayFromYargs(argv['parallel-regions']);\n  const testRegions: string[] = parallelRegions ?? ['us-east-1', 'us-east-2', 'us-west-2'];\n  const profiles = arrayFromYargs(argv.profiles);\n  const runUpdateOnFailed = argv['update-on-failed'] ?? false;\n  const fromFile: string | undefined = argv['from-file'];\n  const exclude: boolean = argv.exclude;\n\n  let failedSnapshots: IntegTestWorkerConfig[] = [];\n  if (argv['max-workers'] < testRegions.length * (profiles ?? [1]).length) {\n    logger.warning('You are attempting to run %s tests in parallel, but only have %s workers. Not all of your profiles+regions will be utilized', argv.profiles * argv['parallel-regions'], argv['max-workers']);\n  }\n\n  let testsSucceeded = false;\n  try {\n    if (argv.list) {\n      const tests = await new IntegrationTests(argv.directory).fromCliArgs();\n      process.stdout.write(tests.map(t => t.discoveryRelativeFileName).join('\\n') + '\\n');\n      return;\n    }\n\n    if (argv._.length > 0 && fromFile) {\n      throw new Error('A list of tests cannot be provided if \"--from-file\" is provided');\n    }\n    const requestedTests = fromFile\n      ? (await fs.readFile(fromFile, { encoding: 'utf8' })).split('\\n').filter(x => x)\n      : (argv._.length > 0 ? argv._ : undefined); // 'undefined' means no request\n\n    testsFromArgs.push(...(await new IntegrationTests(path.resolve(argv.directory)).fromCliArgs(requestedTests, exclude)));\n\n    // always run snapshot tests, but if '--force' is passed then\n    // run integration tests on all failed tests, not just those that\n    // failed snapshot tests\n    failedSnapshots = await runSnapshotTests(pool, testsFromArgs, {\n      retain: argv['inspect-failures'],\n      verbose: Boolean(argv.verbose),\n    });\n    for (const failure of failedSnapshots) {\n      destructiveChanges.push(...failure.destructiveChanges ?? []);\n    }\n    if (!argv.force) {\n      testsToRun.push(...failedSnapshots);\n    } else {\n      // if any of the test failed snapshot tests, keep those results\n      // and merge with the rest of the tests from args\n      testsToRun.push(...mergeTests(testsFromArgs.map(t => t.info), failedSnapshots));\n    }\n\n    // run integration tests if `--update-on-failed` OR `--force` is used\n    if (runUpdateOnFailed || argv.force) {\n      const { success, metrics } = await runIntegrationTests({\n        pool,\n        tests: testsToRun,\n        regions: testRegions,\n        profiles,\n        clean: argv.clean,\n        dryRun: argv['dry-run'],\n        verbosity: argv.verbose,\n        updateWorkflow: !argv['disable-update-workflow'],\n      });\n      testsSucceeded = success;\n\n\n      if (argv.clean === false) {\n        logger.warning('Not cleaning up stacks since \"--no-clean\" was used');\n      }\n\n      if (Boolean(argv.verbose)) {\n        printMetrics(metrics);\n      }\n\n      if (!success) {\n        throw new Error('Some integration tests failed!');\n      }\n    }\n  } finally {\n    void pool.terminate();\n  }\n\n  if (destructiveChanges.length > 0) {\n    printDestructiveChanges(destructiveChanges);\n    throw new Error('Some changes were destructive!');\n  }\n  if (failedSnapshots.length > 0) {\n    let message = '';\n    if (!runUpdateOnFailed) {\n      message = 'To re-run failed tests run: yarn integ-runner --update-on-failed';\n    }\n    if (!testsSucceeded) {\n      throw new Error(`Some tests failed!\\n${message}`);\n    }\n  }\n\n}\n\nfunction printDestructiveChanges(changes: DestructiveChange[]): void {\n  if (changes.length > 0) {\n    logger.warning('!!! This test contains %s !!!', chalk.bold('destructive changes'));\n    changes.forEach(change => {\n      logger.warning('    Stack: %s - Resource: %s - Impact: %s', change.stackName, change.logicalId, change.impact);\n    });\n    logger.warning('!!! If these destructive changes are necessary, please indicate this on the PR !!!');\n  }\n}\n\nfunction printMetrics(metrics: IntegRunnerMetrics[]): void {\n  logger.highlight('   --- Integration test metrics ---');\n  const sortedMetrics = metrics.sort((a, b) => a.duration - b.duration);\n  sortedMetrics.forEach(metric => {\n    logger.print('Profile %s + Region %s total time: %s', metric.profile, metric.region, metric.duration);\n    const sortedTests = Object.entries(metric.tests).sort((a, b) => a[1] - b[1]);\n    sortedTests.forEach(test => logger.print('  %s: %s', test[0], test[1]));\n  });\n}\n\n/**\n * Translate a Yargs input array to something that makes more sense in a programming language\n * model (telling the difference between absence and an empty array)\n *\n * - An empty array is the default case, meaning the user didn't pass any arguments. We return\n *   undefined.\n * - If the user passed a single empty string, they did something like `--array=`, which we'll\n *   take to mean they passed an empty array.\n */\nfunction arrayFromYargs(xs: string[]): string[] | undefined {\n  if (xs.length === 0) { return undefined; }\n  return xs.filter(x => x !== '');\n}\n\n/**\n * Merge the tests we received from command line arguments with\n * tests that failed snapshot tests. The failed snapshot tests have additional\n * information that we want to keep so this should override any test from args\n */\nfunction mergeTests(testFromArgs: IntegTestInfo[], failedSnapshotTests: IntegTestWorkerConfig[]): IntegTestWorkerConfig[] {\n  const failedTestNames = new Set(failedSnapshotTests.map(test => test.fileName));\n  const final: IntegTestWorkerConfig[] = failedSnapshotTests;\n  final.push(...testFromArgs.filter(test => !failedTestNames.has(test.fileName)));\n  return final;\n}\n\nexport function cli() {\n  main().then().catch(err => {\n    logger.error(err);\n    process.exitCode = 1;\n  });\n}\n"]}
217
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli.js","sourceRoot":"","sources":["cli.ts"],"names":[],"mappings":"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","sourcesContent":["// Exercise all integ stacks and if they deploy, update the expected synth files\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport * as chalk from 'chalk';\nimport * as workerpool from 'workerpool';\nimport * as logger from './logger';\nimport { IntegrationTests, IntegTestInfo } from './runner/integration-tests';\nimport { runSnapshotTests, runIntegrationTests, IntegRunnerMetrics, IntegTestWorkerConfig, DestructiveChange } from './workers';\n\n// https://github.com/yargs/yargs/issues/1929\n// https://github.com/evanw/esbuild/issues/1492\n// eslint-disable-next-line @typescript-eslint/no-require-imports\nconst yargs = require('yargs');\n\nexport function parseCliArgs(args: string[] = []) {\n  const argv = yargs\n    .usage('Usage: integ-runner [TEST...]')\n    .option('config', {\n      config: true,\n      configParser: IntegrationTests.configFromFile,\n      default: 'integ.config.json',\n      desc: 'Load options from a JSON config file. Options provided as CLI arguments take precedent.',\n    })\n    .option('list', { type: 'boolean', default: false, desc: 'List tests instead of running them' })\n    .option('clean', { type: 'boolean', default: true, desc: 'Skips stack clean up after test is completed (use --no-clean to negate)' })\n    .option('verbose', { type: 'boolean', default: false, alias: 'v', count: true, desc: 'Verbose logs and metrics on integration tests durations (specify multiple times to increase verbosity)' })\n    .option('dry-run', { type: 'boolean', default: false, desc: 'do not actually deploy the stack. just update the snapshot (not recommended!)' })\n    .option('update-on-failed', { type: 'boolean', default: false, desc: 'rerun integration tests and update snapshots for failed tests.' })\n    .option('force', { type: 'boolean', default: false, desc: 'Rerun all integration tests even if tests are passing' })\n    .option('parallel-regions', { type: 'array', desc: 'Tests are run in parallel across these regions. To prevent tests from running in parallel, provide only a single region', default: [] })\n    .options('directory', { type: 'string', default: 'test', desc: 'starting directory to discover integration tests. Tests will be discovered recursively from this directory' })\n    .options('profiles', { type: 'array', desc: 'list of AWS profiles to use. Tests will be run in parallel across each profile+regions', default: [] })\n    .options('max-workers', { type: 'number', desc: 'The max number of workerpool workers to use when running integration tests in parallel', default: 16 })\n    .options('exclude', { type: 'boolean', desc: 'Run all tests in the directory, except the specified TESTs', default: false })\n    .options('from-file', { type: 'string', desc: 'Read TEST names from a file (one TEST per line)' })\n    .option('inspect-failures', { type: 'boolean', desc: 'Keep the integ test cloud assembly if a failure occurs for inspection', default: false })\n    .option('disable-update-workflow', { type: 'boolean', default: false, desc: 'If this is \"true\" then the stack update workflow will be disabled' })\n    .option('app', { type: 'string', default: undefined, desc: 'The custom CLI command that will be used to run the test files. You can include {filePath} to specify where in the command the test file path should be inserted. Example: --app=\"python3.8 {filePath}\".' })\n    .option('test-regex', { type: 'array', desc: 'Detect integration test files matching this JavaScript regex pattern. If used multiple times, all files matching any one of the patterns are detected.', default: [] })\n    .strict()\n    .parse(args);\n\n  const tests: string[] = argv._;\n  const parallelRegions = arrayFromYargs(argv['parallel-regions']);\n  const testRegions: string[] = parallelRegions ?? ['us-east-1', 'us-east-2', 'us-west-2'];\n  const profiles = arrayFromYargs(argv.profiles);\n  const fromFile: string | undefined = argv['from-file'];\n  const maxWorkers: number = argv['max-workers'];\n  const verbosity: number = argv.verbose;\n  const verbose: boolean = verbosity >= 1;\n\n  const numTests = testRegions.length * (profiles ?? [1]).length;\n  if (maxWorkers < numTests) {\n    logger.warning('You are attempting to run %s tests in parallel, but only have %s workers. Not all of your profiles+regions will be utilized', numTests, maxWorkers);\n  }\n\n  if (tests.length > 0 && fromFile) {\n    throw new Error('A list of tests cannot be provided if \"--from-file\" is provided');\n  }\n  const requestedTests = fromFile\n    ? (fs.readFileSync(fromFile, { encoding: 'utf8' })).split('\\n').filter(x => x)\n    : (tests.length > 0 ? tests : undefined); // 'undefined' means no request\n\n  return {\n    tests: requestedTests,\n    app: argv.app as (string | undefined),\n    testRegex: arrayFromYargs(argv['test-regex']),\n    testRegions,\n    profiles,\n    runUpdateOnFailed: (argv['update-on-failed'] ?? false) as boolean,\n    fromFile,\n    exclude: argv.exclude as boolean,\n    maxWorkers,\n    list: argv.list as boolean,\n    directory: argv.directory as string,\n    inspectFailures: argv['inspect-failures'] as boolean,\n    verbosity,\n    verbose,\n    clean: argv.clean as boolean,\n    force: argv.force as boolean,\n    dryRun: argv['dry-run'] as boolean,\n    disableUpdateWorkflow: argv['disable-update-workflow'] as boolean,\n  };\n}\n\n\nexport async function main(args: string[]) {\n  const options = parseCliArgs(args);\n\n  const testsFromArgs = await new IntegrationTests(path.resolve(options.directory)).fromCliArgs({\n    app: options.app,\n    testRegex: options.testRegex,\n    tests: options.tests,\n    exclude: options.exclude,\n  });\n\n  // List only prints the discoverd tests\n  if (options.list) {\n    process.stdout.write(testsFromArgs.map(t => t.discoveryRelativeFileName).join('\\n') + '\\n');\n    return;\n  }\n\n  const pool = workerpool.pool(path.join(__dirname, '../lib/workers/extract/index.js'), {\n    maxWorkers: options.maxWorkers,\n  });\n\n  const testsToRun: IntegTestWorkerConfig[] = [];\n  const destructiveChanges: DestructiveChange[] = [];\n  let failedSnapshots: IntegTestWorkerConfig[] = [];\n  let testsSucceeded = false;\n\n  try {\n    // always run snapshot tests, but if '--force' is passed then\n    // run integration tests on all failed tests, not just those that\n    // failed snapshot tests\n    failedSnapshots = await runSnapshotTests(pool, testsFromArgs, {\n      retain: options.inspectFailures,\n      verbose: options.verbose,\n    });\n    for (const failure of failedSnapshots) {\n      destructiveChanges.push(...failure.destructiveChanges ?? []);\n    }\n    if (!options.force) {\n      testsToRun.push(...failedSnapshots);\n    } else {\n      // if any of the test failed snapshot tests, keep those results\n      // and merge with the rest of the tests from args\n      testsToRun.push(...mergeTests(testsFromArgs.map(t => t.info), failedSnapshots));\n    }\n\n    // run integration tests if `--update-on-failed` OR `--force` is used\n    if (options.runUpdateOnFailed || options.force) {\n      const { success, metrics } = await runIntegrationTests({\n        pool,\n        tests: testsToRun,\n        regions: options.testRegions,\n        profiles: options.profiles,\n        clean: options.clean,\n        dryRun: options.dryRun,\n        verbosity: options.verbosity,\n        updateWorkflow: !options.disableUpdateWorkflow,\n      });\n      testsSucceeded = success;\n\n\n      if (options.clean === false) {\n        logger.warning('Not cleaning up stacks since \"--no-clean\" was used');\n      }\n\n      if (Boolean(options.verbose)) {\n        printMetrics(metrics);\n      }\n\n      if (!success) {\n        throw new Error('Some integration tests failed!');\n      }\n    }\n  } finally {\n    void pool.terminate();\n  }\n\n  if (destructiveChanges.length > 0) {\n    printDestructiveChanges(destructiveChanges);\n    throw new Error('Some changes were destructive!');\n  }\n  if (failedSnapshots.length > 0) {\n    let message = '';\n    if (!options.runUpdateOnFailed) {\n      message = 'To re-run failed tests run: integ-runner --update-on-failed';\n    }\n    if (!testsSucceeded) {\n      throw new Error(`Some tests failed!\\n${message}`);\n    }\n  }\n\n}\n\nfunction printDestructiveChanges(changes: DestructiveChange[]): void {\n  if (changes.length > 0) {\n    logger.warning('!!! This test contains %s !!!', chalk.bold('destructive changes'));\n    changes.forEach(change => {\n      logger.warning('    Stack: %s - Resource: %s - Impact: %s', change.stackName, change.logicalId, change.impact);\n    });\n    logger.warning('!!! If these destructive changes are necessary, please indicate this on the PR !!!');\n  }\n}\n\nfunction printMetrics(metrics: IntegRunnerMetrics[]): void {\n  logger.highlight('   --- Integration test metrics ---');\n  const sortedMetrics = metrics.sort((a, b) => a.duration - b.duration);\n  sortedMetrics.forEach(metric => {\n    logger.print('Profile %s + Region %s total time: %s', metric.profile, metric.region, metric.duration);\n    const sortedTests = Object.entries(metric.tests).sort((a, b) => a[1] - b[1]);\n    sortedTests.forEach(test => logger.print('  %s: %s', test[0], test[1]));\n  });\n}\n\n/**\n * Translate a Yargs input array to something that makes more sense in a programming language\n * model (telling the difference between absence and an empty array)\n *\n * - An empty array is the default case, meaning the user didn't pass any arguments. We return\n *   undefined.\n * - If the user passed a single empty string, they did something like `--array=`, which we'll\n *   take to mean they passed an empty array.\n */\nfunction arrayFromYargs(xs: string[]): string[] | undefined {\n  if (xs.length === 0) { return undefined; }\n  return xs.filter(x => x !== '');\n}\n\n/**\n * Merge the tests we received from command line arguments with\n * tests that failed snapshot tests. The failed snapshot tests have additional\n * information that we want to keep so this should override any test from args\n */\nfunction mergeTests(testFromArgs: IntegTestInfo[], failedSnapshotTests: IntegTestWorkerConfig[]): IntegTestWorkerConfig[] {\n  const failedTestNames = new Set(failedSnapshotTests.map(test => test.fileName));\n  const final: IntegTestWorkerConfig[] = failedSnapshotTests;\n  final.push(...testFromArgs.filter(test => !failedTestNames.has(test.fileName)));\n  return final;\n}\n\nexport function cli(args: string[] = process.argv.slice(2)) {\n  main(args).then().catch(err => {\n    logger.error(err);\n    process.exitCode = 1;\n  });\n}\n"]}