@aws-cdk-testing/cli-integ 2.173.4 → 3.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.eslintrc.js +9 -0
- package/LICENSE +2 -1
- package/bin/query-github +1 -1
- package/bin/query-github.js +3 -3
- package/bin/query-github.ts +56 -0
- package/bin/run-suite +1 -1
- package/bin/run-suite.js +3 -3
- package/bin/run-suite.ts +140 -0
- package/bin/stage-distribution +1 -1
- package/bin/stage-distribution.js +3 -2
- package/bin/stage-distribution.ts +267 -0
- package/bin/test-root +1 -1
- package/bin/test-root.ts +3 -0
- package/lib/aws.js +9 -6
- package/lib/aws.ts +263 -0
- package/lib/cli/query-github.d.ts +1 -0
- package/lib/cli/query-github.js +54 -0
- package/lib/cli/query-github.ts +56 -0
- package/lib/cli/run-suite.d.ts +1 -0
- package/lib/cli/run-suite.js +131 -0
- package/lib/cli/run-suite.ts +140 -0
- package/lib/cli/stage-distribution.d.ts +1 -0
- package/lib/cli/stage-distribution.js +217 -0
- package/lib/cli/stage-distribution.ts +267 -0
- package/lib/cli/test-root.d.ts +1 -0
- package/lib/cli/test-root.js +6 -0
- package/lib/cli/test-root.ts +3 -0
- package/lib/corking.ts +33 -0
- package/lib/eventually.js +3 -3
- package/lib/eventually.ts +42 -0
- package/lib/files.js +3 -2
- package/lib/files.ts +80 -0
- package/lib/github.js +6 -5
- package/lib/github.ts +43 -0
- package/lib/index.ts +13 -0
- package/lib/integ-test.ts +81 -0
- package/lib/lists.ts +9 -0
- package/lib/memoize.ts +14 -0
- package/lib/npm.ts +41 -0
- package/lib/package-sources/release-source.js +3 -2
- package/lib/package-sources/release-source.ts +81 -0
- package/lib/package-sources/repo-source.ts +111 -0
- package/lib/package-sources/repo-tools/npm.js +5 -4
- package/lib/package-sources/repo-tools/npm.ts +48 -0
- package/lib/package-sources/source.ts +35 -0
- package/lib/package-sources/subprocess.ts +15 -0
- package/lib/resource-pool.js +2 -2
- package/lib/resource-pool.ts +140 -0
- package/lib/resources.ts +4 -0
- package/lib/shell.js +8 -5
- package/lib/shell.ts +168 -0
- package/lib/staging/codeartifact.js +11 -8
- package/lib/staging/codeartifact.ts +387 -0
- package/lib/staging/maven.js +5 -3
- package/lib/staging/maven.ts +95 -0
- package/lib/staging/npm.ts +62 -0
- package/lib/staging/nuget.ts +75 -0
- package/lib/staging/parallel-shell.js +2 -2
- package/lib/staging/parallel-shell.ts +51 -0
- package/lib/staging/pypi.ts +50 -0
- package/lib/staging/usage-dir.ts +99 -0
- package/lib/with-aws.js +3 -2
- package/lib/with-aws.ts +67 -0
- package/lib/with-cdk-app.js +23 -14
- package/lib/with-cdk-app.ts +742 -0
- package/lib/with-cli-lib.ts +134 -0
- package/lib/with-packages.ts +15 -0
- package/lib/with-sam.js +7 -4
- package/lib/with-sam.ts +288 -0
- package/lib/with-temporary-directory.ts +35 -0
- package/lib/with-timeout.ts +33 -0
- package/lib/xpmutex.js +2 -2
- package/lib/xpmutex.ts +218 -0
- package/package.json +86 -62
- package/resources/cloud-assemblies/0.36.0/cdk.out +1 -0
- package/resources/cloud-assemblies/1.10.0-lookup-default-vpc/cdk.out +1 -0
- package/resources/cloud-assemblies/1.10.0-request-azs/cdk.out +1 -0
- package/tests/cli-integ-tests/bootstrapping.integtest.js +22 -13
- package/tests/cli-integ-tests/bootstrapping.integtest.ts +493 -0
- package/tests/cli-integ-tests/cli-lib.integtest.js +3 -2
- package/tests/cli-integ-tests/cli-lib.integtest.ts +90 -0
- package/tests/cli-integ-tests/cli.integtest.js +76 -49
- package/tests/cli-integ-tests/cli.integtest.ts +2874 -0
- package/tests/cli-integ-tests/garbage-collection.integtest.js +2 -2
- package/tests/cli-integ-tests/garbage-collection.integtest.ts +392 -0
- package/tests/init-csharp/init-csharp.integtest.ts +15 -0
- package/tests/init-fsharp/init-fsharp.integtest.ts +15 -0
- package/tests/init-go/init-go.integtest.ts +23 -0
- package/tests/init-java/init-java.integtest.ts +14 -0
- package/tests/init-javascript/init-javascript.integtest.ts +59 -0
- package/tests/init-python/init-python.integtest.ts +20 -0
- package/tests/init-typescript-app/init-typescript-app.integtest.ts +66 -0
- package/tests/init-typescript-lib/init-typescript-lib.integtest.ts +13 -0
- package/tests/tool-integrations/amplify.integtest.ts +43 -0
- package/tests/tool-integrations/with-tool-context.ts +14 -0
- package/tests/uberpackage/uberpackage.integtest.ts +11 -0
- package/resources/cdk-apps/cfn-include-app/.gitignore +0 -1
|
@@ -0,0 +1,267 @@
|
|
|
1
|
+
/* eslint-disable no-console */
|
|
2
|
+
import * as path from 'path';
|
|
3
|
+
import * as fs from 'fs-extra';
|
|
4
|
+
import * as glob from 'glob';
|
|
5
|
+
import * as yargs from 'yargs';
|
|
6
|
+
import { shell } from '../lib';
|
|
7
|
+
import { TestRepository } from '../lib/staging/codeartifact';
|
|
8
|
+
import { uploadJavaPackages, mavenLogin } from '../lib/staging/maven';
|
|
9
|
+
import { uploadNpmPackages, npmLogin } from '../lib/staging/npm';
|
|
10
|
+
import { uploadDotnetPackages, nugetLogin } from '../lib/staging/nuget';
|
|
11
|
+
import { uploadPythonPackages, pypiLogin } from '../lib/staging/pypi';
|
|
12
|
+
import { UsageDir } from '../lib/staging/usage-dir';
|
|
13
|
+
|
|
14
|
+
async function main() {
|
|
15
|
+
await yargs
|
|
16
|
+
.usage('$0 <command>')
|
|
17
|
+
.option('npm', {
|
|
18
|
+
description: 'Upload NPM packages only',
|
|
19
|
+
type: 'boolean',
|
|
20
|
+
requiresArg: false,
|
|
21
|
+
})
|
|
22
|
+
.option('python', {
|
|
23
|
+
description: 'Upload Python packages only',
|
|
24
|
+
type: 'boolean',
|
|
25
|
+
requiresArg: false,
|
|
26
|
+
})
|
|
27
|
+
.option('java', {
|
|
28
|
+
description: 'Upload Java packages only',
|
|
29
|
+
type: 'boolean',
|
|
30
|
+
requiresArg: false,
|
|
31
|
+
})
|
|
32
|
+
.option('dotnet', {
|
|
33
|
+
description: 'Upload Dotnet packages only',
|
|
34
|
+
type: 'boolean',
|
|
35
|
+
requiresArg: false,
|
|
36
|
+
})
|
|
37
|
+
.option('regression', {
|
|
38
|
+
description: 'Enable access to previous versions of the staged packages (this is expensive for CodeArtifact so we only do it when necessary)',
|
|
39
|
+
type: 'boolean',
|
|
40
|
+
requiresArg: false,
|
|
41
|
+
default: false,
|
|
42
|
+
})
|
|
43
|
+
.command('publish <DIRECTORY>', 'Publish a given directory', cmd => cmd
|
|
44
|
+
.positional('DIRECTORY', {
|
|
45
|
+
descripton: 'Directory distribution',
|
|
46
|
+
type: 'string',
|
|
47
|
+
demandOption: true,
|
|
48
|
+
})
|
|
49
|
+
.option('name', {
|
|
50
|
+
alias: 'n',
|
|
51
|
+
description: 'Name of the repository to create (default: generate unique name)',
|
|
52
|
+
type: 'string',
|
|
53
|
+
requiresArg: true,
|
|
54
|
+
}), async (args) => {
|
|
55
|
+
|
|
56
|
+
await validateDirectory(args);
|
|
57
|
+
const repo = await (args.name ? TestRepository.newWithName(args.name) : TestRepository.newRandom());
|
|
58
|
+
const usageDir = UsageDir.default();
|
|
59
|
+
|
|
60
|
+
await doLogin(repo, usageDir, args);
|
|
61
|
+
await publish(repo, usageDir, args);
|
|
62
|
+
|
|
63
|
+
header('Done');
|
|
64
|
+
usageDir.advertise();
|
|
65
|
+
})
|
|
66
|
+
.command('login', 'Login to a given repository', cmd => cmd
|
|
67
|
+
.option('name', {
|
|
68
|
+
alias: 'n',
|
|
69
|
+
description: 'Name of the repository to log in to',
|
|
70
|
+
type: 'string',
|
|
71
|
+
requiresArg: true,
|
|
72
|
+
demandOption: true,
|
|
73
|
+
}), async (args) => {
|
|
74
|
+
|
|
75
|
+
const repo = TestRepository.existing(args.name);
|
|
76
|
+
const usageDir = UsageDir.default();
|
|
77
|
+
|
|
78
|
+
await doLogin(repo, usageDir, args);
|
|
79
|
+
|
|
80
|
+
usageDir.advertise();
|
|
81
|
+
})
|
|
82
|
+
.command('run <DIRECTORY> <COMMAND..>', 'Publish and run a command', cmd => cmd
|
|
83
|
+
.positional('DIRECTORY', {
|
|
84
|
+
descripton: 'Directory distribution',
|
|
85
|
+
type: 'string',
|
|
86
|
+
demandOption: true,
|
|
87
|
+
})
|
|
88
|
+
.positional('COMMAND', {
|
|
89
|
+
alias: 'c',
|
|
90
|
+
description: 'Run the given command with the packages staged',
|
|
91
|
+
type: 'string',
|
|
92
|
+
array: true,
|
|
93
|
+
demandOption: true,
|
|
94
|
+
})
|
|
95
|
+
.option('cleanup', {
|
|
96
|
+
alias: 'C',
|
|
97
|
+
description: 'Cleanup the repository afterwards',
|
|
98
|
+
type: 'boolean',
|
|
99
|
+
default: true,
|
|
100
|
+
requiresArg: false,
|
|
101
|
+
}), async (args) => {
|
|
102
|
+
|
|
103
|
+
await validateDirectory(args);
|
|
104
|
+
const repo = await TestRepository.newRandom();
|
|
105
|
+
const usageDir = UsageDir.default();
|
|
106
|
+
|
|
107
|
+
await doLogin(repo, usageDir, args);
|
|
108
|
+
await publish(repo, usageDir, args);
|
|
109
|
+
|
|
110
|
+
try {
|
|
111
|
+
await usageDir.activateInCurrentProcess();
|
|
112
|
+
|
|
113
|
+
await shell(args.COMMAND ?? [], {
|
|
114
|
+
shell: true,
|
|
115
|
+
show: 'always',
|
|
116
|
+
});
|
|
117
|
+
|
|
118
|
+
} finally {
|
|
119
|
+
if (args.cleanup) {
|
|
120
|
+
await repo.delete();
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
})
|
|
124
|
+
.command('cleanup', 'Clean up testing repository', cmd => cmd
|
|
125
|
+
.option('name', {
|
|
126
|
+
alias: 'n',
|
|
127
|
+
description: 'Name of the repository to cleanup (default: most recent)',
|
|
128
|
+
type: 'string',
|
|
129
|
+
requiresArg: true,
|
|
130
|
+
}), async (args) => {
|
|
131
|
+
|
|
132
|
+
const usageDir = UsageDir.default();
|
|
133
|
+
|
|
134
|
+
let repositoryName = args.name;
|
|
135
|
+
if (!repositoryName) {
|
|
136
|
+
repositoryName = (await usageDir.currentEnv()).CODEARTIFACT_REPO;
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
if (!repositoryName) {
|
|
140
|
+
console.log(`No --name given and no $CODEARTIFACT_REPO found in ${usageDir.directory}, nothing cleaned up`);
|
|
141
|
+
return;
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
const repo = TestRepository.existing(repositoryName);
|
|
145
|
+
await repo.delete();
|
|
146
|
+
})
|
|
147
|
+
.command('gc', 'Clean up day-old testing repositories', cmd => cmd, async () => {
|
|
148
|
+
await TestRepository.gc();
|
|
149
|
+
})
|
|
150
|
+
.demandCommand(1, 'You must supply a command')
|
|
151
|
+
.help()
|
|
152
|
+
.showHelpOnFail(false)
|
|
153
|
+
.parse();
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
async function validateDirectory(args: {
|
|
157
|
+
DIRECTORY: string;
|
|
158
|
+
}) {
|
|
159
|
+
if (!await fs.pathExists(path.join(args.DIRECTORY, 'build.json'))) {
|
|
160
|
+
throw new Error(`${args.DIRECTORY} does not look like a CDK dist directory (build.json missing)`);
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
async function doLogin(repo: TestRepository, usageDir: UsageDir, args: {
|
|
165
|
+
npm?: boolean;
|
|
166
|
+
python?: boolean;
|
|
167
|
+
java?: boolean;
|
|
168
|
+
dotnet?: boolean;
|
|
169
|
+
}) {
|
|
170
|
+
const login = await repo.loginInformation();
|
|
171
|
+
|
|
172
|
+
const oldEnv = await usageDir.currentEnv();
|
|
173
|
+
|
|
174
|
+
await usageDir.clean();
|
|
175
|
+
await usageDir.addToEnv({
|
|
176
|
+
CODEARTIFACT_REPO: login.repositoryName,
|
|
177
|
+
});
|
|
178
|
+
|
|
179
|
+
if (oldEnv.BUILD_VERSION) {
|
|
180
|
+
await usageDir.addToEnv({
|
|
181
|
+
BUILD_VERSION: oldEnv.BUILD_VERSION,
|
|
182
|
+
});
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
const doRepo = whichRepos(args);
|
|
186
|
+
|
|
187
|
+
await doRepo.npm(() => npmLogin(login, usageDir));
|
|
188
|
+
await doRepo.python(() => pypiLogin(login, usageDir));
|
|
189
|
+
await doRepo.java(() => mavenLogin(login, usageDir));
|
|
190
|
+
await doRepo.dotnet(() => nugetLogin(login, usageDir));
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
async function publish(repo: TestRepository, usageDir: UsageDir, args: {
|
|
194
|
+
DIRECTORY: string;
|
|
195
|
+
npm?: boolean;
|
|
196
|
+
python?: boolean;
|
|
197
|
+
java?: boolean;
|
|
198
|
+
dotnet?: boolean;
|
|
199
|
+
regression?: boolean;
|
|
200
|
+
}) {
|
|
201
|
+
const directory = `${args.DIRECTORY}`;
|
|
202
|
+
const login = await repo.loginInformation();
|
|
203
|
+
|
|
204
|
+
const doRepo = whichRepos(args);
|
|
205
|
+
|
|
206
|
+
const buildJson = await fs.readJson(path.join(directory, 'build.json'));
|
|
207
|
+
await usageDir.addToEnv({
|
|
208
|
+
BUILD_VERSION: buildJson.version,
|
|
209
|
+
});
|
|
210
|
+
|
|
211
|
+
await doRepo.npm(async () => {
|
|
212
|
+
header('NPM');
|
|
213
|
+
await uploadNpmPackages(glob.sync(path.join(directory, 'js', '*.tgz')), login, usageDir);
|
|
214
|
+
});
|
|
215
|
+
|
|
216
|
+
await doRepo.python(async () => {
|
|
217
|
+
header('Python');
|
|
218
|
+
await uploadPythonPackages(glob.sync(path.join(directory, 'python', '*')), login);
|
|
219
|
+
});
|
|
220
|
+
|
|
221
|
+
await doRepo.java(async () => {
|
|
222
|
+
header('Java');
|
|
223
|
+
await uploadJavaPackages(glob.sync(path.join(directory, 'java', '**', '*.pom')), login, usageDir);
|
|
224
|
+
});
|
|
225
|
+
|
|
226
|
+
await doRepo.dotnet(async () => {
|
|
227
|
+
header('.NET');
|
|
228
|
+
await uploadDotnetPackages(glob.sync(path.join(directory, 'dotnet', '**', '*.nupkg')), usageDir);
|
|
229
|
+
});
|
|
230
|
+
|
|
231
|
+
if (args.regression) {
|
|
232
|
+
console.log('🛍 Configuring packages for upstream versions');
|
|
233
|
+
await repo.markAllUpstreamAllow();
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
function whichRepos(args: {
|
|
238
|
+
npm?: boolean;
|
|
239
|
+
python?: boolean;
|
|
240
|
+
java?: boolean;
|
|
241
|
+
dotnet?: boolean;
|
|
242
|
+
}) {
|
|
243
|
+
const all = args.npm === undefined && args.python === undefined && args.java === undefined && args.dotnet === undefined;
|
|
244
|
+
|
|
245
|
+
const invoke = (block: () => Promise<void>) => block();
|
|
246
|
+
const skip = () => { };
|
|
247
|
+
|
|
248
|
+
return {
|
|
249
|
+
npm: args.npm || all ? invoke : skip,
|
|
250
|
+
python: args.python || all ? invoke : skip,
|
|
251
|
+
java: args.java || all ? invoke : skip,
|
|
252
|
+
dotnet: args.dotnet || all ? invoke : skip,
|
|
253
|
+
};
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
function header(caption: string) {
|
|
257
|
+
console.log('');
|
|
258
|
+
console.log('/'.repeat(70));
|
|
259
|
+
console.log(`// ${caption}`);
|
|
260
|
+
console.log('');
|
|
261
|
+
}
|
|
262
|
+
|
|
263
|
+
main().catch(e => {
|
|
264
|
+
// eslint-disable-next-line no-console
|
|
265
|
+
console.error(e);
|
|
266
|
+
process.exitCode = 1;
|
|
267
|
+
});
|
package/bin/test-root
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
|
-
require('
|
|
2
|
+
require('../lib/cli/test-root.js');
|
package/bin/test-root.ts
ADDED
package/lib/aws.js
CHANGED
|
@@ -19,7 +19,8 @@ const credential_providers_1 = require("@aws-sdk/credential-providers");
|
|
|
19
19
|
const util_retry_1 = require("@smithy/util-retry");
|
|
20
20
|
class AwsClients {
|
|
21
21
|
static async default(output) {
|
|
22
|
-
|
|
22
|
+
var _a, _b;
|
|
23
|
+
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';
|
|
23
24
|
return AwsClients.forRegion(region, output);
|
|
24
25
|
}
|
|
25
26
|
static async forRegion(region, output) {
|
|
@@ -78,10 +79,11 @@ class AwsClients {
|
|
|
78
79
|
}
|
|
79
80
|
}
|
|
80
81
|
async stackStatus(stackName) {
|
|
82
|
+
var _a;
|
|
81
83
|
try {
|
|
82
|
-
return (await this.cloudFormation.send(new client_cloudformation_1.DescribeStacksCommand({
|
|
84
|
+
return (_a = (await this.cloudFormation.send(new client_cloudformation_1.DescribeStacksCommand({
|
|
83
85
|
StackName: stackName,
|
|
84
|
-
}))).Stacks
|
|
86
|
+
}))).Stacks) === null || _a === void 0 ? void 0 : _a[0].StackStatus;
|
|
85
87
|
}
|
|
86
88
|
catch (e) {
|
|
87
89
|
if (isStackMissingError(e)) {
|
|
@@ -112,7 +114,7 @@ class AwsClients {
|
|
|
112
114
|
Objects: deletes,
|
|
113
115
|
Quiet: false,
|
|
114
116
|
},
|
|
115
|
-
BypassGovernanceRetention: options
|
|
117
|
+
BypassGovernanceRetention: (options === null || options === void 0 ? void 0 : options.bypassGovernance) ? true : undefined,
|
|
116
118
|
}));
|
|
117
119
|
}
|
|
118
120
|
async deleteImageRepository(repositoryName) {
|
|
@@ -184,7 +186,8 @@ retry.abort = (e) => {
|
|
|
184
186
|
return e;
|
|
185
187
|
};
|
|
186
188
|
function outputFromStack(key, stack) {
|
|
187
|
-
|
|
189
|
+
var _a, _b;
|
|
190
|
+
return (_b = ((_a = stack.Outputs) !== null && _a !== void 0 ? _a : []).find((o) => o.OutputKey === key)) === null || _b === void 0 ? void 0 : _b.OutputValue;
|
|
188
191
|
}
|
|
189
192
|
async function sleep(ms) {
|
|
190
193
|
return new Promise((ok) => setTimeout(ok, ms));
|
|
@@ -200,4 +203,4 @@ function chainableCredentials(region) {
|
|
|
200
203
|
}
|
|
201
204
|
return undefined;
|
|
202
205
|
}
|
|
203
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aws.js","sourceRoot":"","sources":["aws.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  CloudFormationClient,\n  DeleteStackCommand,\n  DescribeStacksCommand,\n  UpdateTerminationProtectionCommand,\n  type Stack,\n} from '@aws-sdk/client-cloudformation';\nimport { DeleteRepositoryCommand, ECRClient } from '@aws-sdk/client-ecr';\nimport { ECSClient } from '@aws-sdk/client-ecs';\nimport { IAMClient } from '@aws-sdk/client-iam';\nimport { LambdaClient } from '@aws-sdk/client-lambda';\nimport {\n  S3Client,\n  DeleteObjectsCommand,\n  ListObjectVersionsCommand,\n  type ObjectIdentifier,\n  DeleteBucketCommand,\n} from '@aws-sdk/client-s3';\nimport { SNSClient } from '@aws-sdk/client-sns';\nimport { SSOClient } from '@aws-sdk/client-sso';\nimport { STSClient, GetCallerIdentityCommand } from '@aws-sdk/client-sts';\nimport { fromIni } from '@aws-sdk/credential-providers';\nimport type { AwsCredentialIdentityProvider } from '@smithy/types';\nimport { ConfiguredRetryStrategy } from '@smithy/util-retry';\ninterface ClientConfig {\n  readonly credentials?: AwsCredentialIdentityProvider;\n  readonly region: string;\n  readonly retryStrategy: ConfiguredRetryStrategy;\n}\n\nexport class AwsClients {\n  public static async default(output: NodeJS.WritableStream) {\n    const region = process.env.AWS_REGION ?? process.env.AWS_DEFAULT_REGION ?? 'us-east-1';\n    return AwsClients.forRegion(region, output);\n  }\n\n  public static async forRegion(region: string, output: NodeJS.WritableStream) {\n    return new AwsClients(region, output);\n  }\n\n  private readonly config: ClientConfig;\n\n  public readonly cloudFormation: CloudFormationClient;\n  public readonly s3: S3Client;\n  public readonly ecr: ECRClient;\n  public readonly ecs: ECSClient;\n  public readonly sso: SSOClient;\n  public readonly sns: SNSClient;\n  public readonly iam: IAMClient;\n  public readonly lambda: LambdaClient;\n  public readonly sts: STSClient;\n\n  constructor(public readonly region: string, private readonly output: NodeJS.WritableStream) {\n    this.config = {\n      credentials: chainableCredentials(this.region),\n      region: this.region,\n      retryStrategy: new ConfiguredRetryStrategy(9, (attempt: number) => attempt ** 500),\n    };\n    this.cloudFormation = new CloudFormationClient(this.config);\n    this.s3 = new S3Client(this.config);\n    this.ecr = new ECRClient(this.config);\n    this.ecs = new ECSClient(this.config);\n    this.sso = new SSOClient(this.config);\n    this.sns = new SNSClient(this.config);\n    this.iam = new IAMClient(this.config);\n    this.lambda = new LambdaClient(this.config);\n    this.sts = new STSClient(this.config);\n  }\n\n  public async account(): Promise<string> {\n    // Reduce # of retries, we use this as a circuit breaker for detecting no-config\n    const stsClient = new STSClient({\n      credentials: this.config.credentials,\n      region: this.config.region,\n      maxAttempts: 2,\n    });\n\n    return (await stsClient.send(new GetCallerIdentityCommand({}))).Account!;\n  }\n\n  public async deleteStacks(...stackNames: string[]) {\n    if (stackNames.length === 0) {\n      return;\n    }\n\n    // We purposely do all stacks serially, because they've been ordered\n    // to do the bootstrap stack last.\n    for (const stackName of stackNames) {\n      await this.cloudFormation.send(\n        new UpdateTerminationProtectionCommand({\n          EnableTerminationProtection: false,\n          StackName: stackName,\n        }),\n      );\n      await this.cloudFormation.send(\n        new DeleteStackCommand({\n          StackName: stackName,\n        }),\n      );\n\n      await retry(this.output, `Deleting ${stackName}`, retry.forSeconds(600), async () => {\n        const status = await this.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, status: '${status}'`);\n        }\n      });\n    }\n  }\n\n  public async stackStatus(stackName: string): Promise<string | undefined> {\n    try {\n      return (\n        await this.cloudFormation.send(\n          new DescribeStacksCommand({\n            StackName: stackName,\n          }),\n        )\n      ).Stacks?.[0].StackStatus;\n    } catch (e: any) {\n      if (isStackMissingError(e)) {\n        return undefined;\n      }\n      throw e;\n    }\n  }\n\n  public async emptyBucket(bucketName: string, options?: { bypassGovernance?: boolean }) {\n    const objects = await this.s3.send(\n      new ListObjectVersionsCommand({\n        Bucket: bucketName,\n      }),\n    );\n\n    const deletes = [...(objects.Versions || []), ...(objects.DeleteMarkers || [])].reduce((acc, obj) => {\n      if (typeof obj.VersionId !== 'undefined' && typeof obj.Key !== 'undefined') {\n        acc.push({ Key: obj.Key, VersionId: obj.VersionId });\n      } else if (typeof obj.Key !== 'undefined') {\n        acc.push({ Key: obj.Key });\n      }\n      return acc;\n    }, [] as ObjectIdentifier[]);\n\n    if (deletes.length === 0) {\n      return Promise.resolve();\n    }\n\n    return this.s3.send(\n      new DeleteObjectsCommand({\n        Bucket: bucketName,\n        Delete: {\n          Objects: deletes,\n          Quiet: false,\n        },\n        BypassGovernanceRetention: options?.bypassGovernance ? true : undefined,\n      }),\n    );\n  }\n\n  public async deleteImageRepository(repositoryName: string) {\n    await this.ecr.send(\n      new DeleteRepositoryCommand({\n        repositoryName: repositoryName,\n        force: true,\n      }),\n    );\n  }\n\n  public async deleteBucket(bucketName: string) {\n    try {\n      await this.emptyBucket(bucketName);\n\n      await this.s3.send(\n        new DeleteBucketCommand({\n          Bucket: bucketName,\n        }),\n      );\n    } catch (e: any) {\n      if (isBucketMissingError(e)) {\n        return;\n      }\n      throw e;\n    }\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>(\n  output: NodeJS.WritableStream,\n  operation: string,\n  deadline: Date,\n  block: () => Promise<A>,\n): Promise<A> {\n  let i = 0;\n  output.write(`💈 ${operation}\\n`);\n  while (true) {\n    try {\n      i++;\n      const ret = await block();\n      output.write(`💈 ${operation}: succeeded after ${i} attempts\\n`);\n      return ret;\n    } catch (e: any) {\n      if (e.abort || Date.now() > deadline.getTime()) {\n        throw new Error(`${operation}: did not succeed after ${i} attempts: ${e}`);\n      }\n      output.write(`⏳ ${operation} (${e.message})\\n`);\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 function outputFromStack(key: string, stack: Stack): string | undefined {\n  return (stack.Outputs ?? []).find((o) => o.OutputKey === key)?.OutputValue;\n}\n\nexport async function sleep(ms: number) {\n  return new Promise((ok) => setTimeout(ok, ms));\n}\n\nfunction chainableCredentials(region: string): AwsCredentialIdentityProvider | undefined {\n  if (process.env.CODEBUILD_BUILD_ARN && process.env.AWS_PROFILE) {\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    return fromIni({\n      clientConfig: { region },\n    });\n  }\n\n  return undefined;\n}\n"]}
|
|
206
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aws.js","sourceRoot":"","sources":["aws.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  CloudFormationClient,\n  DeleteStackCommand,\n  DescribeStacksCommand,\n  UpdateTerminationProtectionCommand,\n  type Stack,\n} from '@aws-sdk/client-cloudformation';\nimport { DeleteRepositoryCommand, ECRClient } from '@aws-sdk/client-ecr';\nimport { ECSClient } from '@aws-sdk/client-ecs';\nimport { IAMClient } from '@aws-sdk/client-iam';\nimport { LambdaClient } from '@aws-sdk/client-lambda';\nimport {\n  S3Client,\n  DeleteObjectsCommand,\n  ListObjectVersionsCommand,\n  type ObjectIdentifier,\n  DeleteBucketCommand,\n} from '@aws-sdk/client-s3';\nimport { SNSClient } from '@aws-sdk/client-sns';\nimport { SSOClient } from '@aws-sdk/client-sso';\nimport { STSClient, GetCallerIdentityCommand } from '@aws-sdk/client-sts';\nimport { fromIni } from '@aws-sdk/credential-providers';\nimport type { AwsCredentialIdentityProvider } from '@smithy/types';\nimport { ConfiguredRetryStrategy } from '@smithy/util-retry';\ninterface ClientConfig {\n  readonly credentials?: AwsCredentialIdentityProvider;\n  readonly region: string;\n  readonly retryStrategy: ConfiguredRetryStrategy;\n}\n\nexport class AwsClients {\n  public static async default(output: NodeJS.WritableStream) {\n    const region = process.env.AWS_REGION ?? process.env.AWS_DEFAULT_REGION ?? 'us-east-1';\n    return AwsClients.forRegion(region, output);\n  }\n\n  public static async forRegion(region: string, output: NodeJS.WritableStream) {\n    return new AwsClients(region, output);\n  }\n\n  private readonly config: ClientConfig;\n\n  public readonly cloudFormation: CloudFormationClient;\n  public readonly s3: S3Client;\n  public readonly ecr: ECRClient;\n  public readonly ecs: ECSClient;\n  public readonly sso: SSOClient;\n  public readonly sns: SNSClient;\n  public readonly iam: IAMClient;\n  public readonly lambda: LambdaClient;\n  public readonly sts: STSClient;\n\n  constructor(public readonly region: string, private readonly output: NodeJS.WritableStream) {\n    this.config = {\n      credentials: chainableCredentials(this.region),\n      region: this.region,\n      retryStrategy: new ConfiguredRetryStrategy(9, (attempt: number) => attempt ** 500),\n    };\n    this.cloudFormation = new CloudFormationClient(this.config);\n    this.s3 = new S3Client(this.config);\n    this.ecr = new ECRClient(this.config);\n    this.ecs = new ECSClient(this.config);\n    this.sso = new SSOClient(this.config);\n    this.sns = new SNSClient(this.config);\n    this.iam = new IAMClient(this.config);\n    this.lambda = new LambdaClient(this.config);\n    this.sts = new STSClient(this.config);\n  }\n\n  public async account(): Promise<string> {\n    // Reduce # of retries, we use this as a circuit breaker for detecting no-config\n    const stsClient = new STSClient({\n      credentials: this.config.credentials,\n      region: this.config.region,\n      maxAttempts: 2,\n    });\n\n    return (await stsClient.send(new GetCallerIdentityCommand({}))).Account!;\n  }\n\n  public async deleteStacks(...stackNames: string[]) {\n    if (stackNames.length === 0) {\n      return;\n    }\n\n    // We purposely do all stacks serially, because they've been ordered\n    // to do the bootstrap stack last.\n    for (const stackName of stackNames) {\n      await this.cloudFormation.send(\n        new UpdateTerminationProtectionCommand({\n          EnableTerminationProtection: false,\n          StackName: stackName,\n        }),\n      );\n      await this.cloudFormation.send(\n        new DeleteStackCommand({\n          StackName: stackName,\n        }),\n      );\n\n      await retry(this.output, `Deleting ${stackName}`, retry.forSeconds(600), async () => {\n        const status = await this.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, status: '${status}'`);\n        }\n      });\n    }\n  }\n\n  public async stackStatus(stackName: string): Promise<string | undefined> {\n    try {\n      return (\n        await this.cloudFormation.send(\n          new DescribeStacksCommand({\n            StackName: stackName,\n          }),\n        )\n      ).Stacks?.[0].StackStatus;\n    } catch (e: any) {\n      if (isStackMissingError(e)) {\n        return undefined;\n      }\n      throw e;\n    }\n  }\n\n  public async emptyBucket(bucketName: string, options?: { bypassGovernance?: boolean }) {\n    const objects = await this.s3.send(\n      new ListObjectVersionsCommand({\n        Bucket: bucketName,\n      }),\n    );\n\n    const deletes = [...(objects.Versions || []), ...(objects.DeleteMarkers || [])].reduce((acc, obj) => {\n      if (typeof obj.VersionId !== 'undefined' && typeof obj.Key !== 'undefined') {\n        acc.push({ Key: obj.Key, VersionId: obj.VersionId });\n      } else if (typeof obj.Key !== 'undefined') {\n        acc.push({ Key: obj.Key });\n      }\n      return acc;\n    }, [] as ObjectIdentifier[]);\n\n    if (deletes.length === 0) {\n      return Promise.resolve();\n    }\n\n    return this.s3.send(\n      new DeleteObjectsCommand({\n        Bucket: bucketName,\n        Delete: {\n          Objects: deletes,\n          Quiet: false,\n        },\n        BypassGovernanceRetention: options?.bypassGovernance ? true : undefined,\n      }),\n    );\n  }\n\n  public async deleteImageRepository(repositoryName: string) {\n    await this.ecr.send(\n      new DeleteRepositoryCommand({\n        repositoryName: repositoryName,\n        force: true,\n      }),\n    );\n  }\n\n  public async deleteBucket(bucketName: string) {\n    try {\n      await this.emptyBucket(bucketName);\n\n      await this.s3.send(\n        new DeleteBucketCommand({\n          Bucket: bucketName,\n        }),\n      );\n    } catch (e: any) {\n      if (isBucketMissingError(e)) {\n        return;\n      }\n      throw e;\n    }\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>(\n  output: NodeJS.WritableStream,\n  operation: string,\n  deadline: Date,\n  block: () => Promise<A>,\n): Promise<A> {\n  let i = 0;\n  output.write(`💈 ${operation}\\n`);\n  while (true) {\n    try {\n      i++;\n      const ret = await block();\n      output.write(`💈 ${operation}: succeeded after ${i} attempts\\n`);\n      return ret;\n    } catch (e: any) {\n      if (e.abort || Date.now() > deadline.getTime()) {\n        throw new Error(`${operation}: did not succeed after ${i} attempts: ${e}`);\n      }\n      output.write(`⏳ ${operation} (${e.message})\\n`);\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 function outputFromStack(key: string, stack: Stack): string | undefined {\n  return (stack.Outputs ?? []).find((o) => o.OutputKey === key)?.OutputValue;\n}\n\nexport async function sleep(ms: number) {\n  return new Promise((ok) => setTimeout(ok, ms));\n}\n\nfunction chainableCredentials(region: string): AwsCredentialIdentityProvider | undefined {\n  if (process.env.CODEBUILD_BUILD_ARN && process.env.AWS_PROFILE) {\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    return fromIni({\n      clientConfig: { region },\n    });\n  }\n\n  return undefined;\n}\n"]}
|
package/lib/aws.ts
ADDED
|
@@ -0,0 +1,263 @@
|
|
|
1
|
+
import {
|
|
2
|
+
CloudFormationClient,
|
|
3
|
+
DeleteStackCommand,
|
|
4
|
+
DescribeStacksCommand,
|
|
5
|
+
UpdateTerminationProtectionCommand,
|
|
6
|
+
type Stack,
|
|
7
|
+
} from '@aws-sdk/client-cloudformation';
|
|
8
|
+
import { DeleteRepositoryCommand, ECRClient } from '@aws-sdk/client-ecr';
|
|
9
|
+
import { ECSClient } from '@aws-sdk/client-ecs';
|
|
10
|
+
import { IAMClient } from '@aws-sdk/client-iam';
|
|
11
|
+
import { LambdaClient } from '@aws-sdk/client-lambda';
|
|
12
|
+
import {
|
|
13
|
+
S3Client,
|
|
14
|
+
DeleteObjectsCommand,
|
|
15
|
+
ListObjectVersionsCommand,
|
|
16
|
+
type ObjectIdentifier,
|
|
17
|
+
DeleteBucketCommand,
|
|
18
|
+
} from '@aws-sdk/client-s3';
|
|
19
|
+
import { SNSClient } from '@aws-sdk/client-sns';
|
|
20
|
+
import { SSOClient } from '@aws-sdk/client-sso';
|
|
21
|
+
import { STSClient, GetCallerIdentityCommand } from '@aws-sdk/client-sts';
|
|
22
|
+
import { fromIni } from '@aws-sdk/credential-providers';
|
|
23
|
+
import type { AwsCredentialIdentityProvider } from '@smithy/types';
|
|
24
|
+
import { ConfiguredRetryStrategy } from '@smithy/util-retry';
|
|
25
|
+
interface ClientConfig {
|
|
26
|
+
readonly credentials?: AwsCredentialIdentityProvider;
|
|
27
|
+
readonly region: string;
|
|
28
|
+
readonly retryStrategy: ConfiguredRetryStrategy;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
export class AwsClients {
|
|
32
|
+
public static async default(output: NodeJS.WritableStream) {
|
|
33
|
+
const region = process.env.AWS_REGION ?? process.env.AWS_DEFAULT_REGION ?? 'us-east-1';
|
|
34
|
+
return AwsClients.forRegion(region, output);
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
public static async forRegion(region: string, output: NodeJS.WritableStream) {
|
|
38
|
+
return new AwsClients(region, output);
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
private readonly config: ClientConfig;
|
|
42
|
+
|
|
43
|
+
public readonly cloudFormation: CloudFormationClient;
|
|
44
|
+
public readonly s3: S3Client;
|
|
45
|
+
public readonly ecr: ECRClient;
|
|
46
|
+
public readonly ecs: ECSClient;
|
|
47
|
+
public readonly sso: SSOClient;
|
|
48
|
+
public readonly sns: SNSClient;
|
|
49
|
+
public readonly iam: IAMClient;
|
|
50
|
+
public readonly lambda: LambdaClient;
|
|
51
|
+
public readonly sts: STSClient;
|
|
52
|
+
|
|
53
|
+
constructor(public readonly region: string, private readonly output: NodeJS.WritableStream) {
|
|
54
|
+
this.config = {
|
|
55
|
+
credentials: chainableCredentials(this.region),
|
|
56
|
+
region: this.region,
|
|
57
|
+
retryStrategy: new ConfiguredRetryStrategy(9, (attempt: number) => attempt ** 500),
|
|
58
|
+
};
|
|
59
|
+
this.cloudFormation = new CloudFormationClient(this.config);
|
|
60
|
+
this.s3 = new S3Client(this.config);
|
|
61
|
+
this.ecr = new ECRClient(this.config);
|
|
62
|
+
this.ecs = new ECSClient(this.config);
|
|
63
|
+
this.sso = new SSOClient(this.config);
|
|
64
|
+
this.sns = new SNSClient(this.config);
|
|
65
|
+
this.iam = new IAMClient(this.config);
|
|
66
|
+
this.lambda = new LambdaClient(this.config);
|
|
67
|
+
this.sts = new STSClient(this.config);
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
public async account(): Promise<string> {
|
|
71
|
+
// Reduce # of retries, we use this as a circuit breaker for detecting no-config
|
|
72
|
+
const stsClient = new STSClient({
|
|
73
|
+
credentials: this.config.credentials,
|
|
74
|
+
region: this.config.region,
|
|
75
|
+
maxAttempts: 2,
|
|
76
|
+
});
|
|
77
|
+
|
|
78
|
+
return (await stsClient.send(new GetCallerIdentityCommand({}))).Account!;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
public async deleteStacks(...stackNames: string[]) {
|
|
82
|
+
if (stackNames.length === 0) {
|
|
83
|
+
return;
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
// We purposely do all stacks serially, because they've been ordered
|
|
87
|
+
// to do the bootstrap stack last.
|
|
88
|
+
for (const stackName of stackNames) {
|
|
89
|
+
await this.cloudFormation.send(
|
|
90
|
+
new UpdateTerminationProtectionCommand({
|
|
91
|
+
EnableTerminationProtection: false,
|
|
92
|
+
StackName: stackName,
|
|
93
|
+
}),
|
|
94
|
+
);
|
|
95
|
+
await this.cloudFormation.send(
|
|
96
|
+
new DeleteStackCommand({
|
|
97
|
+
StackName: stackName,
|
|
98
|
+
}),
|
|
99
|
+
);
|
|
100
|
+
|
|
101
|
+
await retry(this.output, `Deleting ${stackName}`, retry.forSeconds(600), async () => {
|
|
102
|
+
const status = await this.stackStatus(stackName);
|
|
103
|
+
if (status !== undefined && status.endsWith('_FAILED')) {
|
|
104
|
+
throw retry.abort(new Error(`'${stackName}' is in state '${status}'`));
|
|
105
|
+
}
|
|
106
|
+
if (status !== undefined) {
|
|
107
|
+
throw new Error(`Delete of '${stackName}' not complete yet, status: '${status}'`);
|
|
108
|
+
}
|
|
109
|
+
});
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
public async stackStatus(stackName: string): Promise<string | undefined> {
|
|
114
|
+
try {
|
|
115
|
+
return (
|
|
116
|
+
await this.cloudFormation.send(
|
|
117
|
+
new DescribeStacksCommand({
|
|
118
|
+
StackName: stackName,
|
|
119
|
+
}),
|
|
120
|
+
)
|
|
121
|
+
).Stacks?.[0].StackStatus;
|
|
122
|
+
} catch (e: any) {
|
|
123
|
+
if (isStackMissingError(e)) {
|
|
124
|
+
return undefined;
|
|
125
|
+
}
|
|
126
|
+
throw e;
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
public async emptyBucket(bucketName: string, options?: { bypassGovernance?: boolean }) {
|
|
131
|
+
const objects = await this.s3.send(
|
|
132
|
+
new ListObjectVersionsCommand({
|
|
133
|
+
Bucket: bucketName,
|
|
134
|
+
}),
|
|
135
|
+
);
|
|
136
|
+
|
|
137
|
+
const deletes = [...(objects.Versions || []), ...(objects.DeleteMarkers || [])].reduce((acc, obj) => {
|
|
138
|
+
if (typeof obj.VersionId !== 'undefined' && typeof obj.Key !== 'undefined') {
|
|
139
|
+
acc.push({ Key: obj.Key, VersionId: obj.VersionId });
|
|
140
|
+
} else if (typeof obj.Key !== 'undefined') {
|
|
141
|
+
acc.push({ Key: obj.Key });
|
|
142
|
+
}
|
|
143
|
+
return acc;
|
|
144
|
+
}, [] as ObjectIdentifier[]);
|
|
145
|
+
|
|
146
|
+
if (deletes.length === 0) {
|
|
147
|
+
return Promise.resolve();
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
return this.s3.send(
|
|
151
|
+
new DeleteObjectsCommand({
|
|
152
|
+
Bucket: bucketName,
|
|
153
|
+
Delete: {
|
|
154
|
+
Objects: deletes,
|
|
155
|
+
Quiet: false,
|
|
156
|
+
},
|
|
157
|
+
BypassGovernanceRetention: options?.bypassGovernance ? true : undefined,
|
|
158
|
+
}),
|
|
159
|
+
);
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
public async deleteImageRepository(repositoryName: string) {
|
|
163
|
+
await this.ecr.send(
|
|
164
|
+
new DeleteRepositoryCommand({
|
|
165
|
+
repositoryName: repositoryName,
|
|
166
|
+
force: true,
|
|
167
|
+
}),
|
|
168
|
+
);
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
public async deleteBucket(bucketName: string) {
|
|
172
|
+
try {
|
|
173
|
+
await this.emptyBucket(bucketName);
|
|
174
|
+
|
|
175
|
+
await this.s3.send(
|
|
176
|
+
new DeleteBucketCommand({
|
|
177
|
+
Bucket: bucketName,
|
|
178
|
+
}),
|
|
179
|
+
);
|
|
180
|
+
} catch (e: any) {
|
|
181
|
+
if (isBucketMissingError(e)) {
|
|
182
|
+
return;
|
|
183
|
+
}
|
|
184
|
+
throw e;
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
export function isStackMissingError(e: Error) {
|
|
190
|
+
return e.message.indexOf('does not exist') > -1;
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
export function isBucketMissingError(e: Error) {
|
|
194
|
+
return e.message.indexOf('does not exist') > -1;
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
/**
|
|
198
|
+
* Retry an async operation until a deadline is hit.
|
|
199
|
+
*
|
|
200
|
+
* Use `retry.forSeconds()` to construct a deadline relative to right now.
|
|
201
|
+
*
|
|
202
|
+
* Exceptions will cause the operation to retry. Use `retry.abort` to annotate an exception
|
|
203
|
+
* to stop the retry and end in a failure.
|
|
204
|
+
*/
|
|
205
|
+
export async function retry<A>(
|
|
206
|
+
output: NodeJS.WritableStream,
|
|
207
|
+
operation: string,
|
|
208
|
+
deadline: Date,
|
|
209
|
+
block: () => Promise<A>,
|
|
210
|
+
): Promise<A> {
|
|
211
|
+
let i = 0;
|
|
212
|
+
output.write(`💈 ${operation}\n`);
|
|
213
|
+
while (true) {
|
|
214
|
+
try {
|
|
215
|
+
i++;
|
|
216
|
+
const ret = await block();
|
|
217
|
+
output.write(`💈 ${operation}: succeeded after ${i} attempts\n`);
|
|
218
|
+
return ret;
|
|
219
|
+
} catch (e: any) {
|
|
220
|
+
if (e.abort || Date.now() > deadline.getTime()) {
|
|
221
|
+
throw new Error(`${operation}: did not succeed after ${i} attempts: ${e}`);
|
|
222
|
+
}
|
|
223
|
+
output.write(`⏳ ${operation} (${e.message})\n`);
|
|
224
|
+
await sleep(5000);
|
|
225
|
+
}
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
/**
|
|
230
|
+
* Make a deadline for the `retry` function relative to the current time.
|
|
231
|
+
*/
|
|
232
|
+
retry.forSeconds = (seconds: number): Date => {
|
|
233
|
+
return new Date(Date.now() + seconds * 1000);
|
|
234
|
+
};
|
|
235
|
+
|
|
236
|
+
/**
|
|
237
|
+
* Annotate an error to stop the retrying
|
|
238
|
+
*/
|
|
239
|
+
retry.abort = (e: Error): Error => {
|
|
240
|
+
(e as any).abort = true;
|
|
241
|
+
return e;
|
|
242
|
+
};
|
|
243
|
+
|
|
244
|
+
export function outputFromStack(key: string, stack: Stack): string | undefined {
|
|
245
|
+
return (stack.Outputs ?? []).find((o) => o.OutputKey === key)?.OutputValue;
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
export async function sleep(ms: number) {
|
|
249
|
+
return new Promise((ok) => setTimeout(ok, ms));
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
function chainableCredentials(region: string): AwsCredentialIdentityProvider | undefined {
|
|
253
|
+
if (process.env.CODEBUILD_BUILD_ARN && process.env.AWS_PROFILE) {
|
|
254
|
+
// in codebuild we must assume the role that the cdk uses
|
|
255
|
+
// otherwise credentials will just be picked up by the normal sdk
|
|
256
|
+
// heuristics and expire after an hour.
|
|
257
|
+
return fromIni({
|
|
258
|
+
clientConfig: { region },
|
|
259
|
+
});
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
return undefined;
|
|
263
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|