@aws-cdk-testing/cli-integ 3.7.2 → 3.8.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/integ-test.d.ts
CHANGED
|
@@ -3,6 +3,7 @@ export interface TestContext {
|
|
|
3
3
|
readonly name: string;
|
|
4
4
|
readonly output: NodeJS.WritableStream;
|
|
5
5
|
log(s: string): void;
|
|
6
|
+
reportWaitTime(ms: number): void;
|
|
6
7
|
}
|
|
7
8
|
/**
|
|
8
9
|
* A wrapper for jest's 'test' which takes regression-disabled tests into account and prints a banner
|
package/lib/integ-test.js
CHANGED
|
@@ -26,7 +26,8 @@ function integTest(name, callback, timeoutMillis) {
|
|
|
26
26
|
output.write('================================================================\n');
|
|
27
27
|
output.write(`${name}\n`);
|
|
28
28
|
output.write('================================================================\n');
|
|
29
|
-
const
|
|
29
|
+
const start = Date.now();
|
|
30
|
+
let waitTime = 0;
|
|
30
31
|
process.stderr.write(`[INTEG TEST::${name}] Starting (pid ${process.pid})...\n`);
|
|
31
32
|
maybePrintMemoryUsage(name);
|
|
32
33
|
try {
|
|
@@ -40,8 +41,16 @@ function integTest(name, callback, timeoutMillis) {
|
|
|
40
41
|
log(s) {
|
|
41
42
|
output.write(`${s}\n`);
|
|
42
43
|
},
|
|
44
|
+
reportWaitTime(n) {
|
|
45
|
+
waitTime += n;
|
|
46
|
+
},
|
|
47
|
+
});
|
|
48
|
+
await writeLog(name, {
|
|
49
|
+
success: true,
|
|
50
|
+
output: output.toString(),
|
|
51
|
+
totalDuration: Date.now() - start,
|
|
52
|
+
waitTime,
|
|
43
53
|
});
|
|
44
|
-
await writeLog(name, true, output.toString());
|
|
45
54
|
return ret;
|
|
46
55
|
}
|
|
47
56
|
catch (e) {
|
|
@@ -49,7 +58,12 @@ function integTest(name, callback, timeoutMillis) {
|
|
|
49
58
|
failed = true;
|
|
50
59
|
output.write(e.message);
|
|
51
60
|
output.write(e.stack);
|
|
52
|
-
await writeLog(name,
|
|
61
|
+
await writeLog(name, {
|
|
62
|
+
success: false,
|
|
63
|
+
output: output.toString(),
|
|
64
|
+
totalDuration: Date.now() - start,
|
|
65
|
+
waitTime,
|
|
66
|
+
});
|
|
53
67
|
process.stderr.write(`[INTEG TEST::${name}] Failed: ${e}\n`);
|
|
54
68
|
const isGitHub = !!process.env.GITHUB_RUN_ID;
|
|
55
69
|
if (isGitHub) {
|
|
@@ -81,8 +95,8 @@ function integTest(name, callback, timeoutMillis) {
|
|
|
81
95
|
throw e;
|
|
82
96
|
}
|
|
83
97
|
finally {
|
|
84
|
-
const duration = Date.now() -
|
|
85
|
-
process.stderr.write(`[INTEG TEST::${name}] Done (${duration}
|
|
98
|
+
const duration = Date.now() - start;
|
|
99
|
+
process.stderr.write(`[INTEG TEST::${name}] Done (${humanTime(duration)}).\n`);
|
|
86
100
|
maybePrintMemoryUsage(name);
|
|
87
101
|
}
|
|
88
102
|
}, timeoutMillis);
|
|
@@ -120,20 +134,51 @@ function randomString() {
|
|
|
120
134
|
* end up empty or with interleaved contents). The other writes are not
|
|
121
135
|
* contended and don't need to be atomic, but the function is just ergonomic to use.
|
|
122
136
|
*/
|
|
123
|
-
async function writeLog(testName,
|
|
137
|
+
async function writeLog(testName, result) {
|
|
124
138
|
if (process.env.INTEG_LOGS) {
|
|
139
|
+
// Write the log file
|
|
125
140
|
const slug = slugify(testName);
|
|
126
|
-
const logFileName = `${process.env.INTEG_LOGS}/${success ? '' : 'FAILED-'}${slug}.txt`;
|
|
127
|
-
await atomicWrite(logFileName, output);
|
|
141
|
+
const logFileName = `${process.env.INTEG_LOGS}/${result.success ? '' : 'FAILED-'}${slug}.txt`;
|
|
142
|
+
await atomicWrite(logFileName, result.output);
|
|
143
|
+
// Write a row for the markdown table
|
|
128
144
|
// Sort failures before successes, and the table header before all
|
|
145
|
+
const mdFileName = `${process.env.INTEG_LOGS}/md/${result.success ? '2' : '1'}-${slug}.md`;
|
|
146
|
+
const columns = [
|
|
147
|
+
['Result', result.success ? 'pass ✅' : 'fail ❌'],
|
|
148
|
+
['Test Name', testName],
|
|
149
|
+
['Test Duration', humanTime(result.totalDuration - result.waitTime)],
|
|
150
|
+
['Wait Time', result.waitTime > 0 ? humanTime(result.waitTime) : '-'],
|
|
151
|
+
];
|
|
129
152
|
await atomicWrite(`${process.env.INTEG_LOGS}/md/0-header.md`, [
|
|
130
|
-
|
|
131
|
-
'
|
|
153
|
+
`| ${columns.map(([col, _val]) => col).join(' | ')} |`,
|
|
154
|
+
`| ${columns.map(() => '-----------').join(' | ')} |`,
|
|
132
155
|
].map(x => `${x}\n`).join(''));
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
156
|
+
await atomicWrite(mdFileName, `| ${columns.map(([_col, val]) => val).join(' | ')} |\n`);
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
function humanTime(delta) {
|
|
160
|
+
const components = [];
|
|
161
|
+
const S = 1000;
|
|
162
|
+
const M = 60 * S;
|
|
163
|
+
const H = 60 * M;
|
|
164
|
+
const hours = Math.floor(delta / H);
|
|
165
|
+
if (hours > 0) {
|
|
166
|
+
components.push(`${hours}h`);
|
|
167
|
+
delta -= hours * H;
|
|
168
|
+
}
|
|
169
|
+
const minutes = Math.floor(delta / M);
|
|
170
|
+
if (minutes > 0) {
|
|
171
|
+
components.push(`${minutes}m`);
|
|
172
|
+
delta -= minutes * M;
|
|
173
|
+
}
|
|
174
|
+
const seconds = Math.floor(delta / S);
|
|
175
|
+
if (seconds > 0) {
|
|
176
|
+
components.push(`${seconds}s`);
|
|
177
|
+
delta -= seconds * S;
|
|
136
178
|
}
|
|
179
|
+
components.push(`${delta}ms`);
|
|
180
|
+
// Retain the 2 most significant components
|
|
181
|
+
return components.slice(0, 2).join('');
|
|
137
182
|
}
|
|
138
183
|
function slugify(x) {
|
|
139
184
|
return x.replace(/[^a-zA-Z0-9_,]+/g, '-');
|
|
@@ -144,4 +189,4 @@ async function atomicWrite(fileName, contents) {
|
|
|
144
189
|
await fs.promises.writeFile(tmp, contents);
|
|
145
190
|
await fs.promises.rename(tmp, fileName);
|
|
146
191
|
}
|
|
147
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"integ-test.js","sourceRoot":"","sources":["integ-test.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport { MemoryStream } from './corking';\n\nconst SKIP_TESTS = fs.readFileSync(path.join(__dirname, '..', 'skip-tests.txt'), { encoding: 'utf-8' })\n  .split('\\n')\n  .map(x => x.trim())\n  .filter(x => x && !x.startsWith('#'));\n\nif (SKIP_TESTS.length > 0) {\n  process.stderr.write(`ℹ️ Skipping tests: ${JSON.stringify(SKIP_TESTS)}\\n`);\n}\n\n// Whether we want to stop after the first failure, for quicker debugging (hopefully).\nconst FAIL_FAST = process.env.FAIL_FAST === 'true';\n\n// Keep track of whether the suite has failed. If so, we stop running.\nlet failed = false;\n\nexport interface TestContext {\n  readonly randomString: string;\n  readonly name: string;\n  readonly output: NodeJS.WritableStream;\n  log(s: string): void;\n}\n\n/**\n * A wrapper for jest's 'test' which takes regression-disabled tests into account and prints a banner\n */\nexport function integTest(\n  name: string,\n  callback: (context: TestContext) => Promise<void>,\n  timeoutMillis?: number,\n): void {\n  const runner = shouldSkip(name) ? test.skip : test;\n\n  runner(name, async () => {\n    const output = new MemoryStream();\n\n    output.write('================================================================\\n');\n    output.write(`${name}\\n`);\n    output.write('================================================================\\n');\n\n    const now = Date.now();\n    process.stderr.write(`[INTEG TEST::${name}] Starting (pid ${process.pid})...\\n`);\n    maybePrintMemoryUsage(name);\n    try {\n      if (FAIL_FAST && failed) {\n        throw new Error('FAIL_FAST requested and currently failing. Stopping test early.');\n      }\n\n      const ret = await callback({\n        output,\n        randomString: randomString(),\n        name,\n        log(s: string) {\n          output.write(`${s}\\n`);\n        },\n      });\n\n      await writeLog(name, true, output.toString());\n\n      return ret;\n    } catch (e: any) {\n      // Print the buffered output, only if the test fails.\n      failed = true;\n\n      output.write(e.message);\n      output.write(e.stack);\n\n      await writeLog(name, false, output.toString());\n      process.stderr.write(`[INTEG TEST::${name}] Failed: ${e}\\n`);\n\n      const isGitHub = !!process.env.GITHUB_RUN_ID;\n\n      if (isGitHub) {\n        // GitHub Actions compatible output formatting\n        // https://docs.github.com/en/actions/writing-workflows/choosing-what-your-workflow-does/workflow-commands-for-github-actions#setting-an-error-message\n        let written = process.stderr.write(`::error title=Failed ${name}::${e.message}\\n`);\n        if (!written) {\n          // Wait for drain\n          await new Promise((ok) => process.stderr.once('drain', ok));\n        }\n\n        // Print output only if the test fails. Use 'console.log' so the output is buffered by\n        // jest and prints without a stack trace (if verbose: false).\n        written = process.stdout.write([\n          `::group::Failure details: ${name} (click to expand)\\n`,\n          `${output.buffer().toString()}\\n`,\n          '::endgroup::\\n',\n        ].join(''));\n        if (!written) {\n          // Wait for drain\n          await new Promise((ok) => process.stdout.once('drain', ok));\n        }\n      } else {\n        // Use 'console.log' so the output is buffered by\n        // jest and prints without a stack trace (if verbose: false).\n        // eslint-disable-next-line no-console\n        console.log(output.buffer().toString());\n      }\n      throw e;\n    } finally {\n      const duration = Date.now() - now;\n      process.stderr.write(`[INTEG TEST::${name}] Done (${duration} ms).\\n`);\n      maybePrintMemoryUsage(name);\n    }\n  }, timeoutMillis);\n}\n\nfunction shouldSkip(testName: string) {\n  return SKIP_TESTS.includes(testName);\n}\n\nfunction maybePrintMemoryUsage(testName: string) {\n  if (process.env.INTEG_MEMORY_DEBUG !== 'true') {\n    return;\n  }\n  const memoryUsage = process.memoryUsage() as any;\n  const report: any = {};\n  for (const [key, value] of Object.entries(memoryUsage)) {\n    report[key] = `${Math.round(value as number / 1024 / 1024)} MB`;\n  }\n  process.stderr.write(`[INTEG TEST::${testName}] Memory Usage: ${JSON.stringify(report)}`);\n}\n\nexport function randomString() {\n  // Crazy\n  return Math.random().toString(36).replace(/[^a-z0-9]+/g, '');\n}\n\n/**\n * Write log files\n *\n * Write a text log to `${INTEG_LOGS}/[FAILED-]description-of-test.txt`, and a single\n * line of a Markdown table to `${INTEG_LOGS}/md/1-description-of-test.md`.\n *\n * The latter are designed to be globcatted to $GITHUB_STEP_SUMMARY after tests\n * (we don't write there directly to avoid concurrency issues with multiple processes\n * reading and mutating the same file).\n *\n * We do use `atomicWrite` to write files -- it's only necessary for the header file,\n * which gets overwritten by every test, just to make sure it properly exists (shouldn't\n * end up empty or with interleaved contents). The other writes are not\n * contended and don't need to be atomic, but the function is just ergonomic to use.\n */\nasync function writeLog(testName: string, success: boolean, output: string) {\n  if (process.env.INTEG_LOGS) {\n    const slug = slugify(testName);\n    const logFileName = `${process.env.INTEG_LOGS}/${success ? '' : 'FAILED-'}${slug}.txt`;\n    await atomicWrite(logFileName, output);\n\n    // Sort failures before successes, and the table header before all\n    await atomicWrite(`${process.env.INTEG_LOGS}/md/0-header.md`, [\n      '| Result | Test Name |',\n      '|--------|-----------|',\n    ].map(x => `${x}\\n`).join(''));\n\n    const mdFileName = `${process.env.INTEG_LOGS}/md/${success ? '2' : '1'}-${slug}.md`;\n    const firstColumn = success ? 'pass ✅' : 'fail ❌';\n    await atomicWrite(mdFileName, `| ${firstColumn} | ${testName} |\\n`);\n  }\n}\n\nfunction slugify(x: string) {\n  return x.replace(/[^a-zA-Z0-9_,]+/g, '-');\n}\n\nasync function atomicWrite(fileName: string, contents: string) {\n  await fs.promises.mkdir(path.dirname(fileName), { recursive: true });\n\n  const tmp = `${fileName}.${process.pid}`;\n  await fs.promises.writeFile(tmp, contents);\n  await fs.promises.rename(tmp, fileName);\n}\n"]}
|
|
192
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"integ-test.js","sourceRoot":"","sources":["integ-test.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport { MemoryStream } from './corking';\n\nconst SKIP_TESTS = fs.readFileSync(path.join(__dirname, '..', 'skip-tests.txt'), { encoding: 'utf-8' })\n  .split('\\n')\n  .map(x => x.trim())\n  .filter(x => x && !x.startsWith('#'));\n\nif (SKIP_TESTS.length > 0) {\n  process.stderr.write(`ℹ️ Skipping tests: ${JSON.stringify(SKIP_TESTS)}\\n`);\n}\n\n// Whether we want to stop after the first failure, for quicker debugging (hopefully).\nconst FAIL_FAST = process.env.FAIL_FAST === 'true';\n\n// Keep track of whether the suite has failed. If so, we stop running.\nlet failed = false;\n\nexport interface TestContext {\n  readonly randomString: string;\n  readonly name: string;\n  readonly output: NodeJS.WritableStream;\n  log(s: string): void;\n  reportWaitTime(ms: number): void;\n}\n\n/**\n * A wrapper for jest's 'test' which takes regression-disabled tests into account and prints a banner\n */\nexport function integTest(\n  name: string,\n  callback: (context: TestContext) => Promise<void>,\n  timeoutMillis?: number,\n): void {\n  const runner = shouldSkip(name) ? test.skip : test;\n\n  runner(name, async () => {\n    const output = new MemoryStream();\n\n    output.write('================================================================\\n');\n    output.write(`${name}\\n`);\n    output.write('================================================================\\n');\n\n    const start = Date.now();\n    let waitTime = 0;\n\n    process.stderr.write(`[INTEG TEST::${name}] Starting (pid ${process.pid})...\\n`);\n    maybePrintMemoryUsage(name);\n    try {\n      if (FAIL_FAST && failed) {\n        throw new Error('FAIL_FAST requested and currently failing. Stopping test early.');\n      }\n\n      const ret = await callback({\n        output,\n        randomString: randomString(),\n        name,\n        log(s: string) {\n          output.write(`${s}\\n`);\n        },\n        reportWaitTime(n) {\n          waitTime += n;\n        },\n      });\n\n      await writeLog(name, {\n        success: true,\n        output: output.toString(),\n        totalDuration: Date.now() - start,\n        waitTime,\n      });\n\n      return ret;\n    } catch (e: any) {\n      // Print the buffered output, only if the test fails.\n      failed = true;\n\n      output.write(e.message);\n      output.write(e.stack);\n\n      await writeLog(name, {\n        success: false,\n        output: output.toString(),\n        totalDuration: Date.now() - start,\n        waitTime,\n      });\n      process.stderr.write(`[INTEG TEST::${name}] Failed: ${e}\\n`);\n\n      const isGitHub = !!process.env.GITHUB_RUN_ID;\n\n      if (isGitHub) {\n        // GitHub Actions compatible output formatting\n        // https://docs.github.com/en/actions/writing-workflows/choosing-what-your-workflow-does/workflow-commands-for-github-actions#setting-an-error-message\n        let written = process.stderr.write(`::error title=Failed ${name}::${e.message}\\n`);\n        if (!written) {\n          // Wait for drain\n          await new Promise((ok) => process.stderr.once('drain', ok));\n        }\n\n        // Print output only if the test fails. Use 'console.log' so the output is buffered by\n        // jest and prints without a stack trace (if verbose: false).\n        written = process.stdout.write([\n          `::group::Failure details: ${name} (click to expand)\\n`,\n          `${output.buffer().toString()}\\n`,\n          '::endgroup::\\n',\n        ].join(''));\n        if (!written) {\n          // Wait for drain\n          await new Promise((ok) => process.stdout.once('drain', ok));\n        }\n      } else {\n        // Use 'console.log' so the output is buffered by\n        // jest and prints without a stack trace (if verbose: false).\n        // eslint-disable-next-line no-console\n        console.log(output.buffer().toString());\n      }\n      throw e;\n    } finally {\n      const duration = Date.now() - start;\n      process.stderr.write(`[INTEG TEST::${name}] Done (${humanTime(duration)}).\\n`);\n      maybePrintMemoryUsage(name);\n    }\n  }, timeoutMillis);\n}\n\nfunction shouldSkip(testName: string) {\n  return SKIP_TESTS.includes(testName);\n}\n\nfunction maybePrintMemoryUsage(testName: string) {\n  if (process.env.INTEG_MEMORY_DEBUG !== 'true') {\n    return;\n  }\n  const memoryUsage = process.memoryUsage() as any;\n  const report: any = {};\n  for (const [key, value] of Object.entries(memoryUsage)) {\n    report[key] = `${Math.round(value as number / 1024 / 1024)} MB`;\n  }\n  process.stderr.write(`[INTEG TEST::${testName}] Memory Usage: ${JSON.stringify(report)}`);\n}\n\nexport function randomString() {\n  // Crazy\n  return Math.random().toString(36).replace(/[^a-z0-9]+/g, '');\n}\n\n/**\n * Write log files\n *\n * Write a text log to `${INTEG_LOGS}/[FAILED-]description-of-test.txt`, and a single\n * line of a Markdown table to `${INTEG_LOGS}/md/1-description-of-test.md`.\n *\n * The latter are designed to be globcatted to $GITHUB_STEP_SUMMARY after tests\n * (we don't write there directly to avoid concurrency issues with multiple processes\n * reading and mutating the same file).\n *\n * We do use `atomicWrite` to write files -- it's only necessary for the header file,\n * which gets overwritten by every test, just to make sure it properly exists (shouldn't\n * end up empty or with interleaved contents). The other writes are not\n * contended and don't need to be atomic, but the function is just ergonomic to use.\n */\nasync function writeLog(testName: string, result: {\n  success: boolean;\n  output: string;\n  totalDuration: number;\n  waitTime: number;\n}) {\n  if (process.env.INTEG_LOGS) {\n    // Write the log file\n    const slug = slugify(testName);\n    const logFileName = `${process.env.INTEG_LOGS}/${result.success ? '' : 'FAILED-'}${slug}.txt`;\n    await atomicWrite(logFileName, result.output);\n\n    // Write a row for the markdown table\n    // Sort failures before successes, and the table header before all\n    const mdFileName = `${process.env.INTEG_LOGS}/md/${result.success ? '2' : '1'}-${slug}.md`;\n    const columns: Array<[string, string]> = [\n      ['Result', result.success ? 'pass ✅' : 'fail ❌'],\n      ['Test Name', testName],\n      ['Test Duration', humanTime(result.totalDuration - result.waitTime)],\n      ['Wait Time', result.waitTime > 0 ? humanTime(result.waitTime) : '-'],\n    ];\n    await atomicWrite(`${process.env.INTEG_LOGS}/md/0-header.md`, [\n      `| ${columns.map(([col, _val]) => col).join(' | ')} |`,\n      `| ${columns.map(() => '-----------').join(' | ')} |`,\n    ].map(x => `${x}\\n`).join(''));\n    await atomicWrite(mdFileName,\n      `| ${columns.map(([_col, val]) => val).join(' | ')} |\\n`);\n  }\n}\n\nfunction humanTime(delta: number) {\n  const components = [];\n\n  const S = 1000;\n  const M = 60 * S;\n  const H = 60 * M;\n\n  const hours = Math.floor(delta / H);\n  if (hours > 0) {\n    components.push(`${hours}h`);\n    delta -= hours * H;\n  }\n  const minutes = Math.floor(delta / M);\n  if (minutes > 0) {\n    components.push(`${minutes}m`);\n    delta -= minutes * M;\n  }\n  const seconds = Math.floor(delta / S);\n  if (seconds > 0) {\n    components.push(`${seconds}s`);\n    delta -= seconds * S;\n  }\n  components.push(`${delta}ms`);\n\n  // Retain the 2 most significant components\n  return components.slice(0, 2).join('');\n}\n\nfunction slugify(x: string) {\n  return x.replace(/[^a-zA-Z0-9_,]+/g, '-');\n}\n\nasync function atomicWrite(fileName: string, contents: string) {\n  await fs.promises.mkdir(path.dirname(fileName), { recursive: true });\n\n  const tmp = `${fileName}.${process.pid}`;\n  await fs.promises.writeFile(tmp, contents);\n  await fs.promises.rename(tmp, fileName);\n}\n"]}
|
package/lib/with-aws.js
CHANGED
|
@@ -37,7 +37,9 @@ function withAws(block, disableBootstrap = false) {
|
|
|
37
37
|
const atmosphere = new cdk_atmosphere_client_1.AtmosphereClient(atmosphereEndpoint(), {
|
|
38
38
|
logStream: context.output,
|
|
39
39
|
});
|
|
40
|
+
const start = Date.now();
|
|
40
41
|
const allocation = await atmosphere.acquire({ pool: atmospherePool(), requester: context.name, timeoutSeconds: 60 * 30 });
|
|
42
|
+
context.reportWaitTime(Date.now() - start);
|
|
41
43
|
const aws = await aws_1.AwsClients.forIdentity(allocation.environment.region, {
|
|
42
44
|
accessKeyId: allocation.credentials.accessKeyId,
|
|
43
45
|
secretAccessKey: allocation.credentials.secretAccessKey,
|
|
@@ -104,4 +106,4 @@ async function sanityCheck(aws) {
|
|
|
104
106
|
}
|
|
105
107
|
}
|
|
106
108
|
let sanityChecked;
|
|
107
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
109
|
+
//# 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 type { TestContext } from './integ-test';\nimport { ResourcePool } from './resource-pool';\nimport type { 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        logStream: context.output,\n      });\n\n      const start = Date.now();\n      const allocation = await atmosphere.acquire({ pool: atmospherePool(), requester: context.name, timeoutSeconds: 60 * 30 });\n      context.reportWaitTime(Date.now() - start);\n\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  _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"]}
|