@aws-cdk-testing/cli-integ 2.177.0 → 2.178.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/lib/aws.d.ts +8 -1
- package/lib/aws.js +23 -5
- package/lib/cli/run-suite.js +2 -1
- package/lib/cli/stage-distribution.js +1 -1
- package/lib/github.js +1 -2
- package/lib/integ-test.d.ts +1 -0
- package/lib/integ-test.js +39 -7
- package/lib/package-sources/release-source.js +1 -1
- package/lib/package-sources/repo-source.d.ts +4 -4
- package/lib/package-sources/repo-source.js +5 -5
- package/lib/proxy.d.ts +13 -0
- package/lib/proxy.js +49 -0
- package/lib/shell.d.ts +5 -2
- package/lib/shell.js +20 -8
- package/lib/with-aws.d.ts +3 -0
- package/lib/with-aws.js +54 -6
- package/lib/with-cdk-app.d.ts +11 -0
- package/lib/with-cdk-app.js +67 -33
- package/lib/with-cli-lib.js +3 -1
- package/lib/with-sam.js +27 -21
- package/package.json +5 -4
- package/resources/cdk-apps/app/app.js +3 -3
- package/resources/cdk-apps/rollback-test-app/app.js +1 -1
- package/resources/cdk-apps/sam_cdk_integ_app/lib/test-stack.js +3 -1
- package/resources/cli-regression-patches/v1.130.0/bootstrapping.integtest.js +1 -1
- package/resources/integ.jest.config.js +4 -1
- package/tests/cli-integ-tests/bootstrapping.integtest.js +1 -1
- package/tests/cli-integ-tests/cli.integtest.js +13 -37
- package/tests/cli-integ-tests/proxy.integtest.d.ts +1 -0
- package/tests/cli-integ-tests/proxy.integtest.js +146 -0
- package/tests/init-go/init-go.integtest.js +6 -2
- package/tests/init-javascript/init-javascript.integtest.js +1 -1
- package/tests/init-typescript-app/init-typescript-app.integtest.js +3 -3
package/lib/proxy.js
ADDED
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.startProxyServer = startProxyServer;
|
|
4
|
+
exports.awsActionsFromRequests = awsActionsFromRequests;
|
|
5
|
+
const fs_1 = require("fs");
|
|
6
|
+
const querystring = require("node:querystring");
|
|
7
|
+
const os = require("os");
|
|
8
|
+
const path = require("path");
|
|
9
|
+
const mockttp = require("mockttp");
|
|
10
|
+
async function startProxyServer(certDirRoot) {
|
|
11
|
+
const certDir = await fs_1.promises.mkdtemp(path.join(certDirRoot ?? os.tmpdir(), 'cdk-'));
|
|
12
|
+
const certPath = path.join(certDir, 'cert.pem');
|
|
13
|
+
const keyPath = path.join(certDir, 'key.pem');
|
|
14
|
+
// Set up key and certificate
|
|
15
|
+
const { key, cert } = await mockttp.generateCACertificate();
|
|
16
|
+
await fs_1.promises.writeFile(keyPath, key);
|
|
17
|
+
await fs_1.promises.writeFile(certPath, cert);
|
|
18
|
+
const server = mockttp.getLocal({
|
|
19
|
+
https: { keyPath: keyPath, certPath: certPath },
|
|
20
|
+
});
|
|
21
|
+
// We don't need to modify any request, so the proxy
|
|
22
|
+
// passes through all requests to the target host.
|
|
23
|
+
const endpoint = await server
|
|
24
|
+
.forAnyRequest()
|
|
25
|
+
.thenPassThrough();
|
|
26
|
+
const port = 9000 + Math.floor(Math.random() * 10000);
|
|
27
|
+
// server.enableDebug();
|
|
28
|
+
await server.start(port);
|
|
29
|
+
return {
|
|
30
|
+
certPath,
|
|
31
|
+
keyPath,
|
|
32
|
+
server,
|
|
33
|
+
url: server.url,
|
|
34
|
+
port: server.port,
|
|
35
|
+
getSeenRequests: () => endpoint.getSeenRequests(),
|
|
36
|
+
async stop() {
|
|
37
|
+
await server.stop();
|
|
38
|
+
await fs_1.promises.rm(certDir, { recursive: true, force: true });
|
|
39
|
+
},
|
|
40
|
+
};
|
|
41
|
+
}
|
|
42
|
+
function awsActionsFromRequests(requests) {
|
|
43
|
+
return [...new Set(requests
|
|
44
|
+
.map(req => req.body.buffer.toString('utf-8'))
|
|
45
|
+
.map(body => querystring.decode(body))
|
|
46
|
+
.map(x => x.Action)
|
|
47
|
+
.filter(action => action != null))];
|
|
48
|
+
}
|
|
49
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJveHkuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJwcm94eS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQU9BLDRDQXFDQztBQWFELHdEQU1DO0FBL0RELDJCQUFvQztBQUNwQyxnREFBZ0Q7QUFDaEQseUJBQXlCO0FBQ3pCLDZCQUE2QjtBQUM3QixtQ0FBbUM7QUFHNUIsS0FBSyxVQUFVLGdCQUFnQixDQUFDLFdBQW9CO0lBQ3pELE1BQU0sT0FBTyxHQUFHLE1BQU0sYUFBRSxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLFdBQVcsSUFBSSxFQUFFLENBQUMsTUFBTSxFQUFFLEVBQUUsTUFBTSxDQUFDLENBQUMsQ0FBQztJQUNoRixNQUFNLFFBQVEsR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFBRSxVQUFVLENBQUMsQ0FBQztJQUNoRCxNQUFNLE9BQU8sR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFBRSxTQUFTLENBQUMsQ0FBQztJQUU5Qyw2QkFBNkI7SUFDN0IsTUFBTSxFQUFFLEdBQUcsRUFBRSxJQUFJLEVBQUUsR0FBRyxNQUFNLE9BQU8sQ0FBQyxxQkFBcUIsRUFBRSxDQUFDO0lBQzVELE1BQU0sYUFBRSxDQUFDLFNBQVMsQ0FBQyxPQUFPLEVBQUUsR0FBRyxDQUFDLENBQUM7SUFDakMsTUFBTSxhQUFFLENBQUMsU0FBUyxDQUFDLFFBQVEsRUFBRSxJQUFJLENBQUMsQ0FBQztJQUVuQyxNQUFNLE1BQU0sR0FBRyxPQUFPLENBQUMsUUFBUSxDQUFDO1FBQzlCLEtBQUssRUFBRSxFQUFFLE9BQU8sRUFBRSxPQUFPLEVBQUUsUUFBUSxFQUFFLFFBQVEsRUFBRTtLQUNoRCxDQUFDLENBQUM7SUFFSCxvREFBb0Q7SUFDcEQsa0RBQWtEO0lBQ2xELE1BQU0sUUFBUSxHQUFHLE1BQU0sTUFBTTtTQUMxQixhQUFhLEVBQUU7U0FDZixlQUFlLEVBQUUsQ0FBQztJQUVyQixNQUFNLElBQUksR0FBRyxJQUFJLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLEdBQUcsS0FBSyxDQUFDLENBQUM7SUFFdEQsd0JBQXdCO0lBQ3hCLE1BQU0sTUFBTSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUV6QixPQUFPO1FBQ0wsUUFBUTtRQUNSLE9BQU87UUFDUCxNQUFNO1FBQ04sR0FBRyxFQUFFLE1BQU0sQ0FBQyxHQUFHO1FBQ2YsSUFBSSxFQUFFLE1BQU0sQ0FBQyxJQUFJO1FBQ2pCLGVBQWUsRUFBRSxHQUFHLEVBQUUsQ0FBQyxRQUFRLENBQUMsZUFBZSxFQUFFO1FBQ2pELEtBQUssQ0FBQyxJQUFJO1lBQ1IsTUFBTSxNQUFNLENBQUMsSUFBSSxFQUFFLENBQUM7WUFDcEIsTUFBTSxhQUFFLENBQUMsRUFBRSxDQUFDLE9BQU8sRUFBRSxFQUFFLFNBQVMsRUFBRSxJQUFJLEVBQUUsS0FBSyxFQUFFLElBQUksRUFBRSxDQUFDLENBQUM7UUFDekQsQ0FBQztLQUNGLENBQUM7QUFDSixDQUFDO0FBYUQsU0FBZ0Isc0JBQXNCLENBQUMsUUFBNEI7SUFDakUsT0FBTyxDQUFDLEdBQUcsSUFBSSxHQUFHLENBQUMsUUFBUTthQUN4QixHQUFHLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxRQUFRLENBQUMsT0FBTyxDQUFDLENBQUM7YUFDN0MsR0FBRyxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsV0FBVyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsQ0FBQzthQUNyQyxHQUFHLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUMsTUFBZ0IsQ0FBQzthQUM1QixNQUFNLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQyxNQUFNLElBQUksSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDO0FBQ3hDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBwcm9taXNlcyBhcyBmcyB9IGZyb20gJ2ZzJztcbmltcG9ydCAqIGFzIHF1ZXJ5c3RyaW5nIGZyb20gJ25vZGU6cXVlcnlzdHJpbmcnO1xuaW1wb3J0ICogYXMgb3MgZnJvbSAnb3MnO1xuaW1wb3J0ICogYXMgcGF0aCBmcm9tICdwYXRoJztcbmltcG9ydCAqIGFzIG1vY2t0dHAgZnJvbSAnbW9ja3R0cCc7XG5pbXBvcnQgeyBDb21wbGV0ZWRSZXF1ZXN0IH0gZnJvbSAnbW9ja3R0cCc7XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBzdGFydFByb3h5U2VydmVyKGNlcnREaXJSb290Pzogc3RyaW5nKTogUHJvbWlzZTxQcm94eVNlcnZlcj4ge1xuICBjb25zdCBjZXJ0RGlyID0gYXdhaXQgZnMubWtkdGVtcChwYXRoLmpvaW4oY2VydERpclJvb3QgPz8gb3MudG1wZGlyKCksICdjZGstJykpO1xuICBjb25zdCBjZXJ0UGF0aCA9IHBhdGguam9pbihjZXJ0RGlyLCAnY2VydC5wZW0nKTtcbiAgY29uc3Qga2V5UGF0aCA9IHBhdGguam9pbihjZXJ0RGlyLCAna2V5LnBlbScpO1xuXG4gIC8vIFNldCB1cCBrZXkgYW5kIGNlcnRpZmljYXRlXG4gIGNvbnN0IHsga2V5LCBjZXJ0IH0gPSBhd2FpdCBtb2NrdHRwLmdlbmVyYXRlQ0FDZXJ0aWZpY2F0ZSgpO1xuICBhd2FpdCBmcy53cml0ZUZpbGUoa2V5UGF0aCwga2V5KTtcbiAgYXdhaXQgZnMud3JpdGVGaWxlKGNlcnRQYXRoLCBjZXJ0KTtcblxuICBjb25zdCBzZXJ2ZXIgPSBtb2NrdHRwLmdldExvY2FsKHtcbiAgICBodHRwczogeyBrZXlQYXRoOiBrZXlQYXRoLCBjZXJ0UGF0aDogY2VydFBhdGggfSxcbiAgfSk7XG5cbiAgLy8gV2UgZG9uJ3QgbmVlZCB0byBtb2RpZnkgYW55IHJlcXVlc3QsIHNvIHRoZSBwcm94eVxuICAvLyBwYXNzZXMgdGhyb3VnaCBhbGwgcmVxdWVzdHMgdG8gdGhlIHRhcmdldCBob3N0LlxuICBjb25zdCBlbmRwb2ludCA9IGF3YWl0IHNlcnZlclxuICAgIC5mb3JBbnlSZXF1ZXN0KClcbiAgICAudGhlblBhc3NUaHJvdWdoKCk7XG5cbiAgY29uc3QgcG9ydCA9IDkwMDAgKyBNYXRoLmZsb29yKE1hdGgucmFuZG9tKCkgKiAxMDAwMCk7XG5cbiAgLy8gc2VydmVyLmVuYWJsZURlYnVnKCk7XG4gIGF3YWl0IHNlcnZlci5zdGFydChwb3J0KTtcblxuICByZXR1cm4ge1xuICAgIGNlcnRQYXRoLFxuICAgIGtleVBhdGgsXG4gICAgc2VydmVyLFxuICAgIHVybDogc2VydmVyLnVybCxcbiAgICBwb3J0OiBzZXJ2ZXIucG9ydCxcbiAgICBnZXRTZWVuUmVxdWVzdHM6ICgpID0+IGVuZHBvaW50LmdldFNlZW5SZXF1ZXN0cygpLFxuICAgIGFzeW5jIHN0b3AoKSB7XG4gICAgICBhd2FpdCBzZXJ2ZXIuc3RvcCgpO1xuICAgICAgYXdhaXQgZnMucm0oY2VydERpciwgeyByZWN1cnNpdmU6IHRydWUsIGZvcmNlOiB0cnVlIH0pO1xuICAgIH0sXG4gIH07XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUHJveHlTZXJ2ZXIge1xuICByZWFkb25seSBjZXJ0UGF0aDogc3RyaW5nO1xuICByZWFkb25seSBrZXlQYXRoOiBzdHJpbmc7XG4gIHJlYWRvbmx5IHNlcnZlcjogbW9ja3R0cC5Nb2NrdHRwO1xuICByZWFkb25seSB1cmw6IHN0cmluZztcbiAgcmVhZG9ubHkgcG9ydDogbnVtYmVyO1xuXG4gIGdldFNlZW5SZXF1ZXN0cygpOiBQcm9taXNlPENvbXBsZXRlZFJlcXVlc3RbXT47XG4gIHN0b3AoKTogUHJvbWlzZTx2b2lkPjtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGF3c0FjdGlvbnNGcm9tUmVxdWVzdHMocmVxdWVzdHM6IENvbXBsZXRlZFJlcXVlc3RbXSk6IHN0cmluZ1tdIHtcbiAgcmV0dXJuIFsuLi5uZXcgU2V0KHJlcXVlc3RzXG4gICAgLm1hcChyZXEgPT4gcmVxLmJvZHkuYnVmZmVyLnRvU3RyaW5nKCd1dGYtOCcpKVxuICAgIC5tYXAoYm9keSA9PiBxdWVyeXN0cmluZy5kZWNvZGUoYm9keSkpXG4gICAgLm1hcCh4ID0+IHguQWN0aW9uIGFzIHN0cmluZylcbiAgICAuZmlsdGVyKGFjdGlvbiA9PiBhY3Rpb24gIT0gbnVsbCkpXTtcbn1cbiJdfQ==
|
package/lib/shell.d.ts
CHANGED
|
@@ -11,7 +11,7 @@ export interface ShellOptions extends child_process.SpawnOptions {
|
|
|
11
11
|
/**
|
|
12
12
|
* Properties to add to 'env'
|
|
13
13
|
*/
|
|
14
|
-
readonly modEnv?: Record<string, string>;
|
|
14
|
+
readonly modEnv?: Record<string, string | undefined>;
|
|
15
15
|
/**
|
|
16
16
|
* Don't fail when exiting with an error
|
|
17
17
|
*
|
|
@@ -51,6 +51,9 @@ export declare class ShellHelper {
|
|
|
51
51
|
}
|
|
52
52
|
/**
|
|
53
53
|
* rm -rf reimplementation, don't want to depend on an NPM package for this
|
|
54
|
+
*
|
|
55
|
+
* Returns `true` if everything got deleted, or `false` if some files could
|
|
56
|
+
* not be deleted due to permissions issues.
|
|
54
57
|
*/
|
|
55
|
-
export declare function rimraf(fsPath: string):
|
|
58
|
+
export declare function rimraf(fsPath: string): boolean;
|
|
56
59
|
export declare function addToShellPath(x: string): void;
|
package/lib/shell.js
CHANGED
|
@@ -22,12 +22,13 @@ async function shell(command, options = {}) {
|
|
|
22
22
|
outputStream.write(x);
|
|
23
23
|
}
|
|
24
24
|
};
|
|
25
|
-
// Always output the command
|
|
26
|
-
writeToOutputs(`💻 ${command.join(' ')}\n`);
|
|
27
25
|
const show = options.show ?? 'always';
|
|
28
|
-
|
|
26
|
+
const verbose = Boolean(process.env.VERBOSE);
|
|
27
|
+
if (verbose) {
|
|
29
28
|
outputs.add(process.stdout);
|
|
30
29
|
}
|
|
30
|
+
// Always output the command
|
|
31
|
+
writeToOutputs(`💻 ${command.join(' ')} (show: ${show}, verbose: ${verbose})\n`);
|
|
31
32
|
const env = options.env ?? (options.modEnv ? { ...process.env, ...options.modEnv } : process.env);
|
|
32
33
|
const child = child_process.spawn(command[0], command.slice(1), {
|
|
33
34
|
...options,
|
|
@@ -89,25 +90,36 @@ class ShellHelper {
|
|
|
89
90
|
exports.ShellHelper = ShellHelper;
|
|
90
91
|
/**
|
|
91
92
|
* rm -rf reimplementation, don't want to depend on an NPM package for this
|
|
93
|
+
*
|
|
94
|
+
* Returns `true` if everything got deleted, or `false` if some files could
|
|
95
|
+
* not be deleted due to permissions issues.
|
|
92
96
|
*/
|
|
93
97
|
function rimraf(fsPath) {
|
|
94
98
|
try {
|
|
99
|
+
let success = true;
|
|
95
100
|
const isDir = fs.lstatSync(fsPath).isDirectory();
|
|
96
101
|
if (isDir) {
|
|
97
102
|
for (const file of fs.readdirSync(fsPath)) {
|
|
98
|
-
rimraf(path.join(fsPath, file));
|
|
103
|
+
success && (success = rimraf(path.join(fsPath, file)));
|
|
99
104
|
}
|
|
100
105
|
fs.rmdirSync(fsPath);
|
|
101
106
|
}
|
|
102
107
|
else {
|
|
103
108
|
fs.unlinkSync(fsPath);
|
|
104
109
|
}
|
|
110
|
+
return success;
|
|
105
111
|
}
|
|
106
112
|
catch (e) {
|
|
107
|
-
//
|
|
108
|
-
|
|
109
|
-
|
|
113
|
+
// Can happen if some files got generated inside a Docker container and are now inadvertently owned by `root`.
|
|
114
|
+
// We can't ever clean those up anymore, but since it only happens inside GitHub Actions containers we also don't care too much.
|
|
115
|
+
if (e.code === 'EACCES' || e.code === 'ENOTEMPTY') {
|
|
116
|
+
return false;
|
|
117
|
+
}
|
|
118
|
+
// Already gone
|
|
119
|
+
if (e.code === 'ENOENT') {
|
|
120
|
+
return true;
|
|
110
121
|
}
|
|
122
|
+
throw e;
|
|
111
123
|
}
|
|
112
124
|
}
|
|
113
125
|
function addToShellPath(x) {
|
|
@@ -117,4 +129,4 @@ function addToShellPath(x) {
|
|
|
117
129
|
}
|
|
118
130
|
process.env.PATH = parts.join(':');
|
|
119
131
|
}
|
|
120
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"shell.js","sourceRoot":"","sources":["shell.ts"],"names":[],"mappings":"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","sourcesContent":["import * as child_process from 'child_process';\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport { TestContext } from './integ-test';\nimport { TemporaryDirectoryContext } from './with-temporary-directory';\n\n/**\n * A shell command that does what you want\n *\n * Is platform-aware, handles errors nicely.\n */\nexport async function shell(command: string[], options: ShellOptions = {}): Promise<string> {\n  if (options.modEnv && options.env) {\n    throw new Error('Use either env or modEnv but not both');\n  }\n\n  const outputs = new Set(options.outputs);\n  const writeToOutputs = (x: string) => {\n    for (const outputStream of outputs) {\n      outputStream.write(x);\n    }\n  };\n\n  // Always output the command\n  writeToOutputs(`💻 ${command.join(' ')}\\n`);\n  const show = options.show ?? 'always';\n\n  if (process.env.VERBOSE) {\n    outputs.add(process.stdout);\n  }\n\n  const env = options.env ?? (options.modEnv ? { ...process.env, ...options.modEnv } : process.env);\n\n  const child = child_process.spawn(command[0], command.slice(1), {\n    ...options,\n    env,\n    // Need this for Windows where we want .cmd and .bat to be found as well.\n    shell: true,\n    stdio: ['ignore', 'pipe', 'pipe'],\n  });\n\n  return new Promise<string>((resolve, reject) => {\n    const stdout = new Array<Buffer>();\n    const stderr = new Array<Buffer>();\n\n    child.stdout!.on('data', chunk => {\n      if (show === 'always') {\n        writeToOutputs(chunk);\n      }\n      stdout.push(chunk);\n    });\n\n    child.stderr!.on('data', chunk => {\n      if (show === 'always') {\n        writeToOutputs(chunk);\n      }\n      if (options.captureStderr ?? true) {\n        stderr.push(chunk);\n      }\n    });\n\n    child.once('error', reject);\n\n    child.once('close', code => {\n      const stderrOutput = Buffer.concat(stderr).toString('utf-8');\n      const stdoutOutput = Buffer.concat(stdout).toString('utf-8');\n      const out = (options.onlyStderr ? stderrOutput : stdoutOutput + stderrOutput).trim();\n      if (code === 0 || options.allowErrExit) {\n        resolve(out);\n      } else {\n        if (show === 'error') {\n          writeToOutputs(`${out}\\n`);\n        }\n        reject(new Error(`'${command.join(' ')}' exited with error code ${code}.`));\n      }\n    });\n  });\n}\n\nexport interface ShellOptions extends child_process.SpawnOptions {\n  /**\n   * Properties to add to 'env'\n   */\n  readonly modEnv?: Record<string, string>;\n\n  /**\n   * Don't fail when exiting with an error\n   *\n   * @default false\n   */\n  readonly allowErrExit?: boolean;\n\n  /**\n   * Whether to capture stderr\n   *\n   * @default true\n   */\n  readonly captureStderr?: boolean;\n\n  /**\n   * Pass output here\n   */\n  readonly outputs?: NodeJS.WritableStream[];\n\n  /**\n   * Only return stderr. For example, this is used to validate\n   * that when CI=true, all logs are sent to stdout.\n   *\n   * @default false\n   */\n  readonly onlyStderr?: boolean;\n\n  /**\n   * Don't log to stdout\n   *\n   * @default always\n   */\n  readonly show?: 'always' | 'never' | 'error';\n}\n\nexport class ShellHelper {\n  public static fromContext(context: TestContext & TemporaryDirectoryContext) {\n    return new ShellHelper(context.integTestDir, context.output);\n  }\n\n  constructor(\n    private readonly _cwd: string,\n    private readonly _output: NodeJS.WritableStream) { }\n\n  public async shell(command: string[], options: Omit<ShellOptions, 'cwd' | 'outputs'> = {}): Promise<string> {\n    return shell(command, {\n      outputs: [this._output],\n      cwd: this._cwd,\n      ...options,\n    });\n  }\n}\n\n/**\n * rm -rf reimplementation, don't want to depend on an NPM package for this\n */\nexport function rimraf(fsPath: string) {\n  try {\n    const isDir = fs.lstatSync(fsPath).isDirectory();\n\n    if (isDir) {\n      for (const file of fs.readdirSync(fsPath)) {\n        rimraf(path.join(fsPath, file));\n      }\n      fs.rmdirSync(fsPath);\n    } else {\n      fs.unlinkSync(fsPath);\n    }\n  } catch (e: any) {\n    // We will survive ENOENT\n    if (e.code !== 'ENOENT') { throw e; }\n  }\n}\n\nexport function addToShellPath(x: string) {\n  const parts = process.env.PATH?.split(':') ?? [];\n\n  if (!parts.includes(x)) {\n    parts.unshift(x);\n  }\n\n  process.env.PATH = parts.join(':');\n}\n"]}
|
|
132
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"shell.js","sourceRoot":"","sources":["shell.ts"],"names":[],"mappings":";;;AAWA,sBAoEC;AAmED,wBAwBC;AAED,wCAQC;AApLD,+CAA+C;AAC/C,yBAAyB;AACzB,6BAA6B;AAI7B;;;;GAIG;AACI,KAAK,UAAU,KAAK,CAAC,OAAiB,EAAE,UAAwB,EAAE;IACvE,IAAI,OAAO,CAAC,MAAM,IAAI,OAAO,CAAC,GAAG,EAAE,CAAC;QAClC,MAAM,IAAI,KAAK,CAAC,uCAAuC,CAAC,CAAC;IAC3D,CAAC;IAED,MAAM,OAAO,GAAG,IAAI,GAAG,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;IACzC,MAAM,cAAc,GAAG,CAAC,CAAS,EAAE,EAAE;QACnC,KAAK,MAAM,YAAY,IAAI,OAAO,EAAE,CAAC;YACnC,YAAY,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QACxB,CAAC;IACH,CAAC,CAAC;IAEF,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,IAAI,QAAQ,CAAC;IACtC,MAAM,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;IAE7C,IAAI,OAAO,EAAE,CAAC;QACZ,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;IAC9B,CAAC;IAED,4BAA4B;IAC5B,cAAc,CAAC,MAAM,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,WAAW,IAAI,cAAc,OAAO,KAAK,CAAC,CAAC;IAEjF,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,GAAG,OAAO,CAAC,GAAG,EAAE,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;IAElG,MAAM,KAAK,GAAG,aAAa,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE;QAC9D,GAAG,OAAO;QACV,GAAG;QACH,yEAAyE;QACzE,KAAK,EAAE,IAAI;QACX,KAAK,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,MAAM,CAAC;KAClC,CAAC,CAAC;IAEH,OAAO,IAAI,OAAO,CAAS,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;QAC7C,MAAM,MAAM,GAAG,IAAI,KAAK,EAAU,CAAC;QACnC,MAAM,MAAM,GAAG,IAAI,KAAK,EAAU,CAAC;QAEnC,KAAK,CAAC,MAAO,CAAC,EAAE,CAAC,MAAM,EAAE,KAAK,CAAC,EAAE;YAC/B,IAAI,IAAI,KAAK,QAAQ,EAAE,CAAC;gBACtB,cAAc,CAAC,KAAK,CAAC,CAAC;YACxB,CAAC;YACD,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACrB,CAAC,CAAC,CAAC;QAEH,KAAK,CAAC,MAAO,CAAC,EAAE,CAAC,MAAM,EAAE,KAAK,CAAC,EAAE;YAC/B,IAAI,IAAI,KAAK,QAAQ,EAAE,CAAC;gBACtB,cAAc,CAAC,KAAK,CAAC,CAAC;YACxB,CAAC;YACD,IAAI,OAAO,CAAC,aAAa,IAAI,IAAI,EAAE,CAAC;gBAClC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YACrB,CAAC;QACH,CAAC,CAAC,CAAC;QAEH,KAAK,CAAC,IAAI,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;QAE5B,KAAK,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC,EAAE;YACzB,MAAM,YAAY,GAAG,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;YAC7D,MAAM,YAAY,GAAG,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;YAC7D,MAAM,GAAG,GAAG,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,YAAY,GAAG,YAAY,CAAC,CAAC,IAAI,EAAE,CAAC;YACrF,IAAI,IAAI,KAAK,CAAC,IAAI,OAAO,CAAC,YAAY,EAAE,CAAC;gBACvC,OAAO,CAAC,GAAG,CAAC,CAAC;YACf,CAAC;iBAAM,CAAC;gBACN,IAAI,IAAI,KAAK,OAAO,EAAE,CAAC;oBACrB,cAAc,CAAC,GAAG,GAAG,IAAI,CAAC,CAAC;gBAC7B,CAAC;gBACD,MAAM,CAAC,IAAI,KAAK,CAAC,IAAI,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,4BAA4B,IAAI,GAAG,CAAC,CAAC,CAAC;YAC9E,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC;AA2CD,MAAa,WAAW;IACf,MAAM,CAAC,WAAW,CAAC,OAAgD;QACxE,OAAO,IAAI,WAAW,CAAC,OAAO,CAAC,YAAY,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC;IAC/D,CAAC;IAED,YACmB,IAAY,EACZ,OAA8B;QAD9B,SAAI,GAAJ,IAAI,CAAQ;QACZ,YAAO,GAAP,OAAO,CAAuB;IAAI,CAAC;IAE/C,KAAK,CAAC,KAAK,CAAC,OAAiB,EAAE,UAAiD,EAAE;QACvF,OAAO,KAAK,CAAC,OAAO,EAAE;YACpB,OAAO,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC;YACvB,GAAG,EAAE,IAAI,CAAC,IAAI;YACd,GAAG,OAAO;SACX,CAAC,CAAC;IACL,CAAC;CACF;AAhBD,kCAgBC;AAED;;;;;GAKG;AACH,SAAgB,MAAM,CAAC,MAAc;IACnC,IAAI,CAAC;QACH,IAAI,OAAO,GAAG,IAAI,CAAC;QACnB,MAAM,KAAK,GAAG,EAAE,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,WAAW,EAAE,CAAC;QAEjD,IAAI,KAAK,EAAE,CAAC;YACV,KAAK,MAAM,IAAI,IAAI,EAAE,CAAC,WAAW,CAAC,MAAM,CAAC,EAAE,CAAC;gBAC1C,OAAO,KAAP,OAAO,GAAK,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC,EAAC;YAC9C,CAAC;YACD,EAAE,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;QACvB,CAAC;aAAM,CAAC;YACN,EAAE,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;QACxB,CAAC;QACD,OAAO,OAAO,CAAC;IACjB,CAAC;IAAC,OAAO,CAAM,EAAE,CAAC;QAChB,8GAA8G;QAC9G,gIAAgI;QAChI,IAAI,CAAC,CAAC,IAAI,KAAK,QAAQ,IAAI,CAAC,CAAC,IAAI,KAAK,WAAW,EAAE,CAAC;YAAC,OAAO,KAAK,CAAC;QAAC,CAAC;QAEpE,eAAe;QACf,IAAI,CAAC,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;YAAC,OAAO,IAAI,CAAC;QAAC,CAAC;QAEzC,MAAM,CAAC,CAAC;IACV,CAAC;AACH,CAAC;AAED,SAAgB,cAAc,CAAC,CAAS;IACtC,MAAM,KAAK,GAAG,OAAO,CAAC,GAAG,CAAC,IAAI,EAAE,KAAK,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC;IAEjD,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAE,CAAC;QACvB,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;IACnB,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,IAAI,GAAG,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;AACrC,CAAC","sourcesContent":["import * as child_process from 'child_process';\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport { TestContext } from './integ-test';\nimport { TemporaryDirectoryContext } from './with-temporary-directory';\n\n/**\n * A shell command that does what you want\n *\n * Is platform-aware, handles errors nicely.\n */\nexport async function shell(command: string[], options: ShellOptions = {}): Promise<string> {\n  if (options.modEnv && options.env) {\n    throw new Error('Use either env or modEnv but not both');\n  }\n\n  const outputs = new Set(options.outputs);\n  const writeToOutputs = (x: string) => {\n    for (const outputStream of outputs) {\n      outputStream.write(x);\n    }\n  };\n\n  const show = options.show ?? 'always';\n  const verbose = Boolean(process.env.VERBOSE);\n\n  if (verbose) {\n    outputs.add(process.stdout);\n  }\n\n  // Always output the command\n  writeToOutputs(`💻 ${command.join(' ')} (show: ${show}, verbose: ${verbose})\\n`);\n\n  const env = options.env ?? (options.modEnv ? { ...process.env, ...options.modEnv } : process.env);\n\n  const child = child_process.spawn(command[0], command.slice(1), {\n    ...options,\n    env,\n    // Need this for Windows where we want .cmd and .bat to be found as well.\n    shell: true,\n    stdio: ['ignore', 'pipe', 'pipe'],\n  });\n\n  return new Promise<string>((resolve, reject) => {\n    const stdout = new Array<Buffer>();\n    const stderr = new Array<Buffer>();\n\n    child.stdout!.on('data', chunk => {\n      if (show === 'always') {\n        writeToOutputs(chunk);\n      }\n      stdout.push(chunk);\n    });\n\n    child.stderr!.on('data', chunk => {\n      if (show === 'always') {\n        writeToOutputs(chunk);\n      }\n      if (options.captureStderr ?? true) {\n        stderr.push(chunk);\n      }\n    });\n\n    child.once('error', reject);\n\n    child.once('close', code => {\n      const stderrOutput = Buffer.concat(stderr).toString('utf-8');\n      const stdoutOutput = Buffer.concat(stdout).toString('utf-8');\n      const out = (options.onlyStderr ? stderrOutput : stdoutOutput + stderrOutput).trim();\n      if (code === 0 || options.allowErrExit) {\n        resolve(out);\n      } else {\n        if (show === 'error') {\n          writeToOutputs(`${out}\\n`);\n        }\n        reject(new Error(`'${command.join(' ')}' exited with error code ${code}.`));\n      }\n    });\n  });\n}\n\nexport interface ShellOptions extends child_process.SpawnOptions {\n  /**\n   * Properties to add to 'env'\n   */\n  readonly modEnv?: Record<string, string | undefined>;\n\n  /**\n   * Don't fail when exiting with an error\n   *\n   * @default false\n   */\n  readonly allowErrExit?: boolean;\n\n  /**\n   * Whether to capture stderr\n   *\n   * @default true\n   */\n  readonly captureStderr?: boolean;\n\n  /**\n   * Pass output here\n   */\n  readonly outputs?: NodeJS.WritableStream[];\n\n  /**\n   * Only return stderr. For example, this is used to validate\n   * that when CI=true, all logs are sent to stdout.\n   *\n   * @default false\n   */\n  readonly onlyStderr?: boolean;\n\n  /**\n   * Don't log to stdout\n   *\n   * @default always\n   */\n  readonly show?: 'always' | 'never' | 'error';\n}\n\nexport class ShellHelper {\n  public static fromContext(context: TestContext & TemporaryDirectoryContext) {\n    return new ShellHelper(context.integTestDir, context.output);\n  }\n\n  constructor(\n    private readonly _cwd: string,\n    private readonly _output: NodeJS.WritableStream) { }\n\n  public async shell(command: string[], options: Omit<ShellOptions, 'cwd' | 'outputs'> = {}): Promise<string> {\n    return shell(command, {\n      outputs: [this._output],\n      cwd: this._cwd,\n      ...options,\n    });\n  }\n}\n\n/**\n * rm -rf reimplementation, don't want to depend on an NPM package for this\n *\n * Returns `true` if everything got deleted, or `false` if some files could\n * not be deleted due to permissions issues.\n */\nexport function rimraf(fsPath: string): boolean {\n  try {\n    let success = true;\n    const isDir = fs.lstatSync(fsPath).isDirectory();\n\n    if (isDir) {\n      for (const file of fs.readdirSync(fsPath)) {\n        success &&= rimraf(path.join(fsPath, file));\n      }\n      fs.rmdirSync(fsPath);\n    } else {\n      fs.unlinkSync(fsPath);\n    }\n    return success;\n  } catch (e: any) {\n    // Can happen if some files got generated inside a Docker container and are now inadvertently owned by `root`.\n    // We can't ever clean those up anymore, but since it only happens inside GitHub Actions containers we also don't care too much.\n    if (e.code === 'EACCES' || e.code === 'ENOTEMPTY') { return false; }\n\n    // Already gone\n    if (e.code === 'ENOENT') { return true; }\n\n    throw e;\n  }\n}\n\nexport function addToShellPath(x: string) {\n  const parts = process.env.PATH?.split(':') ?? [];\n\n  if (!parts.includes(x)) {\n    parts.unshift(x);\n  }\n\n  process.env.PATH = parts.join(':');\n}\n"]}
|
package/lib/with-aws.d.ts
CHANGED
|
@@ -2,6 +2,9 @@ import { AwsClients } from './aws';
|
|
|
2
2
|
import { TestContext } from './integ-test';
|
|
3
3
|
import { ResourcePool } from './resource-pool';
|
|
4
4
|
import { DisableBootstrapContext } from './with-cdk-app';
|
|
5
|
+
export declare function atmosphereEnabled(): boolean;
|
|
6
|
+
export declare function atmosphereEndpoint(): string;
|
|
7
|
+
export declare function atmospherePool(): string;
|
|
5
8
|
export type AwsContext = {
|
|
6
9
|
readonly aws: AwsClients;
|
|
7
10
|
};
|
package/lib/with-aws.js
CHANGED
|
@@ -1,20 +1,68 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.atmosphereEnabled = atmosphereEnabled;
|
|
4
|
+
exports.atmosphereEndpoint = atmosphereEndpoint;
|
|
5
|
+
exports.atmospherePool = atmospherePool;
|
|
3
6
|
exports.withAws = withAws;
|
|
4
7
|
exports.regionPool = regionPool;
|
|
8
|
+
const cdk_atmosphere_client_1 = require("@cdklabs/cdk-atmosphere-client");
|
|
5
9
|
const aws_1 = require("./aws");
|
|
6
10
|
const resource_pool_1 = require("./resource-pool");
|
|
11
|
+
function atmosphereEnabled() {
|
|
12
|
+
const enabled = process.env.CDK_INTEG_ATMOSPHERE_ENABLED;
|
|
13
|
+
return enabled === 'true' || enabled === '1';
|
|
14
|
+
}
|
|
15
|
+
function atmosphereEndpoint() {
|
|
16
|
+
const value = process.env.CDK_INTEG_ATMOSPHERE_ENDPOINT;
|
|
17
|
+
if (!value) {
|
|
18
|
+
throw new Error('CDK_INTEG_ATMOSPHERE_ENDPOINT is not defined');
|
|
19
|
+
}
|
|
20
|
+
return value;
|
|
21
|
+
}
|
|
22
|
+
function atmospherePool() {
|
|
23
|
+
const value = process.env.CDK_INTEG_ATMOSPHERE_POOL;
|
|
24
|
+
if (!value) {
|
|
25
|
+
throw new Error('CDK_INTEG_ATMOSPHERE_POOL is not defined');
|
|
26
|
+
}
|
|
27
|
+
return value;
|
|
28
|
+
}
|
|
7
29
|
/**
|
|
8
30
|
* Higher order function to execute a block with an AWS client setup
|
|
9
31
|
*
|
|
10
32
|
* Allocate the next region from the REGION pool and dispose it afterwards.
|
|
11
33
|
*/
|
|
12
34
|
function withAws(block, disableBootstrap = false) {
|
|
13
|
-
return
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
35
|
+
return async (context) => {
|
|
36
|
+
if (atmosphereEnabled()) {
|
|
37
|
+
const atmosphere = new cdk_atmosphere_client_1.AtmosphereClient(atmosphereEndpoint());
|
|
38
|
+
const allocation = await atmosphere.acquire({ pool: atmospherePool(), requester: context.name });
|
|
39
|
+
const aws = await aws_1.AwsClients.forIdentity(allocation.environment.region, {
|
|
40
|
+
accessKeyId: allocation.credentials.accessKeyId,
|
|
41
|
+
secretAccessKey: allocation.credentials.secretAccessKey,
|
|
42
|
+
sessionToken: allocation.credentials.sessionToken,
|
|
43
|
+
accountId: allocation.environment.account,
|
|
44
|
+
}, context.output);
|
|
45
|
+
await sanityCheck(aws);
|
|
46
|
+
let outcome = 'success';
|
|
47
|
+
try {
|
|
48
|
+
return await block({ ...context, disableBootstrap, aws });
|
|
49
|
+
}
|
|
50
|
+
catch (e) {
|
|
51
|
+
outcome = 'failure';
|
|
52
|
+
throw e;
|
|
53
|
+
}
|
|
54
|
+
finally {
|
|
55
|
+
await atmosphere.release(allocation.id, outcome);
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
else {
|
|
59
|
+
return regionPool().using(async (region) => {
|
|
60
|
+
const aws = await aws_1.AwsClients.forRegion(region, context.output);
|
|
61
|
+
await sanityCheck(aws);
|
|
62
|
+
return block({ ...context, disableBootstrap, aws });
|
|
63
|
+
});
|
|
64
|
+
}
|
|
65
|
+
};
|
|
18
66
|
}
|
|
19
67
|
let _regionPool;
|
|
20
68
|
function regionPool() {
|
|
@@ -56,4 +104,4 @@ async function sanityCheck(aws) {
|
|
|
56
104
|
}
|
|
57
105
|
}
|
|
58
106
|
let sanityChecked;
|
|
59
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
107
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"with-aws.js","sourceRoot":"","sources":["with-aws.ts"],"names":[],"mappings":"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","sourcesContent":["import { AtmosphereClient } from '@cdklabs/cdk-atmosphere-client';\nimport { AwsClients } from './aws';\nimport { TestContext } from './integ-test';\nimport { ResourcePool } from './resource-pool';\nimport { DisableBootstrapContext } from './with-cdk-app';\n\nexport function atmosphereEnabled(): boolean {\n  const enabled = process.env.CDK_INTEG_ATMOSPHERE_ENABLED;\n  return enabled === 'true' || enabled === '1';\n}\n\nexport function atmosphereEndpoint(): string {\n  const value = process.env.CDK_INTEG_ATMOSPHERE_ENDPOINT;\n  if (!value) {\n    throw new Error('CDK_INTEG_ATMOSPHERE_ENDPOINT is not defined');\n  }\n  return value;\n}\n\nexport function atmospherePool() {\n  const value = process.env.CDK_INTEG_ATMOSPHERE_POOL;\n  if (!value) {\n    throw new Error('CDK_INTEG_ATMOSPHERE_POOL is not defined');\n  }\n  return value;\n}\n\nexport type AwsContext = { readonly aws: AwsClients };\n\n/**\n * Higher order function to execute a block with an AWS client setup\n *\n * Allocate the next region from the REGION pool and dispose it afterwards.\n */\nexport function withAws<A extends TestContext>(\n  block: (context: A & AwsContext & DisableBootstrapContext) => Promise<void>,\n  disableBootstrap: boolean = false,\n): (context: A) => Promise<void> {\n  return async (context: A) => {\n    if (atmosphereEnabled()) {\n      const atmosphere = new AtmosphereClient(atmosphereEndpoint());\n      const allocation = await atmosphere.acquire({ pool: atmospherePool(), requester: context.name });\n      const aws = await AwsClients.forIdentity(allocation.environment.region, {\n        accessKeyId: allocation.credentials.accessKeyId,\n        secretAccessKey: allocation.credentials.secretAccessKey,\n        sessionToken: allocation.credentials.sessionToken,\n        accountId: allocation.environment.account,\n      }, context.output);\n\n      await sanityCheck(aws);\n\n      let outcome = 'success';\n      try {\n        return await block({ ...context, disableBootstrap, aws });\n      } catch (e: any) {\n        outcome = 'failure';\n        throw e;\n      } finally {\n        await atmosphere.release(allocation.id, outcome);\n      }\n    } else {\n      return regionPool().using(async (region) => {\n        const aws = await AwsClients.forRegion(region, context.output);\n        await sanityCheck(aws);\n\n        return block({ ...context, disableBootstrap, aws });\n      });\n    }\n  };\n}\n\nlet _regionPool: undefined | ResourcePool;\nexport function regionPool(): ResourcePool {\n  if (_regionPool !== undefined) {\n    return _regionPool;\n  }\n\n  const REGIONS = process.env.AWS_REGIONS\n    ? process.env.AWS_REGIONS.split(',')\n    : [process.env.AWS_REGION ?? process.env.AWS_DEFAULT_REGION ?? 'us-east-1'];\n\n  // eslint-disable-next-line no-console\n  console.log(`Using regions: ${REGIONS}\\n`);\n\n  _regionPool = ResourcePool.withResources('aws_regions', REGIONS);\n  return _regionPool;\n}\n\n/**\n * Perform a one-time quick sanity check that the AWS clients have properly configured credentials\n *\n * If we don't do this, calls are going to fail and they'll be retried and everything will take\n * forever before the user notices a simple misconfiguration.\n *\n * We can't check for the presence of environment variables since credentials could come from\n * anywhere, so do simple account retrieval.\n *\n * Only do it once per process.\n */\nasync function sanityCheck(aws: AwsClients) {\n  if (sanityChecked === undefined) {\n    try {\n      await aws.account();\n      sanityChecked = true;\n    } catch (e: any) {\n      sanityChecked = false;\n      throw new Error(`AWS credentials probably not configured, got error: ${e.message}`);\n    }\n  }\n  if (!sanityChecked) {\n    throw new Error('AWS credentials probably not configured, see previous error');\n  }\n}\nlet sanityChecked: boolean | undefined;\n"]}
|
package/lib/with-cdk-app.d.ts
CHANGED
|
@@ -160,6 +160,7 @@ export declare class TestFixture extends ShellHelper {
|
|
|
160
160
|
constructor(integTestDir: string, stackNamePrefix: string, output: NodeJS.WritableStream, aws: AwsClients, randomString: string);
|
|
161
161
|
log(s: string): void;
|
|
162
162
|
cdkDeploy(stackNames: string | string[], options?: CdkCliOptions, skipStackRename?: boolean): Promise<string>;
|
|
163
|
+
cdkDeployCommandLine(stackNames: string | string[], options?: CdkCliOptions, skipStackRename?: boolean): string[];
|
|
163
164
|
cdkSynth(options?: CdkCliOptions): Promise<string>;
|
|
164
165
|
cdkDestroy(stackNames: string | string[], options?: CdkCliOptions): Promise<string>;
|
|
165
166
|
cdkBootstrapLegacy(options: CdkLegacyBootstrapCommandOptions): Promise<string>;
|
|
@@ -167,6 +168,16 @@ export declare class TestFixture extends ShellHelper {
|
|
|
167
168
|
cdkGarbageCollect(options: CdkGarbageCollectionCommandOptions): Promise<string>;
|
|
168
169
|
cdkMigrate(language: string, stackName: string, inputPath?: string, options?: CdkCliOptions): Promise<string>;
|
|
169
170
|
cdk(args: string[], options?: CdkCliOptions): Promise<string>;
|
|
171
|
+
/**
|
|
172
|
+
* Return the environment variables with which to execute CDK
|
|
173
|
+
*/
|
|
174
|
+
cdkShellEnv(): {
|
|
175
|
+
AWS_REGION: string;
|
|
176
|
+
AWS_DEFAULT_REGION: string;
|
|
177
|
+
STACK_NAME_PREFIX: string;
|
|
178
|
+
PACKAGE_LAYOUT_VERSION: string;
|
|
179
|
+
CI: string;
|
|
180
|
+
};
|
|
170
181
|
template(stackName: string): any;
|
|
171
182
|
bootstrapRepoName(): Promise<string>;
|
|
172
183
|
get bootstrapStackName(): string;
|