@aws-cdk-testing/cli-integ 0.0.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/.eslintrc.js +9 -0
- package/LICENSE +202 -0
- package/NOTICE +16 -0
- package/README.md +205 -0
- package/bin/apply-patches +22 -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 +54 -0
- package/bin/query-github.ts +56 -0
- package/bin/run-suite +2 -0
- package/bin/run-suite.d.ts +1 -0
- package/bin/run-suite.js +131 -0
- package/bin/run-suite.ts +140 -0
- package/bin/stage-distribution +2 -0
- package/bin/stage-distribution.d.ts +1 -0
- package/bin/stage-distribution.js +217 -0
- package/bin/stage-distribution.ts +267 -0
- package/bin/test-root +2 -0
- package/bin/test-root.d.ts +1 -0
- package/bin/test-root.js +6 -0
- package/bin/test-root.ts +3 -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 +51 -0
- package/lib/aws.js +206 -0
- package/lib/aws.ts +263 -0
- package/lib/corking.d.ts +12 -0
- package/lib/corking.js +35 -0
- package/lib/corking.ts +33 -0
- package/lib/eventually.d.ts +20 -0
- package/lib/eventually.js +34 -0
- package/lib/eventually.ts +42 -0
- package/lib/files.d.ts +15 -0
- package/lib/files.js +80 -0
- package/lib/files.ts +80 -0
- package/lib/github.d.ts +4 -0
- package/lib/github.js +43 -0
- package/lib/github.ts +43 -0
- package/lib/index.d.ts +13 -0
- package/lib/index.js +30 -0
- package/lib/index.ts +13 -0
- package/lib/integ-test.d.ts +10 -0
- package/lib/integ-test.js +70 -0
- package/lib/integ-test.ts +81 -0
- package/lib/lists.d.ts +1 -0
- package/lib/lists.js +11 -0
- package/lib/lists.ts +9 -0
- package/lib/memoize.d.ts +6 -0
- package/lib/memoize.js +18 -0
- package/lib/memoize.ts +14 -0
- package/lib/npm.d.ts +8 -0
- package/lib/npm.js +38 -0
- package/lib/npm.ts +41 -0
- package/lib/package-sources/release-source.d.ts +23 -0
- package/lib/package-sources/release-source.js +71 -0
- package/lib/package-sources/release-source.ts +81 -0
- package/lib/package-sources/repo-source.d.ts +30 -0
- package/lib/package-sources/repo-source.js +97 -0
- package/lib/package-sources/repo-source.ts +111 -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 +43 -0
- package/lib/package-sources/repo-tools/npm.ts +48 -0
- package/lib/package-sources/source.d.ts +28 -0
- package/lib/package-sources/source.js +3 -0
- package/lib/package-sources/source.ts +35 -0
- package/lib/package-sources/subprocess.d.ts +3 -0
- package/lib/package-sources/subprocess.js +17 -0
- package/lib/package-sources/subprocess.ts +15 -0
- package/lib/resource-pool.d.ts +50 -0
- package/lib/resource-pool.js +117 -0
- package/lib/resource-pool.ts +140 -0
- package/lib/resources.d.ts +1 -0
- package/lib/resources.js +6 -0
- package/lib/resources.ts +4 -0
- package/lib/shell.d.ts +56 -0
- package/lib/shell.js +123 -0
- package/lib/shell.ts +168 -0
- package/lib/staging/codeartifact.d.ts +44 -0
- package/lib/staging/codeartifact.js +281 -0
- package/lib/staging/codeartifact.ts +387 -0
- package/lib/staging/maven.d.ts +5 -0
- package/lib/staging/maven.js +91 -0
- package/lib/staging/maven.ts +95 -0
- package/lib/staging/npm.d.ts +4 -0
- package/lib/staging/npm.js +55 -0
- package/lib/staging/npm.ts +62 -0
- package/lib/staging/nuget.d.ts +4 -0
- package/lib/staging/nuget.js +69 -0
- package/lib/staging/nuget.ts +75 -0
- package/lib/staging/parallel-shell.d.ts +5 -0
- package/lib/staging/parallel-shell.js +45 -0
- package/lib/staging/parallel-shell.ts +51 -0
- package/lib/staging/pypi.d.ts +4 -0
- package/lib/staging/pypi.js +48 -0
- package/lib/staging/pypi.ts +50 -0
- package/lib/staging/usage-dir.d.ts +31 -0
- package/lib/staging/usage-dir.js +87 -0
- package/lib/staging/usage-dir.ts +99 -0
- package/lib/with-aws.d.ts +14 -0
- package/lib/with-aws.js +60 -0
- package/lib/with-aws.ts +67 -0
- package/lib/with-cdk-app.d.ts +210 -0
- package/lib/with-cdk-app.js +539 -0
- package/lib/with-cdk-app.ts +742 -0
- package/lib/with-cli-lib.d.ts +17 -0
- package/lib/with-cli-lib.js +123 -0
- package/lib/with-cli-lib.ts +134 -0
- package/lib/with-packages.d.ts +5 -0
- package/lib/with-packages.js +13 -0
- package/lib/with-packages.ts +15 -0
- package/lib/with-sam.d.ts +33 -0
- package/lib/with-sam.js +258 -0
- package/lib/with-sam.ts +288 -0
- package/lib/with-temporary-directory.d.ts +5 -0
- package/lib/with-temporary-directory.js +31 -0
- package/lib/with-temporary-directory.ts +35 -0
- package/lib/with-timeout.d.ts +19 -0
- package/lib/with-timeout.js +34 -0
- package/lib/with-timeout.ts +33 -0
- package/lib/xpmutex.d.ts +43 -0
- package/lib/xpmutex.js +207 -0
- package/lib/xpmutex.ts +218 -0
- package/package.json +111 -0
- package/resources/bootstrap-templates/session-tags.all-roles-deny-all.yaml +703 -0
- package/resources/bootstrap-templates/session-tags.deploy-role-deny-sqs.yaml +700 -0
- package/resources/cdk-apps/app/app.js +926 -0
- package/resources/cdk-apps/app/appsync.hotswap.graphql +3 -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 +65 -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/rollback-test-app/app.js +110 -0
- package/resources/cdk-apps/rollback-test-app/cdk.json +7 -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/cdk-apps/simple-app/app.js +26 -0
- package/resources/cdk-apps/simple-app/cdk.json +7 -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/cli-regression-patches/v2.130.0/NOTES.md +1 -0
- package/resources/cli-regression-patches/v2.130.0/node_modules/@aws-cdk-testing/cli-integ/resources/cdk-apps/sam_cdk_integ_app/lib/nested-stack.js +19 -0
- package/resources/cli-regression-patches/v2.130.0/node_modules/@aws-cdk-testing/cli-integ/resources/cdk-apps/sam_cdk_integ_app/lib/test-stack.js +134 -0
- package/resources/cli-regression-patches/v2.130.0/skip-tests.txt +5 -0
- package/resources/cli-regression-patches/v2.132.0/NOTES.md +1 -0
- package/resources/cli-regression-patches/v2.132.0/skip-tests.txt +4 -0
- package/resources/cli-regression-patches/v2.142.0/NOTES.md +1 -0
- package/resources/cli-regression-patches/v2.142.0/skip-tests.txt +4 -0
- package/resources/cli-regression-patches/v2.160.0/skip-tests.txt +2 -0
- package/resources/cli-regression-patches/v2.161.0/NOTES.md +1 -0
- package/resources/cli-regression-patches/v2.161.0/skip-tests.txt +5 -0
- package/resources/cli-regression-patches/v2.166.0/NOTES.md +1 -0
- package/resources/cli-regression-patches/v2.166.0/skip-tests.txt +2 -0
- package/resources/cloud-assemblies/0.36.0/InitStack.template.json +1 -0
- package/resources/cloud-assemblies/0.36.0/cdk.out +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/cdk.out +1 -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/cdk.out +1 -0
- package/resources/cloud-assemblies/1.10.0-request-azs/manifest.json.js +34 -0
- package/resources/integ.jest.config.js +25 -0
- package/resources/templates/sqs-template.json +36 -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 +412 -0
- package/tests/cli-integ-tests/bootstrapping.integtest.ts +493 -0
- package/tests/cli-integ-tests/cli-lib.integtest.d.ts +1 -0
- package/tests/cli-integ-tests/cli-lib.integtest.js +62 -0
- package/tests/cli-integ-tests/cli-lib.integtest.ts +90 -0
- package/tests/cli-integ-tests/cli.integtest.d.ts +1 -0
- package/tests/cli-integ-tests/cli.integtest.js +2104 -0
- package/tests/cli-integ-tests/cli.integtest.ts +2874 -0
- package/tests/cli-integ-tests/garbage-collection.integtest.d.ts +1 -0
- package/tests/cli-integ-tests/garbage-collection.integtest.js +314 -0
- package/tests/cli-integ-tests/garbage-collection.integtest.ts +392 -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-csharp/init-csharp.integtest.ts +15 -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-fsharp/init-fsharp.integtest.ts +15 -0
- package/tests/init-go/init-go.integtest.d.ts +1 -0
- package/tests/init-go/init-go.integtest.js +21 -0
- package/tests/init-go/init-go.integtest.ts +23 -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-java/init-java.integtest.ts +14 -0
- package/tests/init-javascript/init-javascript.integtest.d.ts +1 -0
- package/tests/init-javascript/init-javascript.integtest.js +53 -0
- package/tests/init-javascript/init-javascript.integtest.ts +59 -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-python/init-python.integtest.ts +20 -0
- package/tests/init-typescript-app/init-typescript-app.integtest.d.ts +1 -0
- package/tests/init-typescript-app/init-typescript-app.integtest.js +54 -0
- package/tests/init-typescript-app/init-typescript-app.integtest.ts +66 -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/init-typescript-lib/init-typescript-lib.integtest.ts +13 -0
- package/tests/tool-integrations/amplify.integtest.d.ts +1 -0
- package/tests/tool-integrations/amplify.integtest.js +39 -0
- package/tests/tool-integrations/amplify.integtest.ts +43 -0
- package/tests/tool-integrations/with-tool-context.d.ts +9 -0
- package/tests/tool-integrations/with-tool-context.js +13 -0
- package/tests/tool-integrations/with-tool-context.ts +14 -0
- package/tests/uberpackage/uberpackage.integtest.d.ts +1 -0
- package/tests/uberpackage/uberpackage.integtest.js +11 -0
- package/tests/uberpackage/uberpackage.integtest.ts +11 -0
|
@@ -0,0 +1,126 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const aws_helpers_1 = require("./aws-helpers");
|
|
4
|
+
const cdk_helpers_1 = require("./cdk-helpers");
|
|
5
|
+
jest.setTimeout(600000);
|
|
6
|
+
const QUALIFIER = randomString();
|
|
7
|
+
beforeAll(async () => {
|
|
8
|
+
await cdk_helpers_1.prepareAppFixture();
|
|
9
|
+
});
|
|
10
|
+
beforeEach(async () => {
|
|
11
|
+
await cdk_helpers_1.cleanup();
|
|
12
|
+
});
|
|
13
|
+
afterEach(async () => {
|
|
14
|
+
await cdk_helpers_1.cleanup();
|
|
15
|
+
});
|
|
16
|
+
test('can bootstrap without execution', async () => {
|
|
17
|
+
var _a;
|
|
18
|
+
const bootstrapStackName = cdk_helpers_1.fullStackName('bootstrap-stack');
|
|
19
|
+
await cdk_helpers_1.cdk(['bootstrap',
|
|
20
|
+
'--toolkit-stack-name', bootstrapStackName,
|
|
21
|
+
'--no-execute']);
|
|
22
|
+
const resp = await aws_helpers_1.cloudFormation('describeStacks', {
|
|
23
|
+
StackName: bootstrapStackName,
|
|
24
|
+
});
|
|
25
|
+
expect((_a = resp.Stacks) === null || _a === void 0 ? void 0 : _a[0].StackStatus).toEqual('REVIEW_IN_PROGRESS');
|
|
26
|
+
});
|
|
27
|
+
test('upgrade legacy bootstrap stack to new bootstrap stack while in use', async () => {
|
|
28
|
+
const bootstrapStackName = cdk_helpers_1.fullStackName('bootstrap-stack');
|
|
29
|
+
const legacyBootstrapBucketName = `aws-cdk-bootstrap-integ-test-legacy-bckt-${randomString()}`;
|
|
30
|
+
const newBootstrapBucketName = `aws-cdk-bootstrap-integ-test-v2-bckt-${randomString()}`;
|
|
31
|
+
cdk_helpers_1.rememberToDeleteBucket(legacyBootstrapBucketName); // This one will leak
|
|
32
|
+
cdk_helpers_1.rememberToDeleteBucket(newBootstrapBucketName); // This one shouldn't leak if the test succeeds, but let's be safe in case it doesn't
|
|
33
|
+
// Legacy bootstrap
|
|
34
|
+
await cdk_helpers_1.cdk(['bootstrap',
|
|
35
|
+
'--toolkit-stack-name', bootstrapStackName,
|
|
36
|
+
'--bootstrap-bucket-name', legacyBootstrapBucketName]);
|
|
37
|
+
// Deploy stack that uses file assets
|
|
38
|
+
await cdk_helpers_1.cdkDeploy('lambda', {
|
|
39
|
+
options: ['--toolkit-stack-name', bootstrapStackName],
|
|
40
|
+
});
|
|
41
|
+
// Upgrade bootstrap stack to "new" style
|
|
42
|
+
await cdk_helpers_1.cdk(['bootstrap',
|
|
43
|
+
'--toolkit-stack-name', bootstrapStackName,
|
|
44
|
+
'--bootstrap-bucket-name', newBootstrapBucketName,
|
|
45
|
+
'--qualifier', QUALIFIER], {
|
|
46
|
+
modEnv: {
|
|
47
|
+
CDK_NEW_BOOTSTRAP: '1',
|
|
48
|
+
},
|
|
49
|
+
});
|
|
50
|
+
// (Force) deploy stack again
|
|
51
|
+
// --force to bypass the check which says that the template hasn't changed.
|
|
52
|
+
await cdk_helpers_1.cdkDeploy('lambda', {
|
|
53
|
+
options: [
|
|
54
|
+
'--toolkit-stack-name', bootstrapStackName,
|
|
55
|
+
'--force',
|
|
56
|
+
],
|
|
57
|
+
});
|
|
58
|
+
});
|
|
59
|
+
test.skip('deploy new style synthesis to new style bootstrap', async () => {
|
|
60
|
+
const bootstrapStackName = cdk_helpers_1.fullStackName('bootstrap-stack');
|
|
61
|
+
await cdk_helpers_1.cdk(['bootstrap',
|
|
62
|
+
'--toolkit-stack-name', bootstrapStackName,
|
|
63
|
+
'--qualifier', QUALIFIER,
|
|
64
|
+
'--cloudformation-execution-policies', 'arn:aws:iam::aws:policy/AdministratorAccess',
|
|
65
|
+
], {
|
|
66
|
+
modEnv: {
|
|
67
|
+
CDK_NEW_BOOTSTRAP: '1',
|
|
68
|
+
},
|
|
69
|
+
});
|
|
70
|
+
// Deploy stack that uses file assets
|
|
71
|
+
await cdk_helpers_1.cdkDeploy('lambda', {
|
|
72
|
+
options: [
|
|
73
|
+
'--toolkit-stack-name', bootstrapStackName,
|
|
74
|
+
'--context', `@aws-cdk/core:bootstrapQualifier=${QUALIFIER}`,
|
|
75
|
+
'--context', '@aws-cdk/core:newStyleStackSynthesis=1',
|
|
76
|
+
],
|
|
77
|
+
});
|
|
78
|
+
});
|
|
79
|
+
test('deploy old style synthesis to new style bootstrap', async () => {
|
|
80
|
+
const bootstrapStackName = cdk_helpers_1.fullStackName('bootstrap-stack');
|
|
81
|
+
await cdk_helpers_1.cdk(['bootstrap',
|
|
82
|
+
'--toolkit-stack-name', bootstrapStackName,
|
|
83
|
+
'--qualifier', QUALIFIER,
|
|
84
|
+
'--cloudformation-execution-policies', 'arn:aws:iam::aws:policy/AdministratorAccess',
|
|
85
|
+
], {
|
|
86
|
+
modEnv: {
|
|
87
|
+
CDK_NEW_BOOTSTRAP: '1',
|
|
88
|
+
},
|
|
89
|
+
});
|
|
90
|
+
// Deploy stack that uses file assets
|
|
91
|
+
await cdk_helpers_1.cdkDeploy('lambda', {
|
|
92
|
+
options: [
|
|
93
|
+
'--toolkit-stack-name', bootstrapStackName,
|
|
94
|
+
],
|
|
95
|
+
});
|
|
96
|
+
});
|
|
97
|
+
test('deploying new style synthesis to old style bootstrap fails', async () => {
|
|
98
|
+
const bootstrapStackName = cdk_helpers_1.fullStackName('bootstrap-stack');
|
|
99
|
+
await cdk_helpers_1.cdk(['bootstrap', '--toolkit-stack-name', bootstrapStackName]);
|
|
100
|
+
// Deploy stack that uses file assets, this fails because the bootstrap stack
|
|
101
|
+
// is version checked.
|
|
102
|
+
await expect(cdk_helpers_1.cdkDeploy('lambda', {
|
|
103
|
+
options: [
|
|
104
|
+
'--toolkit-stack-name', bootstrapStackName,
|
|
105
|
+
'--context', '@aws-cdk/core:newStyleStackSynthesis=1',
|
|
106
|
+
],
|
|
107
|
+
})).rejects.toThrow('exited with error');
|
|
108
|
+
});
|
|
109
|
+
test('can create multiple legacy bootstrap stacks', async () => {
|
|
110
|
+
var _a;
|
|
111
|
+
const bootstrapStackName1 = cdk_helpers_1.fullStackName('bootstrap-stack-1');
|
|
112
|
+
const bootstrapStackName2 = cdk_helpers_1.fullStackName('bootstrap-stack-2');
|
|
113
|
+
// deploy two toolkit stacks into the same environment (see #1416)
|
|
114
|
+
// one with tags
|
|
115
|
+
await cdk_helpers_1.cdk(['bootstrap', '-v', '--toolkit-stack-name', bootstrapStackName1, '--tags', 'Foo=Bar']);
|
|
116
|
+
await cdk_helpers_1.cdk(['bootstrap', '-v', '--toolkit-stack-name', bootstrapStackName2]);
|
|
117
|
+
const response = await aws_helpers_1.cloudFormation('describeStacks', { StackName: bootstrapStackName1 });
|
|
118
|
+
expect((_a = response.Stacks) === null || _a === void 0 ? void 0 : _a[0].Tags).toEqual([
|
|
119
|
+
{ Key: 'Foo', Value: 'Bar' },
|
|
120
|
+
]);
|
|
121
|
+
});
|
|
122
|
+
function randomString() {
|
|
123
|
+
// Crazy
|
|
124
|
+
return Math.random().toString(36).replace(/[^a-z0-9]+/g, '');
|
|
125
|
+
}
|
|
126
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bootstrapping.integtest.js","sourceRoot":"","sources":["bootstrapping.integtest.ts"],"names":[],"mappings":";;AAAA,+CAA+C;AAC/C,+CAAkH;AAElH,IAAI,CAAC,UAAU,CAAC,MAAO,CAAC,CAAC;AAEzB,MAAM,SAAS,GAAG,YAAY,EAAE,CAAC;AAEjC,SAAS,CAAC,KAAK,IAAI,EAAE;IACnB,MAAM,+BAAiB,EAAE,CAAC;AAC5B,CAAC,CAAC,CAAC;AAEH,UAAU,CAAC,KAAK,IAAI,EAAE;IACpB,MAAM,qBAAO,EAAE,CAAC;AAClB,CAAC,CAAC,CAAC;AAEH,SAAS,CAAC,KAAK,IAAI,EAAE;IACnB,MAAM,qBAAO,EAAE,CAAC;AAClB,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,iCAAiC,EAAE,KAAK,IAAI,EAAE;;IACjD,MAAM,kBAAkB,GAAG,2BAAa,CAAC,iBAAiB,CAAC,CAAC;IAE5D,MAAM,iBAAG,CAAC,CAAC,WAAW;QACpB,sBAAsB,EAAE,kBAAkB;QAC1C,cAAc,CAAC,CAAC,CAAC;IAEnB,MAAM,IAAI,GAAG,MAAM,4BAAc,CAAC,gBAAgB,EAAE;QAClD,SAAS,EAAE,kBAAkB;KAC9B,CAAC,CAAC;IAEH,MAAM,OAAC,IAAI,CAAC,MAAM,0CAAG,CAAC,EAAE,WAAW,CAAC,CAAC,OAAO,CAAC,oBAAoB,CAAC,CAAC;AACrE,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,oEAAoE,EAAE,KAAK,IAAI,EAAE;IACpF,MAAM,kBAAkB,GAAG,2BAAa,CAAC,iBAAiB,CAAC,CAAC;IAE5D,MAAM,yBAAyB,GAAG,4CAA4C,YAAY,EAAE,EAAE,CAAC;IAC/F,MAAM,sBAAsB,GAAG,wCAAwC,YAAY,EAAE,EAAE,CAAC;IACxF,oCAAsB,CAAC,yBAAyB,CAAC,CAAC,CAAE,qBAAqB;IACzE,oCAAsB,CAAC,sBAAsB,CAAC,CAAC,CAAK,qFAAqF;IAEzI,mBAAmB;IACnB,MAAM,iBAAG,CAAC,CAAC,WAAW;QACpB,sBAAsB,EAAE,kBAAkB;QAC1C,yBAAyB,EAAE,yBAAyB,CAAC,CAAC,CAAC;IAEzD,qCAAqC;IACrC,MAAM,uBAAS,CAAC,QAAQ,EAAE;QACxB,OAAO,EAAE,CAAC,sBAAsB,EAAE,kBAAkB,CAAC;KACtD,CAAC,CAAC;IAEH,yCAAyC;IACzC,MAAM,iBAAG,CAAC,CAAC,WAAW;QACpB,sBAAsB,EAAE,kBAAkB;QAC1C,yBAAyB,EAAE,sBAAsB;QACjD,aAAa,EAAE,SAAS,CAAC,EAAE;QAC3B,MAAM,EAAE;YACN,iBAAiB,EAAE,GAAG;SACvB;KACF,CAAC,CAAC;IAEH,6BAA6B;IAC7B,2EAA2E;IAC3E,MAAM,uBAAS,CAAC,QAAQ,EAAE;QACxB,OAAO,EAAE;YACP,sBAAsB,EAAE,kBAAkB;YAC1C,SAAS;SACV;KACF,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,mDAAmD,EAAE,KAAK,IAAI,EAAE;IACnE,MAAM,kBAAkB,GAAG,2BAAa,CAAC,iBAAiB,CAAC,CAAC;IAE5D,MAAM,iBAAG,CAAC,CAAC,WAAW;QACpB,sBAAsB,EAAE,kBAAkB;QAC1C,aAAa,EAAE,SAAS;QACxB,qCAAqC,EAAE,6CAA6C;KACrF,EAAE;QACD,MAAM,EAAE;YACN,iBAAiB,EAAE,GAAG;SACvB;KACF,CAAC,CAAC;IAEH,qCAAqC;IACrC,MAAM,uBAAS,CAAC,QAAQ,EAAE;QACxB,OAAO,EAAE;YACP,sBAAsB,EAAE,kBAAkB;YAC1C,WAAW,EAAE,oCAAoC,SAAS,EAAE;YAC5D,WAAW,EAAE,wCAAwC;SACtD;KACF,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,mDAAmD,EAAE,KAAK,IAAI,EAAE;IACnE,MAAM,kBAAkB,GAAG,2BAAa,CAAC,iBAAiB,CAAC,CAAC;IAE5D,MAAM,iBAAG,CAAC,CAAC,WAAW;QACpB,sBAAsB,EAAE,kBAAkB;QAC1C,aAAa,EAAE,SAAS;QACxB,qCAAqC,EAAE,6CAA6C;KACrF,EAAE;QACD,MAAM,EAAE;YACN,iBAAiB,EAAE,GAAG;SACvB;KACF,CAAC,CAAC;IAEH,qCAAqC;IACrC,MAAM,uBAAS,CAAC,QAAQ,EAAE;QACxB,OAAO,EAAE;YACP,sBAAsB,EAAE,kBAAkB;SAC3C;KACF,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,4DAA4D,EAAE,KAAK,IAAI,EAAE;IAC5E,MAAM,kBAAkB,GAAG,2BAAa,CAAC,iBAAiB,CAAC,CAAC;IAE5D,MAAM,iBAAG,CAAC,CAAC,WAAW,EAAE,sBAAsB,EAAE,kBAAkB,CAAC,CAAC,CAAC;IAErE,6EAA6E;IAC7E,sBAAsB;IACtB,MAAM,MAAM,CAAC,uBAAS,CAAC,QAAQ,EAAE;QAC/B,OAAO,EAAE;YACP,sBAAsB,EAAE,kBAAkB;YAC1C,WAAW,EAAE,wCAAwC;SACtD;KACF,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,mBAAmB,CAAC,CAAC;AAC3C,CAAC,CAAC,CAAC;AAEH,IAAI,CAAC,6CAA6C,EAAE,KAAK,IAAI,EAAE;;IAC7D,MAAM,mBAAmB,GAAG,2BAAa,CAAC,mBAAmB,CAAC,CAAC;IAC/D,MAAM,mBAAmB,GAAG,2BAAa,CAAC,mBAAmB,CAAC,CAAC;IAE/D,kEAAkE;IAClE,gBAAgB;IAChB,MAAM,iBAAG,CAAC,CAAC,WAAW,EAAE,IAAI,EAAE,sBAAsB,EAAE,mBAAmB,EAAE,QAAQ,EAAE,SAAS,CAAC,CAAC,CAAC;IACjG,MAAM,iBAAG,CAAC,CAAC,WAAW,EAAE,IAAI,EAAE,sBAAsB,EAAE,mBAAmB,CAAC,CAAC,CAAC;IAE5E,MAAM,QAAQ,GAAG,MAAM,4BAAc,CAAC,gBAAgB,EAAE,EAAE,SAAS,EAAE,mBAAmB,EAAE,CAAC,CAAC;IAC5F,MAAM,OAAC,QAAQ,CAAC,MAAM,0CAAG,CAAC,EAAE,IAAI,CAAC,CAAC,OAAO,CAAC;QACxC,EAAE,GAAG,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE;KAC7B,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,SAAS,YAAY;IACnB,QAAQ;IACR,OAAO,IAAI,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,OAAO,CAAC,aAAa,EAAE,EAAE,CAAC,CAAC;AAC/D,CAAC","sourcesContent":["import { cloudFormation } from './aws-helpers';\nimport { cdk, cdkDeploy, cleanup, fullStackName, prepareAppFixture, rememberToDeleteBucket } from './cdk-helpers';\n\njest.setTimeout(600_000);\n\nconst QUALIFIER = randomString();\n\nbeforeAll(async () => {\n  await prepareAppFixture();\n});\n\nbeforeEach(async () => {\n  await cleanup();\n});\n\nafterEach(async () => {\n  await cleanup();\n});\n\ntest('can bootstrap without execution', async () => {\n  const bootstrapStackName = fullStackName('bootstrap-stack');\n\n  await cdk(['bootstrap',\n    '--toolkit-stack-name', bootstrapStackName,\n    '--no-execute']);\n\n  const resp = await cloudFormation('describeStacks', {\n    StackName: bootstrapStackName,\n  });\n\n  expect(resp.Stacks?.[0].StackStatus).toEqual('REVIEW_IN_PROGRESS');\n});\n\ntest('upgrade legacy bootstrap stack to new bootstrap stack while in use', async () => {\n  const bootstrapStackName = fullStackName('bootstrap-stack');\n\n  const legacyBootstrapBucketName = `aws-cdk-bootstrap-integ-test-legacy-bckt-${randomString()}`;\n  const newBootstrapBucketName = `aws-cdk-bootstrap-integ-test-v2-bckt-${randomString()}`;\n  rememberToDeleteBucket(legacyBootstrapBucketName);  // This one will leak\n  rememberToDeleteBucket(newBootstrapBucketName);     // This one shouldn't leak if the test succeeds, but let's be safe in case it doesn't\n\n  // Legacy bootstrap\n  await cdk(['bootstrap',\n    '--toolkit-stack-name', bootstrapStackName,\n    '--bootstrap-bucket-name', legacyBootstrapBucketName]);\n\n  // Deploy stack that uses file assets\n  await cdkDeploy('lambda', {\n    options: ['--toolkit-stack-name', bootstrapStackName],\n  });\n\n  // Upgrade bootstrap stack to \"new\" style\n  await cdk(['bootstrap',\n    '--toolkit-stack-name', bootstrapStackName,\n    '--bootstrap-bucket-name', newBootstrapBucketName,\n    '--qualifier', QUALIFIER], {\n    modEnv: {\n      CDK_NEW_BOOTSTRAP: '1',\n    },\n  });\n\n  // (Force) deploy stack again\n  // --force to bypass the check which says that the template hasn't changed.\n  await cdkDeploy('lambda', {\n    options: [\n      '--toolkit-stack-name', bootstrapStackName,\n      '--force',\n    ],\n  });\n});\n\ntest('deploy new style synthesis to new style bootstrap', async () => {\n  const bootstrapStackName = fullStackName('bootstrap-stack');\n\n  await cdk(['bootstrap',\n    '--toolkit-stack-name', bootstrapStackName,\n    '--qualifier', QUALIFIER,\n    '--cloudformation-execution-policies', 'arn:aws:iam::aws:policy/AdministratorAccess',\n  ], {\n    modEnv: {\n      CDK_NEW_BOOTSTRAP: '1',\n    },\n  });\n\n  // Deploy stack that uses file assets\n  await cdkDeploy('lambda', {\n    options: [\n      '--toolkit-stack-name', bootstrapStackName,\n      '--context', `@aws-cdk/core:bootstrapQualifier=${QUALIFIER}`,\n      '--context', '@aws-cdk/core:newStyleStackSynthesis=1',\n    ],\n  });\n});\n\ntest('deploy old style synthesis to new style bootstrap', async () => {\n  const bootstrapStackName = fullStackName('bootstrap-stack');\n\n  await cdk(['bootstrap',\n    '--toolkit-stack-name', bootstrapStackName,\n    '--qualifier', QUALIFIER,\n    '--cloudformation-execution-policies', 'arn:aws:iam::aws:policy/AdministratorAccess',\n  ], {\n    modEnv: {\n      CDK_NEW_BOOTSTRAP: '1',\n    },\n  });\n\n  // Deploy stack that uses file assets\n  await cdkDeploy('lambda', {\n    options: [\n      '--toolkit-stack-name', bootstrapStackName,\n    ],\n  });\n});\n\ntest('deploying new style synthesis to old style bootstrap fails', async () => {\n  const bootstrapStackName = fullStackName('bootstrap-stack');\n\n  await cdk(['bootstrap', '--toolkit-stack-name', bootstrapStackName]);\n\n  // Deploy stack that uses file assets, this fails because the bootstrap stack\n  // is version checked.\n  await expect(cdkDeploy('lambda', {\n    options: [\n      '--toolkit-stack-name', bootstrapStackName,\n      '--context', '@aws-cdk/core:newStyleStackSynthesis=1',\n    ],\n  })).rejects.toThrow('exited with error');\n});\n\ntest('can create multiple legacy bootstrap stacks', async () => {\n  const bootstrapStackName1 = fullStackName('bootstrap-stack-1');\n  const bootstrapStackName2 = fullStackName('bootstrap-stack-2');\n\n  // deploy two toolkit stacks into the same environment (see #1416)\n  // one with tags\n  await cdk(['bootstrap', '-v', '--toolkit-stack-name', bootstrapStackName1, '--tags', 'Foo=Bar']);\n  await cdk(['bootstrap', '-v', '--toolkit-stack-name', bootstrapStackName2]);\n\n  const response = await cloudFormation('describeStacks', { StackName: bootstrapStackName1 });\n  expect(response.Stacks?.[0].Tags).toEqual([\n    { Key: 'Foo', Value: 'Bar' },\n  ]);\n});\n\nfunction randomString() {\n  // Crazy\n  return Math.random().toString(36).replace(/[^a-z0-9]+/g, '');\n}\n"]}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
#!/bin/bash
|
|
2
|
+
set -euo pipefail
|
|
3
|
+
scriptdir=$(cd $(dirname $0) && pwd)
|
|
4
|
+
|
|
5
|
+
echo '~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~'
|
|
6
|
+
echo 'CLI Integration Tests'
|
|
7
|
+
echo '~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~'
|
|
8
|
+
|
|
9
|
+
current_version=$(node -p "require('${scriptdir}/../../../package.json').version")
|
|
10
|
+
|
|
11
|
+
# This allows injecting different versions, not just the current one.
|
|
12
|
+
# Useful when testing.
|
|
13
|
+
export VERSION_UNDER_TEST=${VERSION_UNDER_TEST:-${current_version}}
|
|
14
|
+
|
|
15
|
+
cd $scriptdir
|
|
16
|
+
|
|
17
|
+
# Install these dependencies that the tests (written in Jest) need.
|
|
18
|
+
# Only if we're not running from the repo, because if we are the
|
|
19
|
+
# dependencies have already been installed by the containing 'aws-cdk' package's
|
|
20
|
+
# package.json.
|
|
21
|
+
if ! npx --no-install jest --version; then
|
|
22
|
+
echo 'Looks like we need to install jest first. Hold on.' >& 2
|
|
23
|
+
npm install --prefix . jest aws-sdk
|
|
24
|
+
fi
|
|
25
|
+
|
|
26
|
+
npx jest --runInBand --verbose "$@"
|
|
@@ -0,0 +1,16 @@
|
|
|
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.
|
|
9
|
+
|
|
10
|
+
upgrade legacy bootstrap stack to new bootstrap stack while in use
|
|
11
|
+
deploy new style synthesis to new style bootstrap
|
|
12
|
+
deploy new style synthesis to new style bootstrap (with docker image)
|
|
13
|
+
deploy old style synthesis to new style bootstrap
|
|
14
|
+
can dump the template, modify and use it to deploy a custom bootstrap stack
|
|
15
|
+
switch on termination protection, switch is left alone on re-bootstrap
|
|
16
|
+
add tags, left alone on re-bootstrap
|
|
@@ -0,0 +1,245 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.outputFromStack = exports.deleteBucket = exports.deleteImageRepository = exports.emptyBucket = exports.sleep = exports.retry = exports.isBucketMissingError = exports.isStackMissingError = exports.stackStatus = exports.deleteStacks = exports.sts = exports.lambda = exports.iam = exports.sns = exports.ecr = exports.s3 = exports.cloudFormation = exports.testEnv = void 0;
|
|
4
|
+
const AWS = require("aws-sdk");
|
|
5
|
+
const cdk_helpers_1 = require("./cdk-helpers");
|
|
6
|
+
|
|
7
|
+
function chainableCredentials(region) {
|
|
8
|
+
const profileName = process.env.AWS_PROFILE;
|
|
9
|
+
if (process.env.CODEBUILD_BUILD_ARN && profileName) {
|
|
10
|
+
// in codebuild we must assume the role that the cdk uses
|
|
11
|
+
// otherwise credentials will just be picked up by the normal sdk
|
|
12
|
+
// heuristics and expire after an hour.
|
|
13
|
+
// can't use '~' since the SDK doesn't seem to expand it...?
|
|
14
|
+
const configPath = `${process.env.HOME}/.aws/config`;
|
|
15
|
+
const ini = new AWS.IniLoader().loadFrom({
|
|
16
|
+
filename: configPath,
|
|
17
|
+
isConfig: true,
|
|
18
|
+
});
|
|
19
|
+
const profile = ini[profileName];
|
|
20
|
+
if (!profile) {
|
|
21
|
+
throw new Error(`Profile '${profileName}' does not exist in config file (${configPath})`);
|
|
22
|
+
}
|
|
23
|
+
const arn = profile.role_arn;
|
|
24
|
+
const externalId = profile.external_id;
|
|
25
|
+
if (!arn) {
|
|
26
|
+
throw new Error(`role_arn does not exist in profile ${profileName}`);
|
|
27
|
+
}
|
|
28
|
+
if (!externalId) {
|
|
29
|
+
throw new Error(`external_id does not exist in profile ${externalId}`);
|
|
30
|
+
}
|
|
31
|
+
return new AWS.ChainableTemporaryCredentials({
|
|
32
|
+
params: {
|
|
33
|
+
RoleArn: arn,
|
|
34
|
+
ExternalId: externalId,
|
|
35
|
+
RoleSessionName: 'integ-tests',
|
|
36
|
+
},
|
|
37
|
+
stsConfig: {
|
|
38
|
+
region,
|
|
39
|
+
},
|
|
40
|
+
masterCredentials: new AWS.ECSCredentials(),
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
return undefined;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
exports.testEnv = async () => {
|
|
47
|
+
var _a, _b;
|
|
48
|
+
const region = (_b = (_a = process.env.AWS_REGION) !== null && _a !== void 0 ? _a : process.env.AWS_DEFAULT_REGION) !== null && _b !== void 0 ? _b : 'us-east-1';
|
|
49
|
+
const sts = new AWS.STS({
|
|
50
|
+
region: region,
|
|
51
|
+
credentials: chainableCredentials(region),
|
|
52
|
+
maxRetries: 8,
|
|
53
|
+
retryDelayOptions: { base: 500 },
|
|
54
|
+
});
|
|
55
|
+
const response = await sts.getCallerIdentity().promise();
|
|
56
|
+
const ret = {
|
|
57
|
+
account: response.Account,
|
|
58
|
+
region,
|
|
59
|
+
};
|
|
60
|
+
exports.testEnv = () => Promise.resolve(ret);
|
|
61
|
+
return ret;
|
|
62
|
+
};
|
|
63
|
+
|
|
64
|
+
exports.cloudFormation = makeAwsCaller(AWS.CloudFormation);
|
|
65
|
+
exports.s3 = makeAwsCaller(AWS.S3);
|
|
66
|
+
exports.ecr = makeAwsCaller(AWS.ECR);
|
|
67
|
+
exports.sns = makeAwsCaller(AWS.SNS);
|
|
68
|
+
exports.iam = makeAwsCaller(AWS.IAM);
|
|
69
|
+
exports.lambda = makeAwsCaller(AWS.Lambda);
|
|
70
|
+
exports.sts = makeAwsCaller(AWS.STS);
|
|
71
|
+
/**
|
|
72
|
+
* Perform an AWS call from nothing
|
|
73
|
+
*
|
|
74
|
+
* Create the correct client, do the call and resole the promise().
|
|
75
|
+
*/
|
|
76
|
+
async function awsCall(ctor, call, request) {
|
|
77
|
+
const env = await exports.testEnv();
|
|
78
|
+
const cfn = new ctor({
|
|
79
|
+
region: env.region,
|
|
80
|
+
credentials: chainableCredentials(env.region),
|
|
81
|
+
maxRetries: 6,
|
|
82
|
+
retryDelayOptions: {
|
|
83
|
+
base: 500,
|
|
84
|
+
},
|
|
85
|
+
});
|
|
86
|
+
const response = cfn[call](request);
|
|
87
|
+
try {
|
|
88
|
+
return await response.promise();
|
|
89
|
+
}
|
|
90
|
+
catch (e) {
|
|
91
|
+
const newErr = new Error(`${call}(${JSON.stringify(request)}): ${e.message}`);
|
|
92
|
+
newErr.code = e.code;
|
|
93
|
+
throw newErr;
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
/**
|
|
97
|
+
* Factory function to invoke 'awsCall' for specific services.
|
|
98
|
+
*
|
|
99
|
+
* Not strictly necessary but calling this replaces a whole bunch of annoying generics you otherwise have to type:
|
|
100
|
+
*
|
|
101
|
+
* ```ts
|
|
102
|
+
* export function cloudFormation<
|
|
103
|
+
* C extends keyof ServiceCalls<AWS.CloudFormation>,
|
|
104
|
+
* >(call: C, request: First<ServiceCalls<AWS.CloudFormation>[C]>): Promise<Second<ServiceCalls<AWS.CloudFormation>[C]>> {
|
|
105
|
+
* return awsCall(AWS.CloudFormation, call, request);
|
|
106
|
+
* }
|
|
107
|
+
* ```
|
|
108
|
+
*/
|
|
109
|
+
function makeAwsCaller(ctor) {
|
|
110
|
+
return (call, request) => {
|
|
111
|
+
return awsCall(ctor, call, request);
|
|
112
|
+
};
|
|
113
|
+
}
|
|
114
|
+
async function deleteStacks(...stackNames) {
|
|
115
|
+
if (stackNames.length === 0) {
|
|
116
|
+
return;
|
|
117
|
+
}
|
|
118
|
+
for (const stackName of stackNames) {
|
|
119
|
+
await exports.cloudFormation('updateTerminationProtection', {
|
|
120
|
+
EnableTerminationProtection: false,
|
|
121
|
+
StackName: stackName,
|
|
122
|
+
});
|
|
123
|
+
await exports.cloudFormation('deleteStack', {
|
|
124
|
+
StackName: stackName,
|
|
125
|
+
});
|
|
126
|
+
}
|
|
127
|
+
await retry(`Deleting ${stackNames}`, retry.forSeconds(600), async () => {
|
|
128
|
+
for (const stackName of stackNames) {
|
|
129
|
+
const status = await stackStatus(stackName);
|
|
130
|
+
if (status !== undefined && status.endsWith('_FAILED')) {
|
|
131
|
+
throw retry.abort(new Error(`'${stackName}' is in state '${status}'`));
|
|
132
|
+
}
|
|
133
|
+
if (status !== undefined) {
|
|
134
|
+
throw new Error(`Delete of '${stackName}' not complete yet`);
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
});
|
|
138
|
+
}
|
|
139
|
+
exports.deleteStacks = deleteStacks;
|
|
140
|
+
async function stackStatus(stackName) {
|
|
141
|
+
var _a;
|
|
142
|
+
try {
|
|
143
|
+
return (_a = (await exports.cloudFormation('describeStacks', { StackName: stackName })).Stacks) === null || _a === void 0 ? void 0 : _a[0].StackStatus;
|
|
144
|
+
}
|
|
145
|
+
catch (e) {
|
|
146
|
+
if (isStackMissingError(e)) {
|
|
147
|
+
return undefined;
|
|
148
|
+
}
|
|
149
|
+
throw e;
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
exports.stackStatus = stackStatus;
|
|
153
|
+
function isStackMissingError(e) {
|
|
154
|
+
return e.message.indexOf('does not exist') > -1;
|
|
155
|
+
}
|
|
156
|
+
exports.isStackMissingError = isStackMissingError;
|
|
157
|
+
function isBucketMissingError(e) {
|
|
158
|
+
return e.message.indexOf('does not exist') > -1;
|
|
159
|
+
}
|
|
160
|
+
exports.isBucketMissingError = isBucketMissingError;
|
|
161
|
+
/**
|
|
162
|
+
* Retry an async operation until a deadline is hit.
|
|
163
|
+
*
|
|
164
|
+
* Use `retry.forSeconds()` to construct a deadline relative to right now.
|
|
165
|
+
*
|
|
166
|
+
* Exceptions will cause the operation to retry. Use `retry.abort` to annotate an exception
|
|
167
|
+
* to stop the retry and end in a failure.
|
|
168
|
+
*/
|
|
169
|
+
async function retry(operation, deadline, block) {
|
|
170
|
+
let i = 0;
|
|
171
|
+
cdk_helpers_1.log(`💈 ${operation}`);
|
|
172
|
+
while (true) {
|
|
173
|
+
try {
|
|
174
|
+
i++;
|
|
175
|
+
const ret = await block();
|
|
176
|
+
cdk_helpers_1.log(`💈 ${operation}: succeeded after ${i} attempts`);
|
|
177
|
+
return ret;
|
|
178
|
+
}
|
|
179
|
+
catch (e) {
|
|
180
|
+
if (e.abort || Date.now() > deadline.getTime()) {
|
|
181
|
+
throw new Error(`${operation}: did not succeed after ${i} attempts: ${e}`);
|
|
182
|
+
}
|
|
183
|
+
cdk_helpers_1.log(`⏳ ${operation} (${e.message})`);
|
|
184
|
+
await sleep(5000);
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
exports.retry = retry;
|
|
189
|
+
/**
|
|
190
|
+
* Make a deadline for the `retry` function relative to the current time.
|
|
191
|
+
*/
|
|
192
|
+
retry.forSeconds = (seconds) => {
|
|
193
|
+
return new Date(Date.now() + seconds * 1000);
|
|
194
|
+
};
|
|
195
|
+
/**
|
|
196
|
+
* Annotate an error to stop the retrying
|
|
197
|
+
*/
|
|
198
|
+
retry.abort = (e) => {
|
|
199
|
+
e.abort = true;
|
|
200
|
+
return e;
|
|
201
|
+
};
|
|
202
|
+
async function sleep(ms) {
|
|
203
|
+
return new Promise(ok => setTimeout(ok, ms));
|
|
204
|
+
}
|
|
205
|
+
exports.sleep = sleep;
|
|
206
|
+
async function emptyBucket(bucketName) {
|
|
207
|
+
const objects = await exports.s3('listObjects', { Bucket: bucketName });
|
|
208
|
+
const deletes = (objects.Contents || []).map(obj => obj.Key || '').filter(d => !!d);
|
|
209
|
+
if (deletes.length === 0) {
|
|
210
|
+
return Promise.resolve();
|
|
211
|
+
}
|
|
212
|
+
return exports.s3('deleteObjects', {
|
|
213
|
+
Bucket: bucketName,
|
|
214
|
+
Delete: {
|
|
215
|
+
Objects: deletes.map(d => ({ Key: d })),
|
|
216
|
+
Quiet: false,
|
|
217
|
+
},
|
|
218
|
+
});
|
|
219
|
+
}
|
|
220
|
+
exports.emptyBucket = emptyBucket;
|
|
221
|
+
async function deleteImageRepository(repositoryName) {
|
|
222
|
+
await exports.ecr('deleteRepository', { repositoryName, force: true });
|
|
223
|
+
}
|
|
224
|
+
exports.deleteImageRepository = deleteImageRepository;
|
|
225
|
+
async function deleteBucket(bucketName) {
|
|
226
|
+
try {
|
|
227
|
+
await emptyBucket(bucketName);
|
|
228
|
+
await exports.s3('deleteBucket', {
|
|
229
|
+
Bucket: bucketName,
|
|
230
|
+
});
|
|
231
|
+
}
|
|
232
|
+
catch (e) {
|
|
233
|
+
if (isBucketMissingError(e)) {
|
|
234
|
+
return;
|
|
235
|
+
}
|
|
236
|
+
throw e;
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
exports.deleteBucket = deleteBucket;
|
|
240
|
+
function outputFromStack(key, stack) {
|
|
241
|
+
var _a, _b;
|
|
242
|
+
return (_b = ((_a = stack.Outputs) !== null && _a !== void 0 ? _a : []).find(o => o.OutputKey === key)) === null || _b === void 0 ? void 0 : _b.OutputValue;
|
|
243
|
+
}
|
|
244
|
+
exports.outputFromStack = outputFromStack;
|
|
245
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aws-helpers.js","sourceRoot":"","sources":["aws-helpers.ts"],"names":[],"mappings":"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","sourcesContent":["import * as AWS from 'aws-sdk';\nimport { log } from './cdk-helpers';\n\ninterface Env {\n  account: string;\n  region: string;\n}\n\nexport let testEnv = async (): Promise<Env> => {\n  const response = await new AWS.STS().getCallerIdentity().promise();\n\n  const ret: Env = {\n    account: response.Account!,\n    region: process.env.AWS_REGION ?? process.env.AWS_DEFAULT_REGION ?? 'us-east-1',\n  };\n\n  testEnv = () => Promise.resolve(ret);\n  return ret;\n};\n\nexport const cloudFormation = makeAwsCaller(AWS.CloudFormation);\nexport const s3 = makeAwsCaller(AWS.S3);\nexport const ecr = makeAwsCaller(AWS.ECR);\nexport const sns = makeAwsCaller(AWS.SNS);\nexport const iam = makeAwsCaller(AWS.IAM);\nexport const lambda = makeAwsCaller(AWS.Lambda);\nexport const sts = makeAwsCaller(AWS.STS);\n\n/**\n * Perform an AWS call from nothing\n *\n * Create the correct client, do the call and resole the promise().\n */\nasync function awsCall<\n  A extends AWS.Service,\n  B extends keyof ServiceCalls<A>,\n>(ctor: new (config: any) => A, call: B, request: First<ServiceCalls<A>[B]>): Promise<Second<ServiceCalls<A>[B]>> {\n  const env = await testEnv();\n\n  const profileName = process.env.AWS_PROFILE;\n  let creds = undefined;\n  if (process.env.CODEBUILD_BUILD_ARN && profileName) {\n\n    // in codebuild we must assume the role that the cdk uses\n    // otherwise credentials will just be picked up by the normal sdk\n    // heuristics and expire after an hour.\n\n    // can't use '~' since the SDK doesn't seem to expand it...?\n    const configPath = `${process.env.HOME}/.aws/config`;\n    const ini = new AWS.IniLoader().loadFrom({\n      filename: configPath,\n      isConfig: true,\n    });\n\n    const profile = ini[profileName];\n\n    if (!profile) {\n      throw new Error(`Profile '${profileName}' does not exist in config file (${configPath})`);\n    }\n\n    const arn = profile.role_arn;\n    const externalId = profile.external_id;\n\n    if (!arn) {\n      throw new Error(`role_arn does not exist in profile ${profileName}`);\n    }\n\n    if (!externalId) {\n      throw new Error(`external_id does not exist in profile ${externalId}`);\n    }\n\n    creds = new AWS.ChainableTemporaryCredentials({\n      params: {\n        RoleArn: arn,\n        ExternalId: externalId,\n        RoleSessionName: 'integ-tests',\n      },\n      stsConfig: {\n        region: env.region,\n      },\n      masterCredentials: new AWS.ECSCredentials(),\n    });\n\n  }\n\n  const cfn = new ctor({\n    region: env.region,\n    credentials: creds,\n    maxRetries: 6,\n    retryDelayOptions: {\n      base: 500,\n    },\n  });\n\n  const response = cfn[call](request);\n  try {\n    return await response.promise();\n  } catch (e) {\n    const newErr = new Error(`${call}(${JSON.stringify(request)}): ${e.message}`);\n    (newErr as any).code = e.code;\n    throw newErr;\n  }\n}\n\n/**\n * Factory function to invoke 'awsCall' for specific services.\n *\n * Not strictly necessary but calling this replaces a whole bunch of annoying generics you otherwise have to type:\n *\n * ```ts\n * export function cloudFormation<\n *   C extends keyof ServiceCalls<AWS.CloudFormation>,\n * >(call: C, request: First<ServiceCalls<AWS.CloudFormation>[C]>): Promise<Second<ServiceCalls<AWS.CloudFormation>[C]>> {\n *   return awsCall(AWS.CloudFormation, call, request);\n * }\n * ```\n */\nfunction makeAwsCaller<A extends AWS.Service>(ctor: new (config: any) => A) {\n  return <B extends keyof ServiceCalls<A>>(call: B, request: First<ServiceCalls<A>[B]>): Promise<Second<ServiceCalls<A>[B]>> => {\n    return awsCall(ctor, call, request);\n  };\n}\n\ntype ServiceCalls<T> = NoNayNever<SimplifiedService<T>>;\n// Map ever member in the type to the important AWS call overload, or to 'never'\ntype SimplifiedService<T> = {[k in keyof T]: AwsCallIO<T[k]>};\n// Remove all 'never' types from an object type\ntype NoNayNever<T> = Pick<T, {[k in keyof T]: T[k] extends never ? never : k }[keyof T]>;\n\n// Because of the overloads an AWS handler type looks like this:\n//\n//   {\n//      (params: INPUTSTRUCT, callback?: ((err: AWSError, data: {}) => void) | undefined): Request<OUTPUT, ...>;\n//      (callback?: ((err: AWS.AWSError, data: {}) => void) | undefined): AWS.Request<...>;\n//   }\n//\n// Get the first overload and extract the input and output struct types\ntype AwsCallIO<T> =\n  T extends {\n    (args: infer INPUT, callback?: ((err: AWS.AWSError, data: any) => void) | undefined): AWS.Request<infer OUTPUT, AWS.AWSError>;\n    (callback?: ((err: AWS.AWSError, data: {}) => void) | undefined): AWS.Request<any, any>;\n  } ? [INPUT, OUTPUT] : never;\n\ntype First<T> = T extends [any, any] ? T[0] : never;\ntype Second<T> = T extends [any, any] ? T[1] : never;\n\nexport async function deleteStacks(...stackNames: string[]) {\n  if (stackNames.length === 0) { return; }\n\n  for (const stackName of stackNames) {\n    await cloudFormation('updateTerminationProtection', {\n      EnableTerminationProtection: false,\n      StackName: stackName,\n    });\n    await cloudFormation('deleteStack', {\n      StackName: stackName,\n    });\n  }\n\n  await retry(`Deleting ${stackNames}`, retry.forSeconds(600), async () => {\n    for (const stackName of stackNames) {\n      const status = await stackStatus(stackName);\n      if (status !== undefined && status.endsWith('_FAILED')) {\n        throw retry.abort(new Error(`'${stackName}' is in state '${status}'`));\n      }\n      if (status !== undefined) {\n        throw new Error(`Delete of '${stackName}' not complete yet`);\n      }\n    }\n  });\n}\n\nexport async function stackStatus(stackName: string): Promise<string | undefined> {\n  try {\n    return (await cloudFormation('describeStacks', { StackName: stackName })).Stacks?.[0].StackStatus;\n  } catch (e) {\n    if (isStackMissingError(e)) { return undefined; }\n    throw e;\n  }\n}\n\nexport function isStackMissingError(e: Error) {\n  return e.message.indexOf('does not exist') > -1;\n}\n\nexport function isBucketMissingError(e: Error) {\n  return e.message.indexOf('does not exist') > -1;\n}\n\n/**\n * Retry an async operation until a deadline is hit.\n *\n * Use `retry.forSeconds()` to construct a deadline relative to right now.\n *\n * Exceptions will cause the operation to retry. Use `retry.abort` to annotate an exception\n * to stop the retry and end in a failure.\n */\nexport async function retry<A>(operation: string, deadline: Date, block: () => Promise<A>): Promise<A> {\n  let i = 0;\n  log(`💈 ${operation}`);\n  while (true) {\n    try {\n      i++;\n      const ret = await block();\n      log(`💈 ${operation}: succeeded after ${i} attempts`);\n      return ret;\n    } catch (e) {\n      if (e.abort || Date.now() > deadline.getTime( )) {\n        throw new Error(`${operation}: did not succeed after ${i} attempts: ${e}`);\n      }\n      log(`⏳ ${operation} (${e.message})`);\n      await sleep(5000);\n    }\n  }\n}\n\n/**\n * Make a deadline for the `retry` function relative to the current time.\n */\nretry.forSeconds = (seconds: number): Date => {\n  return new Date(Date.now() + seconds * 1000);\n};\n\n/**\n * Annotate an error to stop the retrying\n */\nretry.abort = (e: Error): Error => {\n  (e as any).abort = true;\n  return e;\n};\n\nexport async function sleep(ms: number) {\n  return new Promise(ok => setTimeout(ok, ms));\n}\n\nexport async function emptyBucket(bucketName: string) {\n  const objects = await s3('listObjects', { Bucket: bucketName });\n  const deletes = (objects.Contents || []).map(obj => obj.Key || '').filter(d => !!d);\n  if (deletes.length === 0) {\n    return Promise.resolve();\n  }\n  return s3('deleteObjects', {\n    Bucket: bucketName,\n    Delete: {\n      Objects: deletes.map(d => ({ Key: d })),\n      Quiet: false,\n    },\n  });\n}\n\nexport async function deleteImageRepository(repositoryName: string) {\n  await ecr('deleteRepository', { repositoryName, force: true });\n}\n\nexport async function deleteBucket(bucketName: string) {\n  try {\n    await emptyBucket(bucketName);\n    await s3('deleteBucket', {\n      Bucket: bucketName,\n    });\n  } catch (e) {\n    if (isBucketMissingError(e)) { return; }\n    throw e;\n  }\n}\n\nexport function outputFromStack(key: string, stack: AWS.CloudFormation.Stack): string | undefined {\n  return (stack.Outputs ?? []).find(o => o.OutputKey === key)?.OutputValue;\n}\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
We made --cloudformation-execution-policies required, old tests don't pass it yet
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
# We made --cloudformation-execution-policies required, old tests don't pass it yet
|
|
2
|
+
|
|
3
|
+
upgrade legacy bootstrap stack to new bootstrap stack while in use
|
|
4
|
+
deploy new style synthesis to new style bootstrap
|
|
5
|
+
deploy new style synthesis to new style bootstrap (with docker image)
|
|
6
|
+
switch on termination protection, switch is left alone on re-bootstrap
|
|
7
|
+
add tags, left alone on re-bootstrap
|