@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 +33 -5
- package/THIRD_PARTY_LICENSES +4 -4
- package/lib/cli.d.ts +22 -1
- package/lib/cli.js +87 -47
- package/lib/index.js +114 -65
- package/lib/runner/integration-tests.d.ts +37 -10
- package/lib/runner/integration-tests.js +34 -21
- package/lib/runner/private/cloud-assembly.d.ts +10 -0
- package/lib/runner/private/cloud-assembly.js +28 -2
- package/lib/runner/runner-base.d.ts +0 -1
- package/lib/runner/runner-base.js +7 -12
- package/lib/runner/snapshot-test-runner.d.ts +9 -0
- package/lib/runner/snapshot-test-runner.js +114 -96
- package/lib/utils.js +2 -1
- package/lib/workers/common.d.ts +8 -0
- package/lib/workers/common.js +1 -1
- package/lib/workers/extract/index.js +14296 -6110
- package/package.json +12 -12
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. `/*.
|
|
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
|
-
|
|
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:
|
|
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
|
-
|
|
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.
|
package/THIRD_PARTY_LICENSES
CHANGED
|
@@ -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.
|
|
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.
|
|
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.
|
|
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.
|
|
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
|
|
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
|
|
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
|
-
|
|
17
|
-
var _a
|
|
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
|
-
.
|
|
36
|
-
const
|
|
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
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
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:
|
|
72
|
-
verbose:
|
|
110
|
+
retain: options.inspectFailures,
|
|
111
|
+
verbose: options.verbose,
|
|
73
112
|
});
|
|
74
113
|
for (const failure of failedSnapshots) {
|
|
75
|
-
destructiveChanges.push(...(
|
|
114
|
+
destructiveChanges.push(...(_a = failure.destructiveChanges) !== null && _a !== void 0 ? _a : []);
|
|
76
115
|
}
|
|
77
|
-
if (!
|
|
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 ||
|
|
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:
|
|
93
|
-
dryRun:
|
|
94
|
-
verbosity:
|
|
95
|
-
updateWorkflow: !
|
|
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 (
|
|
137
|
+
if (options.clean === false) {
|
|
99
138
|
logger.warning('Not cleaning up stacks since "--no-clean" was used');
|
|
100
139
|
}
|
|
101
|
-
if (Boolean(
|
|
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:
|
|
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"]}
|