@aws-cdk-testing/cli-integ 2.61.0
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/LICENSE +201 -0
- package/NOTICE +16 -0
- package/README.md +151 -0
- package/bin/apply-patches +19 -0
- package/bin/download-and-run-old-tests +52 -0
- package/bin/query-github +2 -0
- package/bin/query-github.d.ts +1 -0
- package/bin/query-github.js +55 -0
- package/bin/run-suite +2 -0
- package/bin/run-suite.d.ts +1 -0
- package/bin/run-suite.js +126 -0
- package/bin/stage-distribution +2 -0
- package/bin/stage-distribution.d.ts +1 -0
- package/bin/stage-distribution.js +209 -0
- package/bin/test-root +2 -0
- package/bin/test-root.d.ts +1 -0
- package/bin/test-root.js +6 -0
- package/entrypoints/test-cli-regression-against-current-code.sh +11 -0
- package/entrypoints/test-cli-regression-against-latest-release.sh +11 -0
- package/entrypoints/test-cli-regression.bash +83 -0
- package/entrypoints/test.sh +12 -0
- package/lib/aws.d.ts +55 -0
- package/lib/aws.js +243 -0
- package/lib/corking.d.ts +13 -0
- package/lib/corking.js +34 -0
- package/lib/files.d.ts +15 -0
- package/lib/files.js +80 -0
- package/lib/github.d.ts +4 -0
- package/lib/github.js +43 -0
- package/lib/index.d.ts +12 -0
- package/lib/index.js +25 -0
- package/lib/integ-test.d.ts +11 -0
- package/lib/integ-test.js +55 -0
- package/lib/lists.d.ts +1 -0
- package/lib/lists.js +12 -0
- package/lib/memoize.d.ts +6 -0
- package/lib/memoize.js +19 -0
- package/lib/npm.d.ts +4 -0
- package/lib/npm.js +15 -0
- package/lib/package-sources/release-source.d.ts +22 -0
- package/lib/package-sources/release-source.js +67 -0
- package/lib/package-sources/repo-source.d.ts +23 -0
- package/lib/package-sources/repo-source.js +92 -0
- package/lib/package-sources/repo-tools/npm +2 -0
- package/lib/package-sources/repo-tools/npm.d.ts +1 -0
- package/lib/package-sources/repo-tools/npm.js +42 -0
- package/lib/package-sources/source.d.ts +24 -0
- package/lib/package-sources/source.js +3 -0
- package/lib/package-sources/subprocess.d.ts +3 -0
- package/lib/package-sources/subprocess.js +18 -0
- package/lib/resource-pool.d.ts +54 -0
- package/lib/resource-pool.js +120 -0
- package/lib/resources.d.ts +1 -0
- package/lib/resources.js +6 -0
- package/lib/shell.d.ts +59 -0
- package/lib/shell.js +118 -0
- package/lib/staging/codeartifact.d.ts +44 -0
- package/lib/staging/codeartifact.js +258 -0
- package/lib/staging/maven.d.ts +5 -0
- package/lib/staging/maven.js +83 -0
- package/lib/staging/npm.d.ts +4 -0
- package/lib/staging/npm.js +56 -0
- package/lib/staging/nuget.d.ts +4 -0
- package/lib/staging/nuget.js +71 -0
- package/lib/staging/parallel-shell.d.ts +6 -0
- package/lib/staging/parallel-shell.js +46 -0
- package/lib/staging/pypi.d.ts +4 -0
- package/lib/staging/pypi.js +49 -0
- package/lib/staging/usage-dir.d.ts +31 -0
- package/lib/staging/usage-dir.js +87 -0
- package/lib/with-aws.d.ts +13 -0
- package/lib/with-aws.js +60 -0
- package/lib/with-cdk-app.d.ts +146 -0
- package/lib/with-cdk-app.js +398 -0
- package/lib/with-packages.d.ts +5 -0
- package/lib/with-packages.js +14 -0
- package/lib/with-sam.d.ts +33 -0
- package/lib/with-sam.js +240 -0
- package/lib/with-temporary-directory.d.ts +5 -0
- package/lib/with-temporary-directory.js +32 -0
- package/lib/xpmutex.d.ts +43 -0
- package/lib/xpmutex.js +207 -0
- package/package.json +73 -0
- package/resources/cdk-apps/app/app.js +463 -0
- package/resources/cdk-apps/app/cdk.json +7 -0
- package/resources/cdk-apps/app/docker/Dockerfile +2 -0
- package/resources/cdk-apps/app/docker/Dockerfile.Custom +2 -0
- package/resources/cdk-apps/app/lambda/index.js +4 -0
- package/resources/cdk-apps/app/lambda/response.json +3 -0
- package/resources/cdk-apps/app/nested-stack.js +49 -0
- package/resources/cdk-apps/cfn-include-app/.gitignore +1 -0
- package/resources/cdk-apps/cfn-include-app/cdk.json +4 -0
- package/resources/cdk-apps/cfn-include-app/cfn-include-app.js +21 -0
- package/resources/cdk-apps/cfn-include-app/example-template.json +13 -0
- package/resources/cdk-apps/sam_cdk_integ_app/bin/test-app.js +11 -0
- package/resources/cdk-apps/sam_cdk_integ_app/cdk.json +6 -0
- package/resources/cdk-apps/sam_cdk_integ_app/lib/nested-stack.js +19 -0
- package/resources/cdk-apps/sam_cdk_integ_app/lib/test-stack.js +134 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/docker/DockerImageFunctionConstruct/.no-packagejson-validator +0 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/docker/DockerImageFunctionConstruct/Dockerfile +9 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/docker/DockerImageFunctionConstruct/app.js +22 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/docker/DockerImageFunctionConstruct/package.json +18 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/go/GoFunctionConstruct/go.mod +5 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/go/GoFunctionConstruct/go.sum +17 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/go/GoFunctionConstruct/main.go +17 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/nodejs/NodeJsFunctionConstruct/.no-packagejson-validator +0 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/nodejs/NodeJsFunctionConstruct/app.ts +16 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/nodejs/NodeJsFunctionConstruct/package-lock.json +12 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/nodejs/NodeJsFunctionConstruct/package.json +5 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/python/Function/app.py +15 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/python/Function/requirements.txt +1 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/python/Layer/layer_version_dependency.py +5 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/python/Layer/requirements.txt +1 -0
- package/resources/cdk-apps/sam_cdk_integ_app/src/rest-api-definition.yaml +12 -0
- package/resources/cli-regression-patches/v1.119.0/NOTES.md +5 -0
- package/resources/cli-regression-patches/v1.119.0/cli.integtest.js +659 -0
- package/resources/cli-regression-patches/v1.130.0/NOTES.md +12 -0
- package/resources/cli-regression-patches/v1.130.0/app/app.js +378 -0
- package/resources/cli-regression-patches/v1.130.0/bootstrapping.integtest.js +220 -0
- package/resources/cli-regression-patches/v1.44.0/NOTES.md +18 -0
- package/resources/cli-regression-patches/v1.44.0/bootstrapping.integtest.js +126 -0
- package/resources/cli-regression-patches/v1.44.0/test.sh +26 -0
- package/resources/cli-regression-patches/v1.61.1/NOTES.md +2 -0
- package/resources/cli-regression-patches/v1.61.1/skip-tests.txt +16 -0
- package/resources/cli-regression-patches/v1.62.0/NOTES.md +2 -0
- package/resources/cli-regression-patches/v1.62.0/aws-helpers.js +245 -0
- package/resources/cli-regression-patches/v1.63.0/NOTES.md +1 -0
- package/resources/cli-regression-patches/v1.63.0/skip-tests.txt +7 -0
- package/resources/cli-regression-patches/v1.64.0/NOTES.md +3 -0
- package/resources/cli-regression-patches/v1.64.0/cdk-helpers.js +325 -0
- package/resources/cli-regression-patches/v1.64.0/cli.integtest.js +599 -0
- package/resources/cli-regression-patches/v1.64.1/NOTES.md +3 -0
- package/resources/cli-regression-patches/v1.64.1/cdk-helpers.js +324 -0
- package/resources/cli-regression-patches/v1.64.1/cli.integtest.js +599 -0
- package/resources/cli-regression-patches/v1.67.0/NOTES.md +2 -0
- package/resources/cli-regression-patches/v1.67.0/cdk-helpers.js +331 -0
- package/resources/cloud-assemblies/0.36.0/InitStack.template.json +1 -0
- package/resources/cloud-assemblies/0.36.0/manifest.json +19 -0
- package/resources/cloud-assemblies/1.10.0-lookup-default-vpc/InitStack.template.json +2 -0
- package/resources/cloud-assemblies/1.10.0-lookup-default-vpc/manifest.json.js +37 -0
- package/resources/cloud-assemblies/1.10.0-request-azs/InitStack.template.json +2 -0
- package/resources/cloud-assemblies/1.10.0-request-azs/manifest.json.js +34 -0
- package/resources/integ.jest.config.js +25 -0
- package/skip-tests.txt +8 -0
- package/tests/cli-integ-tests/README.md +47 -0
- package/tests/cli-integ-tests/bootstrapping.integtest.d.ts +1 -0
- package/tests/cli-integ-tests/bootstrapping.integtest.js +271 -0
- package/tests/cli-integ-tests/cli.integtest.d.ts +1 -0
- package/tests/cli-integ-tests/cli.integtest.js +1029 -0
- package/tests/init-csharp/init-csharp.integtest.d.ts +1 -0
- package/tests/init-csharp/init-csharp.integtest.js +14 -0
- package/tests/init-fsharp/init-fsharp.integtest.d.ts +1 -0
- package/tests/init-fsharp/init-fsharp.integtest.js +14 -0
- package/tests/init-java/init-java.integtest.d.ts +1 -0
- package/tests/init-java/init-java.integtest.js +14 -0
- package/tests/init-javascript/init-javascript.integtest.d.ts +1 -0
- package/tests/init-javascript/init-javascript.integtest.js +15 -0
- package/tests/init-python/init-python.integtest.d.ts +1 -0
- package/tests/init-python/init-python.integtest.js +19 -0
- package/tests/init-typescript-app/init-typescript-app.integtest.d.ts +1 -0
- package/tests/init-typescript-app/init-typescript-app.integtest.js +49 -0
- package/tests/init-typescript-lib/init-typescript-lib.integtest.d.ts +1 -0
- package/tests/init-typescript-lib/init-typescript-lib.integtest.js +13 -0
- package/tests/uberpackage/uberpackage.integtest.d.ts +1 -0
- package/tests/uberpackage/uberpackage.integtest.js +11 -0
|
@@ -0,0 +1,331 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a, _b;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.randomString = exports.rimraf = exports.shell = exports.TestFixture = exports.cloneDirectory = exports.withDefaultFixture = exports.withCdkApp = exports.withAws = void 0;
|
|
5
|
+
const child_process = require("child_process");
|
|
6
|
+
const fs = require("fs");
|
|
7
|
+
const os = require("os");
|
|
8
|
+
const path = require("path");
|
|
9
|
+
const aws_helpers_1 = require("./aws-helpers");
|
|
10
|
+
const resource_pool_1 = require("./resource-pool");
|
|
11
|
+
const REGIONS = process.env.AWS_REGIONS
|
|
12
|
+
? process.env.AWS_REGIONS.split(',')
|
|
13
|
+
: [(_b = (_a = process.env.AWS_REGION) !== null && _a !== void 0 ? _a : process.env.AWS_DEFAULT_REGION) !== null && _b !== void 0 ? _b : 'us-east-1'];
|
|
14
|
+
const FRAMEWORK_VERSION = process.env.FRAMEWORK_VERSION;
|
|
15
|
+
process.stdout.write(`Using regions: ${REGIONS}\n`);
|
|
16
|
+
process.stdout.write(`Using framework version: ${FRAMEWORK_VERSION}\n`);
|
|
17
|
+
const REGION_POOL = new resource_pool_1.ResourcePool(REGIONS);
|
|
18
|
+
/**
|
|
19
|
+
* Higher order function to execute a block with an AWS client setup
|
|
20
|
+
*
|
|
21
|
+
* Allocate the next region from the REGION pool and dispose it afterwards.
|
|
22
|
+
*/
|
|
23
|
+
function withAws(block) {
|
|
24
|
+
return (context) => REGION_POOL.using(async (region) => {
|
|
25
|
+
const aws = await aws_helpers_1.AwsClients.forRegion(region, context.output);
|
|
26
|
+
await sanityCheck(aws);
|
|
27
|
+
return block({ ...context, aws });
|
|
28
|
+
});
|
|
29
|
+
}
|
|
30
|
+
exports.withAws = withAws;
|
|
31
|
+
/**
|
|
32
|
+
* Higher order function to execute a block with a CDK app fixture
|
|
33
|
+
*
|
|
34
|
+
* Requires an AWS client to be passed in.
|
|
35
|
+
*
|
|
36
|
+
* For backwards compatibility with existing tests (so we don't have to change
|
|
37
|
+
* too much) the inner block is expected to take a `TestFixture` object.
|
|
38
|
+
*/
|
|
39
|
+
function withCdkApp(block) {
|
|
40
|
+
return async (context) => {
|
|
41
|
+
const randy = randomString();
|
|
42
|
+
const stackNamePrefix = `cdktest-${randy}`;
|
|
43
|
+
const integTestDir = path.join(os.tmpdir(), `cdk-integ-${randy}`);
|
|
44
|
+
context.output.write(` Stack prefix: ${stackNamePrefix}\n`);
|
|
45
|
+
context.output.write(` Test directory: ${integTestDir}\n`);
|
|
46
|
+
context.output.write(` Region: ${context.aws.region}\n`);
|
|
47
|
+
await cloneDirectory(path.join(__dirname, 'app'), integTestDir, context.output);
|
|
48
|
+
const fixture = new TestFixture(integTestDir, stackNamePrefix, context.output, context.aws);
|
|
49
|
+
let success = true;
|
|
50
|
+
try {
|
|
51
|
+
let modules = [
|
|
52
|
+
'@aws-cdk/core',
|
|
53
|
+
'@aws-cdk/aws-sns',
|
|
54
|
+
'@aws-cdk/aws-iam',
|
|
55
|
+
'@aws-cdk/aws-lambda',
|
|
56
|
+
'@aws-cdk/aws-ssm',
|
|
57
|
+
'@aws-cdk/aws-ecr-assets',
|
|
58
|
+
'@aws-cdk/aws-cloudformation',
|
|
59
|
+
'@aws-cdk/aws-ec2',
|
|
60
|
+
];
|
|
61
|
+
if (FRAMEWORK_VERSION) {
|
|
62
|
+
modules = modules.map(module => `${module}@${FRAMEWORK_VERSION}`);
|
|
63
|
+
}
|
|
64
|
+
await fixture.shell(['npm', 'install', ...modules]);
|
|
65
|
+
await ensureBootstrapped(fixture);
|
|
66
|
+
await block(fixture);
|
|
67
|
+
}
|
|
68
|
+
catch (e) {
|
|
69
|
+
success = false;
|
|
70
|
+
throw e;
|
|
71
|
+
}
|
|
72
|
+
finally {
|
|
73
|
+
await fixture.dispose(success);
|
|
74
|
+
}
|
|
75
|
+
};
|
|
76
|
+
}
|
|
77
|
+
exports.withCdkApp = withCdkApp;
|
|
78
|
+
/**
|
|
79
|
+
* Default test fixture for most (all?) integ tests
|
|
80
|
+
*
|
|
81
|
+
* It's a composition of withAws/withCdkApp, expecting the test block to take a `TestFixture`
|
|
82
|
+
* object.
|
|
83
|
+
*
|
|
84
|
+
* We could have put `withAws(withCdkApp(fixture => { /... actual test here.../ }))` in every
|
|
85
|
+
* test declaration but centralizing it is going to make it convenient to modify in the future.
|
|
86
|
+
*/
|
|
87
|
+
function withDefaultFixture(block) {
|
|
88
|
+
return withAws(withCdkApp(block));
|
|
89
|
+
// ^~~~~~ this is disappointing TypeScript! Feels like you should have been able to derive this.
|
|
90
|
+
}
|
|
91
|
+
exports.withDefaultFixture = withDefaultFixture;
|
|
92
|
+
/**
|
|
93
|
+
* Prepare a target dir byreplicating a source directory
|
|
94
|
+
*/
|
|
95
|
+
async function cloneDirectory(source, target, output) {
|
|
96
|
+
await shell(['rm', '-rf', target], { output });
|
|
97
|
+
await shell(['mkdir', '-p', target], { output });
|
|
98
|
+
await shell(['cp', '-R', source + '/*', target], { output });
|
|
99
|
+
}
|
|
100
|
+
exports.cloneDirectory = cloneDirectory;
|
|
101
|
+
class TestFixture {
|
|
102
|
+
constructor(integTestDir, stackNamePrefix, output, aws) {
|
|
103
|
+
this.integTestDir = integTestDir;
|
|
104
|
+
this.stackNamePrefix = stackNamePrefix;
|
|
105
|
+
this.output = output;
|
|
106
|
+
this.aws = aws;
|
|
107
|
+
this.qualifier = randomString().slice(0, 10);
|
|
108
|
+
this.bucketsToDelete = new Array();
|
|
109
|
+
}
|
|
110
|
+
log(s) {
|
|
111
|
+
this.output.write(`${s}\n`);
|
|
112
|
+
}
|
|
113
|
+
async shell(command, options = {}) {
|
|
114
|
+
return shell(command, {
|
|
115
|
+
output: this.output,
|
|
116
|
+
cwd: this.integTestDir,
|
|
117
|
+
...options,
|
|
118
|
+
});
|
|
119
|
+
}
|
|
120
|
+
async cdkDeploy(stackNames, options = {}) {
|
|
121
|
+
var _a, _b;
|
|
122
|
+
stackNames = typeof stackNames === 'string' ? [stackNames] : stackNames;
|
|
123
|
+
const neverRequireApproval = (_a = options.neverRequireApproval) !== null && _a !== void 0 ? _a : true;
|
|
124
|
+
return this.cdk(['deploy',
|
|
125
|
+
...(neverRequireApproval ? ['--require-approval=never'] : []), // Default to no approval in an unattended test
|
|
126
|
+
...((_b = options.options) !== null && _b !== void 0 ? _b : []), ...this.fullStackName(stackNames)], options);
|
|
127
|
+
}
|
|
128
|
+
async cdkDestroy(stackNames, options = {}) {
|
|
129
|
+
var _a;
|
|
130
|
+
stackNames = typeof stackNames === 'string' ? [stackNames] : stackNames;
|
|
131
|
+
return this.cdk(['destroy',
|
|
132
|
+
'-f', // We never want a prompt in an unattended test
|
|
133
|
+
...((_a = options.options) !== null && _a !== void 0 ? _a : []), ...this.fullStackName(stackNames)], options);
|
|
134
|
+
}
|
|
135
|
+
async cdk(args, options = {}) {
|
|
136
|
+
var _a;
|
|
137
|
+
const verbose = (_a = options.verbose) !== null && _a !== void 0 ? _a : true;
|
|
138
|
+
return this.shell(['cdk', ...(verbose ? ['-v'] : []), ...args], {
|
|
139
|
+
...options,
|
|
140
|
+
modEnv: {
|
|
141
|
+
AWS_REGION: this.aws.region,
|
|
142
|
+
AWS_DEFAULT_REGION: this.aws.region,
|
|
143
|
+
STACK_NAME_PREFIX: this.stackNamePrefix,
|
|
144
|
+
...options.modEnv,
|
|
145
|
+
},
|
|
146
|
+
});
|
|
147
|
+
}
|
|
148
|
+
fullStackName(stackNames) {
|
|
149
|
+
if (typeof stackNames === 'string') {
|
|
150
|
+
return `${this.stackNamePrefix}-${stackNames}`;
|
|
151
|
+
}
|
|
152
|
+
else {
|
|
153
|
+
return stackNames.map(s => `${this.stackNamePrefix}-${s}`);
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
/**
|
|
157
|
+
* Append this to the list of buckets to potentially delete
|
|
158
|
+
*
|
|
159
|
+
* At the end of a test, we clean up buckets that may not have gotten destroyed
|
|
160
|
+
* (for whatever reason).
|
|
161
|
+
*/
|
|
162
|
+
rememberToDeleteBucket(bucketName) {
|
|
163
|
+
this.bucketsToDelete.push(bucketName);
|
|
164
|
+
}
|
|
165
|
+
/**
|
|
166
|
+
* Cleanup leftover stacks and buckets
|
|
167
|
+
*/
|
|
168
|
+
async dispose(success) {
|
|
169
|
+
const stacksToDelete = await this.deleteableStacks(this.stackNamePrefix);
|
|
170
|
+
// Bootstrap stacks have buckets that need to be cleaned
|
|
171
|
+
const bucketNames = stacksToDelete.map(stack => aws_helpers_1.outputFromStack('BucketName', stack)).filter(defined);
|
|
172
|
+
await Promise.all(bucketNames.map(b => this.aws.emptyBucket(b)));
|
|
173
|
+
// Bootstrap stacks have ECR repositories with images which should be deleted
|
|
174
|
+
const imageRepositoryNames = stacksToDelete.map(stack => aws_helpers_1.outputFromStack('ImageRepositoryName', stack)).filter(defined);
|
|
175
|
+
await Promise.all(imageRepositoryNames.map(r => this.aws.deleteImageRepository(r)));
|
|
176
|
+
await this.aws.deleteStacks(...stacksToDelete.map(s => s.StackName));
|
|
177
|
+
// We might have leaked some buckets by upgrading the bootstrap stack. Be
|
|
178
|
+
// sure to clean everything.
|
|
179
|
+
for (const bucket of this.bucketsToDelete) {
|
|
180
|
+
await this.aws.deleteBucket(bucket);
|
|
181
|
+
}
|
|
182
|
+
// If the tests completed successfully, happily delete the fixture
|
|
183
|
+
// (otherwise leave it for humans to inspect)
|
|
184
|
+
if (success) {
|
|
185
|
+
rimraf(this.integTestDir);
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
/**
|
|
189
|
+
* Return the stacks starting with our testing prefix that should be deleted
|
|
190
|
+
*/
|
|
191
|
+
async deleteableStacks(prefix) {
|
|
192
|
+
var _a;
|
|
193
|
+
const statusFilter = [
|
|
194
|
+
'CREATE_IN_PROGRESS', 'CREATE_FAILED', 'CREATE_COMPLETE',
|
|
195
|
+
'ROLLBACK_IN_PROGRESS', 'ROLLBACK_FAILED', 'ROLLBACK_COMPLETE',
|
|
196
|
+
'DELETE_FAILED',
|
|
197
|
+
'UPDATE_IN_PROGRESS', 'UPDATE_COMPLETE_CLEANUP_IN_PROGRESS',
|
|
198
|
+
'UPDATE_COMPLETE', 'UPDATE_ROLLBACK_IN_PROGRESS',
|
|
199
|
+
'UPDATE_ROLLBACK_FAILED',
|
|
200
|
+
'UPDATE_ROLLBACK_COMPLETE_CLEANUP_IN_PROGRESS',
|
|
201
|
+
'UPDATE_ROLLBACK_COMPLETE', 'REVIEW_IN_PROGRESS',
|
|
202
|
+
'IMPORT_IN_PROGRESS', 'IMPORT_COMPLETE',
|
|
203
|
+
'IMPORT_ROLLBACK_IN_PROGRESS', 'IMPORT_ROLLBACK_FAILED',
|
|
204
|
+
'IMPORT_ROLLBACK_COMPLETE',
|
|
205
|
+
];
|
|
206
|
+
const response = await this.aws.cloudFormation('describeStacks', {});
|
|
207
|
+
return ((_a = response.Stacks) !== null && _a !== void 0 ? _a : [])
|
|
208
|
+
.filter(s => s.StackName.startsWith(prefix))
|
|
209
|
+
.filter(s => statusFilter.includes(s.StackStatus))
|
|
210
|
+
.filter(s => s.RootId === undefined); // Only delete parent stacks. Nested stacks are deleted in the process
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
exports.TestFixture = TestFixture;
|
|
214
|
+
/**
|
|
215
|
+
* Perform a one-time quick sanity check that the AWS clients has properly configured credentials
|
|
216
|
+
*
|
|
217
|
+
* If we don't do this, calls are going to fail and they'll be retried and everything will take
|
|
218
|
+
* forever before the user notices a simple misconfiguration.
|
|
219
|
+
*
|
|
220
|
+
* We can't check for the presence of environment variables since credentials could come from
|
|
221
|
+
* anywhere, so do simple account retrieval.
|
|
222
|
+
*
|
|
223
|
+
* Only do it once per process.
|
|
224
|
+
*/
|
|
225
|
+
async function sanityCheck(aws) {
|
|
226
|
+
if (sanityChecked === undefined) {
|
|
227
|
+
try {
|
|
228
|
+
await aws.account();
|
|
229
|
+
sanityChecked = true;
|
|
230
|
+
}
|
|
231
|
+
catch (e) {
|
|
232
|
+
sanityChecked = false;
|
|
233
|
+
throw new Error(`AWS credentials probably not configured, got error: ${e.message}`);
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
if (!sanityChecked) {
|
|
237
|
+
throw new Error('AWS credentials probably not configured, see previous error');
|
|
238
|
+
}
|
|
239
|
+
}
|
|
240
|
+
let sanityChecked;
|
|
241
|
+
/**
|
|
242
|
+
* Make sure that the given environment is bootstrapped
|
|
243
|
+
*
|
|
244
|
+
* Since we go striping across regions, it's going to suck doing this
|
|
245
|
+
* by hand so let's just mass-automate it.
|
|
246
|
+
*/
|
|
247
|
+
async function ensureBootstrapped(fixture) {
|
|
248
|
+
// Old-style bootstrap stack with default name
|
|
249
|
+
if (await fixture.aws.stackStatus('CDKToolkit') === undefined) {
|
|
250
|
+
await fixture.cdk(['bootstrap', `aws://${await fixture.aws.account()}/${fixture.aws.region}`]);
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
/**
|
|
254
|
+
* A shell command that does what you want
|
|
255
|
+
*
|
|
256
|
+
* Is platform-aware, handles errors nicely.
|
|
257
|
+
*/
|
|
258
|
+
async function shell(command, options = {}) {
|
|
259
|
+
var _a, _b;
|
|
260
|
+
if (options.modEnv && options.env) {
|
|
261
|
+
throw new Error('Use either env or modEnv but not both');
|
|
262
|
+
}
|
|
263
|
+
(_a = options.output) === null || _a === void 0 ? void 0 : _a.write(`💻 ${command.join(' ')}\n`);
|
|
264
|
+
const env = (_b = options.env) !== null && _b !== void 0 ? _b : (options.modEnv ? { ...process.env, ...options.modEnv } : undefined);
|
|
265
|
+
const child = child_process.spawn(command[0], command.slice(1), {
|
|
266
|
+
...options,
|
|
267
|
+
env,
|
|
268
|
+
// Need this for Windows where we want .cmd and .bat to be found as well.
|
|
269
|
+
shell: true,
|
|
270
|
+
stdio: ['ignore', 'pipe', 'pipe'],
|
|
271
|
+
});
|
|
272
|
+
return new Promise((resolve, reject) => {
|
|
273
|
+
const stdout = new Array();
|
|
274
|
+
const stderr = new Array();
|
|
275
|
+
child.stdout.on('data', chunk => {
|
|
276
|
+
var _a;
|
|
277
|
+
(_a = options.output) === null || _a === void 0 ? void 0 : _a.write(chunk);
|
|
278
|
+
stdout.push(chunk);
|
|
279
|
+
});
|
|
280
|
+
child.stderr.on('data', chunk => {
|
|
281
|
+
var _a, _b;
|
|
282
|
+
(_a = options.output) === null || _a === void 0 ? void 0 : _a.write(chunk);
|
|
283
|
+
if ((_b = options.captureStderr) !== null && _b !== void 0 ? _b : true) {
|
|
284
|
+
stderr.push(chunk);
|
|
285
|
+
}
|
|
286
|
+
});
|
|
287
|
+
child.once('error', reject);
|
|
288
|
+
child.once('close', code => {
|
|
289
|
+
if (code === 0 || options.allowErrExit) {
|
|
290
|
+
resolve((Buffer.concat(stdout).toString('utf-8') + Buffer.concat(stderr).toString('utf-8')).trim());
|
|
291
|
+
}
|
|
292
|
+
else {
|
|
293
|
+
reject(new Error(`'${command.join(' ')}' exited with error code ${code}`));
|
|
294
|
+
}
|
|
295
|
+
});
|
|
296
|
+
});
|
|
297
|
+
}
|
|
298
|
+
exports.shell = shell;
|
|
299
|
+
function defined(x) {
|
|
300
|
+
return x !== undefined;
|
|
301
|
+
}
|
|
302
|
+
/**
|
|
303
|
+
* rm -rf reimplementation, don't want to depend on an NPM package for this
|
|
304
|
+
*/
|
|
305
|
+
function rimraf(fsPath) {
|
|
306
|
+
try {
|
|
307
|
+
const isDir = fs.lstatSync(fsPath).isDirectory();
|
|
308
|
+
if (isDir) {
|
|
309
|
+
for (const file of fs.readdirSync(fsPath)) {
|
|
310
|
+
rimraf(path.join(fsPath, file));
|
|
311
|
+
}
|
|
312
|
+
fs.rmdirSync(fsPath);
|
|
313
|
+
}
|
|
314
|
+
else {
|
|
315
|
+
fs.unlinkSync(fsPath);
|
|
316
|
+
}
|
|
317
|
+
}
|
|
318
|
+
catch (e) {
|
|
319
|
+
// We will survive ENOENT
|
|
320
|
+
if (e.code !== 'ENOENT') {
|
|
321
|
+
throw e;
|
|
322
|
+
}
|
|
323
|
+
}
|
|
324
|
+
}
|
|
325
|
+
exports.rimraf = rimraf;
|
|
326
|
+
function randomString() {
|
|
327
|
+
// Crazy
|
|
328
|
+
return Math.random().toString(36).replace(/[^a-z0-9]+/g, '');
|
|
329
|
+
}
|
|
330
|
+
exports.randomString = randomString;
|
|
331
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cdk-helpers.js","sourceRoot":"","sources":["cdk-helpers.ts"],"names":[],"mappings":"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","sourcesContent":["import * as child_process from 'child_process';\nimport * as fs from 'fs';\nimport * as os from 'os';\nimport * as path from 'path';\nimport { outputFromStack, AwsClients } from './aws-helpers';\nimport { ResourcePool } from './resource-pool';\nimport { TestContext } from './test-helpers';\n\nconst REGIONS = process.env.AWS_REGIONS\n  ? process.env.AWS_REGIONS.split(',')\n  : [process.env.AWS_REGION ?? process.env.AWS_DEFAULT_REGION ?? 'us-east-1'];\n\nconst FRAMEWORK_VERSION = process.env.FRAMEWORK_VERSION;\n\nprocess.stdout.write(`Using regions: ${REGIONS}\\n`);\nprocess.stdout.write(`Using framework version: ${FRAMEWORK_VERSION}\\n`);\n\nconst REGION_POOL = new ResourcePool(REGIONS);\n\n\nexport type AwsContext = { readonly aws: AwsClients };\n\n/**\n * Higher order function to execute a block with an AWS client setup\n *\n * Allocate the next region from the REGION pool and dispose it afterwards.\n */\nexport function withAws<A extends TestContext>(block: (context: A & AwsContext) => Promise<void>) {\n  return (context: A) => REGION_POOL.using(async (region) => {\n    const aws = await AwsClients.forRegion(region, context.output);\n    await sanityCheck(aws);\n\n    return block({ ...context, aws });\n  });\n}\n\n/**\n * Higher order function to execute a block with a CDK app fixture\n *\n * Requires an AWS client to be passed in.\n *\n * For backwards compatibility with existing tests (so we don't have to change\n * too much) the inner block is expected to take a `TestFixture` object.\n */\nexport function withCdkApp<A extends TestContext & AwsContext>(block: (context: TestFixture) => Promise<void>) {\n  return async (context: A) => {\n    const randy = randomString();\n    const stackNamePrefix = `cdktest-${randy}`;\n    const integTestDir = path.join(os.tmpdir(), `cdk-integ-${randy}`);\n\n    context.output.write(` Stack prefix:   ${stackNamePrefix}\\n`);\n    context.output.write(` Test directory: ${integTestDir}\\n`);\n    context.output.write(` Region:         ${context.aws.region}\\n`);\n\n    await cloneDirectory(path.join(__dirname, 'app'), integTestDir, context.output);\n    const fixture = new TestFixture(\n      integTestDir,\n      stackNamePrefix,\n      context.output,\n      context.aws);\n\n    let success = true;\n    try {\n      let modules = [\n        '@aws-cdk/core',\n        '@aws-cdk/aws-sns',\n        '@aws-cdk/aws-iam',\n        '@aws-cdk/aws-lambda',\n        '@aws-cdk/aws-ssm',\n        '@aws-cdk/aws-ecr-assets',\n        '@aws-cdk/aws-cloudformation',\n        '@aws-cdk/aws-ec2',\n      ];\n      if (FRAMEWORK_VERSION) {\n        modules = modules.map(module => `${module}@${FRAMEWORK_VERSION}`);\n      }\n      await fixture.shell(['npm', 'install', ...modules]);\n\n      await ensureBootstrapped(fixture);\n\n      await block(fixture);\n    } catch (e) {\n      success = false;\n      throw e;\n    } finally {\n      await fixture.dispose(success);\n    }\n  };\n}\n\n/**\n * Default test fixture for most (all?) integ tests\n *\n * It's a composition of withAws/withCdkApp, expecting the test block to take a `TestFixture`\n * object.\n *\n * We could have put `withAws(withCdkApp(fixture => { /... actual test here.../ }))` in every\n * test declaration but centralizing it is going to make it convenient to modify in the future.\n */\nexport function withDefaultFixture(block: (context: TestFixture) => Promise<void>) {\n  return withAws<TestContext>(withCdkApp(block));\n  //              ^~~~~~ this is disappointing TypeScript! Feels like you should have been able to derive this.\n}\n\nexport interface ShellOptions extends child_process.SpawnOptions {\n  /**\n   * Properties to add to 'env'\n   */\n  modEnv?: Record<string, string>;\n\n  /**\n   * Don't fail when exiting with an error\n   *\n   * @default false\n   */\n  allowErrExit?: boolean;\n\n  /**\n   * Whether to capture stderr\n   *\n   * @default true\n   */\n  captureStderr?: boolean;\n\n  /**\n   * Pass output here\n   */\n  output?: NodeJS.WritableStream;\n}\n\nexport interface CdkCliOptions extends ShellOptions {\n  options?: string[];\n  neverRequireApproval?: boolean;\n  verbose?: boolean;\n}\n\n/**\n * Prepare a target dir byreplicating a source directory\n */\nexport async function cloneDirectory(source: string, target: string, output?: NodeJS.WritableStream) {\n  await shell(['rm', '-rf', target], { output });\n  await shell(['mkdir', '-p', target], { output });\n  await shell(['cp', '-R', source + '/*', target], { output });\n}\n\nexport class TestFixture {\n  public readonly qualifier = randomString().substr(0, 10);\n  private readonly bucketsToDelete = new Array<string>();\n\n  constructor(\n    public readonly integTestDir: string,\n    public readonly stackNamePrefix: string,\n    public readonly output: NodeJS.WritableStream,\n    public readonly aws: AwsClients) {\n  }\n\n  public log(s: string) {\n    this.output.write(`${s}\\n`);\n  }\n\n  public async shell(command: string[], options: Omit<ShellOptions, 'cwd'|'output'> = {}): Promise<string> {\n    return shell(command, {\n      output: this.output,\n      cwd: this.integTestDir,\n      ...options,\n    });\n  }\n\n  public async cdkDeploy(stackNames: string | string[], options: CdkCliOptions = {}) {\n    stackNames = typeof stackNames === 'string' ? [stackNames] : stackNames;\n\n    const neverRequireApproval = options.neverRequireApproval ?? true;\n\n    return this.cdk(['deploy',\n      ...(neverRequireApproval ? ['--require-approval=never'] : []), // Default to no approval in an unattended test\n      ...(options.options ?? []),\n      ...this.fullStackName(stackNames)], options);\n  }\n\n  public async cdkDestroy(stackNames: string | string[], options: CdkCliOptions = {}) {\n    stackNames = typeof stackNames === 'string' ? [stackNames] : stackNames;\n\n    return this.cdk(['destroy',\n      '-f', // We never want a prompt in an unattended test\n      ...(options.options ?? []),\n      ...this.fullStackName(stackNames)], options);\n  }\n\n  public async cdk(args: string[], options: CdkCliOptions = {}) {\n    const verbose = options.verbose ?? true;\n\n    return this.shell(['cdk', ...(verbose ? ['-v'] : []), ...args], {\n      ...options,\n      modEnv: {\n        AWS_REGION: this.aws.region,\n        AWS_DEFAULT_REGION: this.aws.region,\n        STACK_NAME_PREFIX: this.stackNamePrefix,\n        ...options.modEnv,\n      },\n    });\n  }\n\n  public fullStackName(stackName: string): string;\n  public fullStackName(stackNames: string[]): string[];\n  public fullStackName(stackNames: string | string[]): string | string[] {\n    if (typeof stackNames === 'string') {\n      return `${this.stackNamePrefix}-${stackNames}`;\n    } else {\n      return stackNames.map(s => `${this.stackNamePrefix}-${s}`);\n    }\n  }\n\n  /**\n   * Append this to the list of buckets to potentially delete\n   *\n   * At the end of a test, we clean up buckets that may not have gotten destroyed\n   * (for whatever reason).\n   */\n  public rememberToDeleteBucket(bucketName: string) {\n    this.bucketsToDelete.push(bucketName);\n  }\n\n  /**\n   * Cleanup leftover stacks and buckets\n   */\n  public async dispose(success: boolean) {\n    const stacksToDelete = await this.deleteableStacks(this.stackNamePrefix);\n\n    // Bootstrap stacks have buckets that need to be cleaned\n    const bucketNames = stacksToDelete.map(stack => outputFromStack('BucketName', stack)).filter(defined);\n    await Promise.all(bucketNames.map(b => this.aws.emptyBucket(b)));\n\n    // Bootstrap stacks have ECR repositories with images which should be deleted\n    const imageRepositoryNames = stacksToDelete.map(stack => outputFromStack('ImageRepositoryName', stack)).filter(defined);\n    await Promise.all(imageRepositoryNames.map(r => this.aws.deleteImageRepository(r)));\n\n    await this.aws.deleteStacks(...stacksToDelete.map(s => s.StackName));\n\n    // We might have leaked some buckets by upgrading the bootstrap stack. Be\n    // sure to clean everything.\n    for (const bucket of this.bucketsToDelete) {\n      await this.aws.deleteBucket(bucket);\n    }\n\n    // If the tests completed successfully, happily delete the fixture\n    // (otherwise leave it for humans to inspect)\n    if (success) {\n      rimraf(this.integTestDir);\n    }\n  }\n\n  /**\n   * Return the stacks starting with our testing prefix that should be deleted\n   */\n  private async deleteableStacks(prefix: string): Promise<AWS.CloudFormation.Stack[]> {\n    const statusFilter = [\n      'CREATE_IN_PROGRESS', 'CREATE_FAILED', 'CREATE_COMPLETE',\n      'ROLLBACK_IN_PROGRESS', 'ROLLBACK_FAILED', 'ROLLBACK_COMPLETE',\n      'DELETE_FAILED',\n      'UPDATE_IN_PROGRESS', 'UPDATE_COMPLETE_CLEANUP_IN_PROGRESS',\n      'UPDATE_COMPLETE', 'UPDATE_ROLLBACK_IN_PROGRESS',\n      'UPDATE_ROLLBACK_FAILED',\n      'UPDATE_ROLLBACK_COMPLETE_CLEANUP_IN_PROGRESS',\n      'UPDATE_ROLLBACK_COMPLETE', 'REVIEW_IN_PROGRESS',\n      'IMPORT_IN_PROGRESS', 'IMPORT_COMPLETE',\n      'IMPORT_ROLLBACK_IN_PROGRESS', 'IMPORT_ROLLBACK_FAILED',\n      'IMPORT_ROLLBACK_COMPLETE',\n    ];\n\n    const response = await this.aws.cloudFormation('describeStacks', {});\n\n    return (response.Stacks ?? [])\n      .filter(s => s.StackName.startsWith(prefix))\n      .filter(s => statusFilter.includes(s.StackStatus))\n      .filter(s => s.RootId === undefined); // Only delete parent stacks. Nested stacks are deleted in the process\n  }\n}\n\n/**\n * Perform a one-time quick sanity check that the AWS clients has properly configured credentials\n *\n * If we don't do this, calls are going to fail and they'll be retried and everything will take\n * forever before the user notices a simple misconfiguration.\n *\n * We can't check for the presence of environment variables since credentials could come from\n * anywhere, so do simple account retrieval.\n *\n * Only do it once per process.\n */\nasync function sanityCheck(aws: AwsClients) {\n  if (sanityChecked === undefined) {\n    try {\n      await aws.account();\n      sanityChecked = true;\n    } catch (e) {\n      sanityChecked = false;\n      throw new Error(`AWS credentials probably not configured, got error: ${e.message}`);\n    }\n  }\n  if (!sanityChecked) {\n    throw new Error('AWS credentials probably not configured, see previous error');\n  }\n}\nlet sanityChecked: boolean | undefined;\n\n/**\n * Make sure that the given environment is bootstrapped\n *\n * Since we go striping across regions, it's going to suck doing this\n * by hand so let's just mass-automate it.\n */\nasync function ensureBootstrapped(fixture: TestFixture) {\n  // Old-style bootstrap stack with default name\n  if (await fixture.aws.stackStatus('CDKToolkit') === undefined) {\n    await fixture.cdk(['bootstrap', `aws://${await fixture.aws.account()}/${fixture.aws.region}`]);\n  }\n}\n\n/**\n * A shell command that does what you want\n *\n * Is platform-aware, handles errors nicely.\n */\nexport async function shell(command: string[], options: ShellOptions = {}): Promise<string> {\n  if (options.modEnv && options.env) {\n    throw new Error('Use either env or modEnv but not both');\n  }\n\n  options.output?.write(`💻 ${command.join(' ')}\\n`);\n\n  const env = options.env ?? (options.modEnv ? { ...process.env, ...options.modEnv } : undefined);\n\n  const child = child_process.spawn(command[0], command.slice(1), {\n    ...options,\n    env,\n    // Need this for Windows where we want .cmd and .bat to be found as well.\n    shell: true,\n    stdio: ['ignore', 'pipe', 'pipe'],\n  });\n\n  return new Promise<string>((resolve, reject) => {\n    const stdout = new Array<Buffer>();\n    const stderr = new Array<Buffer>();\n\n    child.stdout!.on('data', chunk => {\n      options.output?.write(chunk);\n      stdout.push(chunk);\n    });\n\n    child.stderr!.on('data', chunk => {\n      options.output?.write(chunk);\n      if (options.captureStderr ?? true) {\n        stderr.push(chunk);\n      }\n    });\n\n    child.once('error', reject);\n\n    child.once('close', code => {\n      if (code === 0 || options.allowErrExit) {\n        resolve((Buffer.concat(stdout).toString('utf-8') + Buffer.concat(stderr).toString('utf-8')).trim());\n      } else {\n        reject(new Error(`'${command.join(' ')}' exited with error code ${code}`));\n      }\n    });\n  });\n}\n\nfunction defined<A>(x: A): x is NonNullable<A> {\n  return x !== undefined;\n}\n\n/**\n * rm -rf reimplementation, don't want to depend on an NPM package for this\n */\nexport function rimraf(fsPath: string) {\n  try {\n    const isDir = fs.lstatSync(fsPath).isDirectory();\n\n    if (isDir) {\n      for (const file of fs.readdirSync(fsPath)) {\n        rimraf(path.join(fsPath, file));\n      }\n      fs.rmdirSync(fsPath);\n    } else {\n      fs.unlinkSync(fsPath);\n    }\n  } catch (e) {\n    // We will survive ENOENT\n    if (e.code !== 'ENOENT') { throw e; }\n  }\n}\n\nexport function randomString() {\n  // Crazy\n  return Math.random().toString(36).replace(/[^a-z0-9]+/g, '');\n}"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": "0.36.0",
|
|
3
|
+
"artifacts": {
|
|
4
|
+
"InitStack": {
|
|
5
|
+
"type": "aws:cloudformation:stack",
|
|
6
|
+
"environment": "aws://unknown-account/unknown-region",
|
|
7
|
+
"properties": {
|
|
8
|
+
"templateFile": "InitStack.template.json"
|
|
9
|
+
}
|
|
10
|
+
}
|
|
11
|
+
},
|
|
12
|
+
"runtime": {
|
|
13
|
+
"libraries": {
|
|
14
|
+
"@aws-cdk/core": "1.12.0",
|
|
15
|
+
"@aws-cdk/cx-api": "1.12.0",
|
|
16
|
+
"jsii-runtime": "node.js/v8.11.4"
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
process.stdout.write(JSON.stringify({
|
|
2
|
+
"version": "1.10.0",
|
|
3
|
+
"artifacts": {
|
|
4
|
+
"InitStack": {
|
|
5
|
+
"type": "aws:cloudformation:stack",
|
|
6
|
+
"environment": `aws://${process.env.TEST_ACCOUNT}/${process.env.TEST_REGION}`,
|
|
7
|
+
"properties": {
|
|
8
|
+
"templateFile": "InitStack.template.json"
|
|
9
|
+
}
|
|
10
|
+
}
|
|
11
|
+
},
|
|
12
|
+
"runtime": {
|
|
13
|
+
"libraries": {
|
|
14
|
+
"@aws-cdk/core": "1.14.0",
|
|
15
|
+
"@aws-cdk/cx-api": "1.14.0",
|
|
16
|
+
"@aws-cdk/aws-ec2": "1.14.0",
|
|
17
|
+
"@aws-cdk/aws-iam": "1.14.0",
|
|
18
|
+
"@aws-cdk/region-info": "1.14.0",
|
|
19
|
+
"@aws-cdk/aws-ssm": "1.14.0",
|
|
20
|
+
"@aws-cdk/aws-cloudwatch": "1.14.0",
|
|
21
|
+
"jsii-runtime": "node.js/v8.11.4"
|
|
22
|
+
}
|
|
23
|
+
},
|
|
24
|
+
"missing": [
|
|
25
|
+
{
|
|
26
|
+
"key": `vpc-provider:account=${process.env.TEST_ACCOUNT}:filter.isDefault=true:region=${process.env.TEST_REGION}`,
|
|
27
|
+
"props": {
|
|
28
|
+
"account": process.env.TEST_ACCOUNT,
|
|
29
|
+
"region": process.env.TEST_REGION,
|
|
30
|
+
"filter": {
|
|
31
|
+
"isDefault": "true"
|
|
32
|
+
}
|
|
33
|
+
},
|
|
34
|
+
"provider": "vpc-provider"
|
|
35
|
+
}
|
|
36
|
+
]
|
|
37
|
+
}, undefined, 2));
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
process.stdout.write(JSON.stringify({
|
|
2
|
+
"version": "1.10.0",
|
|
3
|
+
"artifacts": {
|
|
4
|
+
"InitStack": {
|
|
5
|
+
"type": "aws:cloudformation:stack",
|
|
6
|
+
"environment": `aws://${process.env.TEST_ACCOUNT}/${process.env.TEST_REGION}`,
|
|
7
|
+
"properties": {
|
|
8
|
+
"templateFile": "InitStack.template.json"
|
|
9
|
+
}
|
|
10
|
+
}
|
|
11
|
+
},
|
|
12
|
+
"runtime": {
|
|
13
|
+
"libraries": {
|
|
14
|
+
"@aws-cdk/core": "1.14.0",
|
|
15
|
+
"@aws-cdk/cx-api": "1.14.0",
|
|
16
|
+
"@aws-cdk/aws-ec2": "1.14.0",
|
|
17
|
+
"@aws-cdk/aws-iam": "1.14.0",
|
|
18
|
+
"@aws-cdk/region-info": "1.14.0",
|
|
19
|
+
"@aws-cdk/aws-ssm": "1.14.0",
|
|
20
|
+
"@aws-cdk/aws-cloudwatch": "1.14.0",
|
|
21
|
+
"jsii-runtime": "node.js/v8.11.4"
|
|
22
|
+
}
|
|
23
|
+
},
|
|
24
|
+
"missing": [
|
|
25
|
+
{
|
|
26
|
+
"key": `availability-zones:account=${process.env.TEST_ACCOUNT}:region=${process.env.TEST_REGION}`,
|
|
27
|
+
"props": {
|
|
28
|
+
"account": process.env.TEST_ACCOUNT,
|
|
29
|
+
"region": process.env.TEST_REGION,
|
|
30
|
+
},
|
|
31
|
+
"provider": "availability-zones"
|
|
32
|
+
}
|
|
33
|
+
]
|
|
34
|
+
}, undefined, 2));
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
const path = require('path');
|
|
2
|
+
|
|
3
|
+
const rootDir = path.resolve(__dirname, '..', 'tests', process.env.TEST_SUITE_NAME);
|
|
4
|
+
|
|
5
|
+
if (rootDir.includes('node_modules')) {
|
|
6
|
+
// Jest < 28 under no circumstances supports loading test if there's node_modules anywhere in the path,
|
|
7
|
+
// and Jest >= 28 requires a newer TypeScript version than the one we support.
|
|
8
|
+
throw new Error(`This package must not be 'npm install'ed (found node_modules in dir: ${rootDir})`);
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
module.exports = {
|
|
12
|
+
rootDir,
|
|
13
|
+
testMatch: [`**/*.integtest.js`],
|
|
14
|
+
moduleFileExtensions: ["js"],
|
|
15
|
+
|
|
16
|
+
testEnvironment: "node",
|
|
17
|
+
testTimeout: 300000,
|
|
18
|
+
|
|
19
|
+
// Affects test.concurrent(), these are self-limiting anyway
|
|
20
|
+
maxConcurrency: 10,
|
|
21
|
+
reporters: [
|
|
22
|
+
"default",
|
|
23
|
+
[ "jest-junit", { suiteName: "jest tests", outputDirectory: "coverage" } ]
|
|
24
|
+
]
|
|
25
|
+
};
|
package/skip-tests.txt
ADDED
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
# This file is empty on purpose. Leave it here as documentation
|
|
2
|
+
# and an example.
|
|
3
|
+
#
|
|
4
|
+
# Copy this file to cli-regression-patches/vX.Y.Z/skip-tests.txt
|
|
5
|
+
# and edit it there if you want to exclude certain tests from running
|
|
6
|
+
# when performing a certain version's regression tests.
|
|
7
|
+
#
|
|
8
|
+
# Put a test name on a line by itself to skip it.
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
# CDK CLI Integration Tests
|
|
2
|
+
|
|
3
|
+
These tests require AWS credentials, and exercise various aspects of the
|
|
4
|
+
CLI on a simple JavaScript CDK app (stored in `app/`).
|
|
5
|
+
|
|
6
|
+
## Entry point
|
|
7
|
+
|
|
8
|
+
```
|
|
9
|
+
../run-against-repo ./test.sh
|
|
10
|
+
```
|
|
11
|
+
|
|
12
|
+
Running against a failing dist build:
|
|
13
|
+
|
|
14
|
+
```
|
|
15
|
+
# Download and unpack the zip
|
|
16
|
+
.../CMkBR4V$ tar xzvf js/aws-cdk-[0-9]*.tgz
|
|
17
|
+
.../CMkBR4V$ package/test/integ/run-against-dist package/test/integ/cli/test.sh
|
|
18
|
+
```
|
|
19
|
+
|
|
20
|
+
## Adding tests
|
|
21
|
+
|
|
22
|
+
Even though tests are now written in TypeScript, this does not
|
|
23
|
+
conceptually change their SUT! They are still testing the CLI via
|
|
24
|
+
running it as a subprocess, they are NOT reaching directly into the CLI
|
|
25
|
+
code to test its private methods. It's just that setup/teardown/error
|
|
26
|
+
handling/assertions/AWS calls are much more convenient to do in a real
|
|
27
|
+
programming language.
|
|
28
|
+
|
|
29
|
+
Compilation of the tests is done as part of the normal package build, at
|
|
30
|
+
which point it is using the dependencies brought in by the containing
|
|
31
|
+
`aws-cdk` package's `package.json`.
|
|
32
|
+
|
|
33
|
+
When run in a non-development repo (as done during integ tests or canary runs),
|
|
34
|
+
the required dependencies are brought in just-in-time via `test.sh`. Any
|
|
35
|
+
new dependencies added for the tests should be added there as well. But, better
|
|
36
|
+
yet, don't add any dependencies at all. You shouldn't need to, these tests
|
|
37
|
+
are simple.
|
|
38
|
+
|
|
39
|
+
## Configuration
|
|
40
|
+
|
|
41
|
+
AWS credentials must be configured.
|
|
42
|
+
|
|
43
|
+
Optional configuration:
|
|
44
|
+
|
|
45
|
+
* `AWS_DEFAULT_REGION`, what region to deploy the stacks in.
|
|
46
|
+
* `IS_CANARY`, true or false. Affects the default stack name prefix to make
|
|
47
|
+
integration test and canary runs unique.
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|