aws-cdk 2.1006.0 → 2.1007.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.
@@ -9,21 +9,23 @@ exports.spaceAvailableForContext = spaceAvailableForContext;
9
9
  const childProcess = require("child_process");
10
10
  const os = require("os");
11
11
  const path = require("path");
12
+ const util_1 = require("util");
12
13
  const cxschema = require("@aws-cdk/cloud-assembly-schema");
13
14
  const cxapi = require("@aws-cdk/cx-api");
14
15
  const fs = require("fs-extra");
15
16
  const semver = require("semver");
16
17
  const api_1 = require("../../../../@aws-cdk/tmp-toolkit-helpers/src/api");
18
+ const private_1 = require("../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private");
17
19
  const tree_1 = require("../../api/tree");
18
20
  const user_configuration_1 = require("../../cli/user-configuration");
19
21
  const version_1 = require("../../cli/version");
20
- const logging_1 = require("../../logging");
21
- const util_1 = require("../../util");
22
+ const util_2 = require("../../util");
22
23
  const rwlock_1 = require("../util/rwlock");
23
24
  /** Invokes the cloud executable and returns JSON output */
24
- async function execProgram(aws, config) {
25
- const env = await prepareDefaultEnvironment(aws);
26
- const context = await prepareContext(config.settings, config.context.all, env);
25
+ async function execProgram(aws, ioHelper, config) {
26
+ const debugFn = (msg) => ioHelper.notify(private_1.IO.DEFAULT_ASSEMBLY_DEBUG.msg(msg));
27
+ const env = await prepareDefaultEnvironment(aws, debugFn);
28
+ const context = await prepareContext(config.settings, config.context.all, env, debugFn);
27
29
  const build = config.settings.get(['build']);
28
30
  if (build) {
29
31
  await exec(build);
@@ -34,12 +36,12 @@ async function execProgram(aws, config) {
34
36
  }
35
37
  // bypass "synth" if app points to a cloud assembly
36
38
  if (await fs.pathExists(app) && (await fs.stat(app)).isDirectory()) {
37
- (0, logging_1.debug)('--app points to a cloud assembly, so we bypass synth');
39
+ await debugFn('--app points to a cloud assembly, so we bypass synth');
38
40
  // Acquire a read lock on this directory
39
41
  const lock = await new rwlock_1.RWLock(app).acquireRead();
40
42
  return { assembly: createAssembly(app), lock };
41
43
  }
42
- const commandLine = await guessExecutable(app);
44
+ const commandLine = await guessExecutable(app, debugFn);
43
45
  const outdir = config.settings.get(['output']);
44
46
  if (!outdir) {
45
47
  throw new api_1.ToolkitError('unexpected: --output is required');
@@ -53,7 +55,7 @@ async function execProgram(aws, config) {
53
55
  catch (error) {
54
56
  throw new api_1.ToolkitError(`Could not create output directory ${outdir} (${error.message})`);
55
57
  }
56
- (0, logging_1.debug)('outdir:', outdir);
58
+ await debugFn(`outdir: ${outdir}`);
57
59
  env[cxapi.OUTDIR_ENV] = outdir;
58
60
  // Acquire a lock on the output directory
59
61
  const writerLock = await new rwlock_1.RWLock(outdir).acquireWrite();
@@ -61,9 +63,9 @@ async function execProgram(aws, config) {
61
63
  // Send version information
62
64
  env[cxapi.CLI_ASM_VERSION_ENV] = cxschema.Manifest.version();
63
65
  env[cxapi.CLI_VERSION_ENV] = (0, version_1.versionNumber)();
64
- (0, logging_1.debug)('env:', env);
66
+ await debugFn((0, util_1.format)('env:', env));
65
67
  const envVariableSizeLimit = os.platform() === 'win32' ? 32760 : 131072;
66
- const [smallContext, overflow] = (0, util_1.splitBySize)(context, spaceAvailableForContext(env, envVariableSizeLimit));
68
+ const [smallContext, overflow] = (0, util_2.splitBySize)(context, spaceAvailableForContext(env, envVariableSizeLimit));
67
69
  // Store the safe part in the environment variable
68
70
  env[cxapi.CONTEXT_ENV] = JSON.stringify(smallContext);
69
71
  // If there was any overflow, write it to a temporary file
@@ -76,7 +78,7 @@ async function execProgram(aws, config) {
76
78
  }
77
79
  await exec(commandLine.join(' '));
78
80
  const assembly = createAssembly(outdir);
79
- contextOverflowCleanup(contextOverflowLocation, assembly);
81
+ await contextOverflowCleanup(contextOverflowLocation, assembly, ioHelper);
80
82
  return { assembly, lock: await writerLock.convertToReaderLock() };
81
83
  }
82
84
  catch (e) {
@@ -84,37 +86,42 @@ async function execProgram(aws, config) {
84
86
  throw e;
85
87
  }
86
88
  async function exec(commandAndArgs) {
87
- return new Promise((ok, fail) => {
88
- // We use a slightly lower-level interface to:
89
- //
90
- // - Pass arguments in an array instead of a string, to get around a
91
- // number of quoting issues introduced by the intermediate shell layer
92
- // (which would be different between Linux and Windows).
93
- //
94
- // - Inherit stderr from controlling terminal. We don't use the captured value
95
- // anyway, and if the subprocess is printing to it for debugging purposes the
96
- // user gets to see it sooner. Plus, capturing doesn't interact nicely with some
97
- // processes like Maven.
98
- const proc = childProcess.spawn(commandAndArgs, {
99
- stdio: ['ignore', 'inherit', 'inherit'],
100
- detached: false,
101
- shell: true,
102
- env: {
103
- ...process.env,
104
- ...env,
105
- },
106
- });
107
- proc.on('error', fail);
108
- proc.on('exit', code => {
109
- if (code === 0) {
110
- return ok();
111
- }
112
- else {
113
- (0, logging_1.debug)('failed command:', commandAndArgs);
114
- return fail(new api_1.ToolkitError(`Subprocess exited with error ${code}`));
115
- }
89
+ try {
90
+ await new Promise((ok, fail) => {
91
+ // We use a slightly lower-level interface to:
92
+ //
93
+ // - Pass arguments in an array instead of a string, to get around a
94
+ // number of quoting issues introduced by the intermediate shell layer
95
+ // (which would be different between Linux and Windows).
96
+ //
97
+ // - Inherit stderr from controlling terminal. We don't use the captured value
98
+ // anyway, and if the subprocess is printing to it for debugging purposes the
99
+ // user gets to see it sooner. Plus, capturing doesn't interact nicely with some
100
+ // processes like Maven.
101
+ const proc = childProcess.spawn(commandAndArgs, {
102
+ stdio: ['ignore', 'inherit', 'inherit'],
103
+ detached: false,
104
+ shell: true,
105
+ env: {
106
+ ...process.env,
107
+ ...env,
108
+ },
109
+ });
110
+ proc.on('error', fail);
111
+ proc.on('exit', code => {
112
+ if (code === 0) {
113
+ return ok();
114
+ }
115
+ else {
116
+ return fail(new api_1.ToolkitError(`Subprocess exited with error ${code}`));
117
+ }
118
+ });
116
119
  });
117
- });
120
+ }
121
+ catch (e) {
122
+ await debugFn(`failed command: ${commandAndArgs}`);
123
+ throw e;
124
+ }
118
125
  }
119
126
  }
120
127
  /**
@@ -148,14 +155,14 @@ function createAssembly(appDir) {
148
155
  *
149
156
  * @param context The context key/value bash.
150
157
  */
151
- async function prepareDefaultEnvironment(aws, logFn = logging_1.debug) {
158
+ async function prepareDefaultEnvironment(aws, debugFn) {
152
159
  const env = {};
153
160
  env[cxapi.DEFAULT_REGION_ENV] = aws.defaultRegion;
154
- await logFn(`Setting "${cxapi.DEFAULT_REGION_ENV}" environment variable to`, env[cxapi.DEFAULT_REGION_ENV]);
161
+ await debugFn(`Setting "${cxapi.DEFAULT_REGION_ENV}" environment variable to ${env[cxapi.DEFAULT_REGION_ENV]}`);
155
162
  const accountId = (await aws.defaultAccount())?.accountId;
156
163
  if (accountId) {
157
164
  env[cxapi.DEFAULT_ACCOUNT_ENV] = accountId;
158
- await logFn(`Setting "${cxapi.DEFAULT_ACCOUNT_ENV}" environment variable to`, env[cxapi.DEFAULT_ACCOUNT_ENV]);
165
+ await debugFn(`Setting "${cxapi.DEFAULT_ACCOUNT_ENV}" environment variable to ${env[cxapi.DEFAULT_ACCOUNT_ENV]}`);
159
166
  }
160
167
  return env;
161
168
  }
@@ -164,7 +171,7 @@ async function prepareDefaultEnvironment(aws, logFn = logging_1.debug) {
164
171
  * The merging of various configuration sources like cli args or cdk.json has already happened.
165
172
  * We now need to set the final values to the context.
166
173
  */
167
- async function prepareContext(settings, context, env) {
174
+ async function prepareContext(settings, context, env, debugFn) {
168
175
  const debugMode = settings.get(['debug']) ?? true;
169
176
  if (debugMode) {
170
177
  env.CDK_DEBUG = 'true';
@@ -191,7 +198,7 @@ async function prepareContext(settings, context, env) {
191
198
  }
192
199
  const bundlingStacks = settings.get(['bundlingStacks']) ?? ['**'];
193
200
  context[cxapi.BUNDLING_STACKS] = bundlingStacks;
194
- (0, logging_1.debug)('context:', context);
201
+ await debugFn((0, util_1.format)('context:', context));
195
202
  return context;
196
203
  }
197
204
  /**
@@ -224,7 +231,7 @@ const EXTENSION_MAP = new Map([
224
231
  * verify if registry associations have or have not been set up for this
225
232
  * file type, so we'll assume the worst and take control.
226
233
  */
227
- async function guessExecutable(app) {
234
+ async function guessExecutable(app, debugFn) {
228
235
  const commandLine = appToArray(app);
229
236
  if (commandLine.length === 1) {
230
237
  let fstat;
@@ -232,7 +239,7 @@ async function guessExecutable(app) {
232
239
  fstat = await fs.stat(commandLine[0]);
233
240
  }
234
241
  catch {
235
- (0, logging_1.debug)(`Not a file: '${commandLine[0]}'. Using '${commandLine}' as command-line`);
242
+ await debugFn(`Not a file: '${commandLine[0]}'. Using '${commandLine}' as command-line`);
236
243
  return commandLine;
237
244
  }
238
245
  // eslint-disable-next-line no-bitwise
@@ -245,10 +252,10 @@ async function guessExecutable(app) {
245
252
  }
246
253
  return commandLine;
247
254
  }
248
- function contextOverflowCleanup(location, assembly) {
255
+ async function contextOverflowCleanup(location, assembly, ioHelper) {
249
256
  if (location) {
250
257
  fs.removeSync(path.dirname(location));
251
- const tree = (0, tree_1.loadTree)(assembly, logging_1.trace);
258
+ const tree = await (0, tree_1.loadTree)(assembly, (msg) => ioHelper.notify(private_1.IO.DEFAULT_ASSEMBLY_TRACE.msg(msg)));
252
259
  const frameworkDoesNotSupportContextOverflow = (0, tree_1.some)(tree, node => {
253
260
  const fqn = node.constructInfo?.fqn;
254
261
  const version = node.constructInfo?.version;
@@ -258,7 +265,7 @@ function contextOverflowCleanup(location, assembly) {
258
265
  // We're dealing with an old version of the framework here. It is unaware of the temporary
259
266
  // file, which means that it will ignore the context overflow.
260
267
  if (frameworkDoesNotSupportContextOverflow) {
261
- (0, logging_1.warning)('Part of the context could not be sent to the application. Please update the AWS CDK library to the latest version.');
268
+ await ioHelper.notify(private_1.IO.DEFAULT_ASSEMBLY_WARN.msg('Part of the context could not be sent to the application. Please update the AWS CDK library to the latest version.'));
262
269
  }
263
270
  }
264
271
  }
@@ -269,4 +276,4 @@ function spaceAvailableForContext(env, limit) {
269
276
  .reduce((a, b) => a + b, 0);
270
277
  return Math.max(0, limit - usedSpace);
271
278
  }
272
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"exec.js","sourceRoot":"","sources":["exec.ts"],"names":[],"mappings":"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","sourcesContent":["import * as childProcess from 'child_process';\nimport * as os from 'os';\nimport * as path from 'path';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as fs from 'fs-extra';\nimport * as semver from 'semver';\nimport { ToolkitError } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api';\nimport { loadTree, some } from '../../api/tree';\nimport type { Configuration } from '../../cli/user-configuration';\nimport { PROJECT_CONFIG, USER_DEFAULTS } from '../../cli/user-configuration';\nimport { versionNumber } from '../../cli/version';\nimport { debug, trace, warning } from '../../logging';\nimport { splitBySize } from '../../util';\nimport type { SdkProvider } from '../aws-auth';\nimport type { Settings } from '../settings';\nimport type { ILock } from '../util/rwlock';\nimport { RWLock } from '../util/rwlock';\n\nexport interface ExecProgramResult {\n  readonly assembly: cxapi.CloudAssembly;\n  readonly lock: ILock;\n}\n\n/** Invokes the cloud executable and returns JSON output */\nexport async function execProgram(aws: SdkProvider, config: Configuration): Promise<ExecProgramResult> {\n  const env = await prepareDefaultEnvironment(aws);\n  const context = await prepareContext(config.settings, config.context.all, env);\n\n  const build = config.settings.get(['build']);\n  if (build) {\n    await exec(build);\n  }\n\n  const app = config.settings.get(['app']);\n  if (!app) {\n    throw new ToolkitError(`--app is required either in command-line, in ${PROJECT_CONFIG} or in ${USER_DEFAULTS}`);\n  }\n\n  // bypass \"synth\" if app points to a cloud assembly\n  if (await fs.pathExists(app) && (await fs.stat(app)).isDirectory()) {\n    debug('--app points to a cloud assembly, so we bypass synth');\n\n    // Acquire a read lock on this directory\n    const lock = await new RWLock(app).acquireRead();\n\n    return { assembly: createAssembly(app), lock };\n  }\n\n  const commandLine = await guessExecutable(app);\n\n  const outdir = config.settings.get(['output']);\n  if (!outdir) {\n    throw new ToolkitError('unexpected: --output is required');\n  }\n  if (typeof outdir !== 'string') {\n    throw new ToolkitError(`--output takes a string, got ${JSON.stringify(outdir)}`);\n  }\n  try {\n    await fs.mkdirp(outdir);\n  } catch (error: any) {\n    throw new ToolkitError(`Could not create output directory ${outdir} (${error.message})`);\n  }\n\n  debug('outdir:', outdir);\n  env[cxapi.OUTDIR_ENV] = outdir;\n\n  // Acquire a lock on the output directory\n  const writerLock = await new RWLock(outdir).acquireWrite();\n\n  try {\n    // Send version information\n    env[cxapi.CLI_ASM_VERSION_ENV] = cxschema.Manifest.version();\n    env[cxapi.CLI_VERSION_ENV] = versionNumber();\n\n    debug('env:', env);\n\n    const envVariableSizeLimit = os.platform() === 'win32' ? 32760 : 131072;\n    const [smallContext, overflow] = splitBySize(context, spaceAvailableForContext(env, envVariableSizeLimit));\n\n    // Store the safe part in the environment variable\n    env[cxapi.CONTEXT_ENV] = JSON.stringify(smallContext);\n\n    // If there was any overflow, write it to a temporary file\n    let contextOverflowLocation;\n    if (Object.keys(overflow ?? {}).length > 0) {\n      const contextDir = await fs.mkdtemp(path.join(os.tmpdir(), 'cdk-context'));\n      contextOverflowLocation = path.join(contextDir, 'context-overflow.json');\n      fs.writeJSONSync(contextOverflowLocation, overflow);\n      env[cxapi.CONTEXT_OVERFLOW_LOCATION_ENV] = contextOverflowLocation;\n    }\n\n    await exec(commandLine.join(' '));\n\n    const assembly = createAssembly(outdir);\n\n    contextOverflowCleanup(contextOverflowLocation, assembly);\n\n    return { assembly, lock: await writerLock.convertToReaderLock() };\n  } catch (e) {\n    await writerLock.release();\n    throw e;\n  }\n\n  async function exec(commandAndArgs: string) {\n    return new Promise<void>((ok, fail) => {\n      // We use a slightly lower-level interface to:\n      //\n      // - Pass arguments in an array instead of a string, to get around a\n      //   number of quoting issues introduced by the intermediate shell layer\n      //   (which would be different between Linux and Windows).\n      //\n      // - Inherit stderr from controlling terminal. We don't use the captured value\n      //   anyway, and if the subprocess is printing to it for debugging purposes the\n      //   user gets to see it sooner. Plus, capturing doesn't interact nicely with some\n      //   processes like Maven.\n      const proc = childProcess.spawn(commandAndArgs, {\n        stdio: ['ignore', 'inherit', 'inherit'],\n        detached: false,\n        shell: true,\n        env: {\n          ...process.env,\n          ...env,\n        },\n      });\n\n      proc.on('error', fail);\n\n      proc.on('exit', code => {\n        if (code === 0) {\n          return ok();\n        } else {\n          debug('failed command:', commandAndArgs);\n          return fail(new ToolkitError(`Subprocess exited with error ${code}`));\n        }\n      });\n    });\n  }\n}\n\n/**\n * Creates an assembly with error handling\n */\nexport function createAssembly(appDir: string) {\n  try {\n    return new cxapi.CloudAssembly(appDir, {\n      // We sort as we deploy\n      topoSort: false,\n    });\n  } catch (error: any) {\n    if (error.message.includes(cxschema.VERSION_MISMATCH)) {\n      // this means the CLI version is too old.\n      // we instruct the user to upgrade.\n      throw new ToolkitError(`This CDK CLI is not compatible with the CDK library used by your application. Please upgrade the CLI to the latest version.\\n(${error.message})`);\n    }\n    throw error;\n  }\n}\n\n/**\n * If we don't have region/account defined in context, we fall back to the default SDK behavior\n * where region is retrieved from ~/.aws/config and account is based on default credentials provider\n * chain and then STS is queried.\n *\n * This is done opportunistically: for example, if we can't access STS for some reason or the region\n * is not configured, the context value will be 'null' and there could failures down the line. In\n * some cases, synthesis does not require region/account information at all, so that might be perfectly\n * fine in certain scenarios.\n *\n * @param context The context key/value bash.\n */\nexport async function prepareDefaultEnvironment(\n  aws: SdkProvider,\n  logFn: (msg: string, ...args: any) => any = debug,\n): Promise<{ [key: string]: string }> {\n  const env: { [key: string]: string } = { };\n\n  env[cxapi.DEFAULT_REGION_ENV] = aws.defaultRegion;\n  await logFn(`Setting \"${cxapi.DEFAULT_REGION_ENV}\" environment variable to`, env[cxapi.DEFAULT_REGION_ENV]);\n\n  const accountId = (await aws.defaultAccount())?.accountId;\n  if (accountId) {\n    env[cxapi.DEFAULT_ACCOUNT_ENV] = accountId;\n    await logFn(`Setting \"${cxapi.DEFAULT_ACCOUNT_ENV}\" environment variable to`, env[cxapi.DEFAULT_ACCOUNT_ENV]);\n  }\n\n  return env;\n}\n\n/**\n * Settings related to synthesis are read from context.\n * The merging of various configuration sources like cli args or cdk.json has already happened.\n * We now need to set the final values to the context.\n */\nexport async function prepareContext(settings: Settings, context: {[key: string]: any}, env: { [key: string]: string | undefined}) {\n  const debugMode: boolean = settings.get(['debug']) ?? true;\n  if (debugMode) {\n    env.CDK_DEBUG = 'true';\n  }\n\n  const pathMetadata: boolean = settings.get(['pathMetadata']) ?? true;\n  if (pathMetadata) {\n    context[cxapi.PATH_METADATA_ENABLE_CONTEXT] = true;\n  }\n\n  const assetMetadata: boolean = settings.get(['assetMetadata']) ?? true;\n  if (assetMetadata) {\n    context[cxapi.ASSET_RESOURCE_METADATA_ENABLED_CONTEXT] = true;\n  }\n\n  const versionReporting: boolean = settings.get(['versionReporting']) ?? true;\n  if (versionReporting) {\n    context[cxapi.ANALYTICS_REPORTING_ENABLED_CONTEXT] = true;\n  }\n  // We need to keep on doing this for framework version from before this flag was deprecated.\n  if (!versionReporting) {\n    context['aws:cdk:disable-version-reporting'] = true;\n  }\n\n  const stagingEnabled = settings.get(['staging']) ?? true;\n  if (!stagingEnabled) {\n    context[cxapi.DISABLE_ASSET_STAGING_CONTEXT] = true;\n  }\n\n  const bundlingStacks = settings.get(['bundlingStacks']) ?? ['**'];\n  context[cxapi.BUNDLING_STACKS] = bundlingStacks;\n\n  debug('context:', context);\n\n  return context;\n}\n\n/**\n * Make sure the 'app' is an array\n *\n * If it's a string, split on spaces as a trivial way of tokenizing the command line.\n */\nfunction appToArray(app: any) {\n  return typeof app === 'string' ? app.split(' ') : app;\n}\n\ntype CommandGenerator = (file: string) => string[];\n\n/**\n * Execute the given file with the same 'node' process as is running the current process\n */\nfunction executeNode(scriptFile: string): string[] {\n  return [process.execPath, scriptFile];\n}\n\n/**\n * Mapping of extensions to command-line generators\n */\nconst EXTENSION_MAP = new Map<string, CommandGenerator>([\n  ['.js', executeNode],\n]);\n\n/**\n * Guess the executable from the command-line argument\n *\n * Only do this if the file is NOT marked as executable. If it is,\n * we'll defer to the shebang inside the file itself.\n *\n * If we're on Windows, we ALWAYS take the handler, since it's hard to\n * verify if registry associations have or have not been set up for this\n * file type, so we'll assume the worst and take control.\n */\nexport async function guessExecutable(app: string) {\n  const commandLine = appToArray(app);\n  if (commandLine.length === 1) {\n    let fstat;\n\n    try {\n      fstat = await fs.stat(commandLine[0]);\n    } catch {\n      debug(`Not a file: '${commandLine[0]}'. Using '${commandLine}' as command-line`);\n      return commandLine;\n    }\n\n    // eslint-disable-next-line no-bitwise\n    const isExecutable = (fstat.mode & fs.constants.X_OK) !== 0;\n    const isWindows = process.platform === 'win32';\n\n    const handler = EXTENSION_MAP.get(path.extname(commandLine[0]));\n    if (handler && (!isExecutable || isWindows)) {\n      return handler(commandLine[0]);\n    }\n  }\n  return commandLine;\n}\n\nfunction contextOverflowCleanup(location: string | undefined, assembly: cxapi.CloudAssembly) {\n  if (location) {\n    fs.removeSync(path.dirname(location));\n\n    const tree = loadTree(assembly, trace);\n    const frameworkDoesNotSupportContextOverflow = some(tree, node => {\n      const fqn = node.constructInfo?.fqn;\n      const version = node.constructInfo?.version;\n      return (fqn === 'aws-cdk-lib.App' && version != null && semver.lte(version, '2.38.0'))\n        || fqn === '@aws-cdk/core.App'; // v1\n    });\n\n    // We're dealing with an old version of the framework here. It is unaware of the temporary\n    // file, which means that it will ignore the context overflow.\n    if (frameworkDoesNotSupportContextOverflow) {\n      warning('Part of the context could not be sent to the application. Please update the AWS CDK library to the latest version.');\n    }\n  }\n}\n\nexport function spaceAvailableForContext(env: { [key: string]: string }, limit: number) {\n  const size = (value: string) => value != null ? Buffer.byteLength(value) : 0;\n\n  const usedSpace = Object.entries(env)\n    .map(([k, v]) => k === cxapi.CONTEXT_ENV ? size(k) : size(k) + size(v))\n    .reduce((a, b) => a + b, 0);\n\n  return Math.max(0, limit - usedSpace);\n}\n"]}
279
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"exec.js","sourceRoot":"","sources":["exec.ts"],"names":[],"mappings":"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","sourcesContent":["import * as childProcess from 'child_process';\nimport * as os from 'os';\nimport * as path from 'path';\nimport { format } from 'util';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as fs from 'fs-extra';\nimport * as semver from 'semver';\nimport { ToolkitError } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api';\nimport { IO, type IoHelper } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';\nimport { loadTree, some } from '../../api/tree';\nimport type { Configuration } from '../../cli/user-configuration';\nimport { PROJECT_CONFIG, USER_DEFAULTS } from '../../cli/user-configuration';\nimport { versionNumber } from '../../cli/version';\nimport { splitBySize } from '../../util';\nimport type { SdkProvider } from '../aws-auth';\nimport type { Settings } from '../settings';\nimport type { ILock } from '../util/rwlock';\nimport { RWLock } from '../util/rwlock';\n\nexport interface ExecProgramResult {\n  readonly assembly: cxapi.CloudAssembly;\n  readonly lock: ILock;\n}\n\n/** Invokes the cloud executable and returns JSON output */\nexport async function execProgram(aws: SdkProvider, ioHelper: IoHelper, config: Configuration): Promise<ExecProgramResult> {\n  const debugFn = (msg: string) => ioHelper.notify(IO.DEFAULT_ASSEMBLY_DEBUG.msg(msg));\n  const env = await prepareDefaultEnvironment(aws, debugFn);\n  const context = await prepareContext(config.settings, config.context.all, env, debugFn);\n\n  const build = config.settings.get(['build']);\n  if (build) {\n    await exec(build);\n  }\n\n  const app = config.settings.get(['app']);\n  if (!app) {\n    throw new ToolkitError(`--app is required either in command-line, in ${PROJECT_CONFIG} or in ${USER_DEFAULTS}`);\n  }\n\n  // bypass \"synth\" if app points to a cloud assembly\n  if (await fs.pathExists(app) && (await fs.stat(app)).isDirectory()) {\n    await debugFn('--app points to a cloud assembly, so we bypass synth');\n\n    // Acquire a read lock on this directory\n    const lock = await new RWLock(app).acquireRead();\n\n    return { assembly: createAssembly(app), lock };\n  }\n\n  const commandLine = await guessExecutable(app, debugFn);\n\n  const outdir = config.settings.get(['output']);\n  if (!outdir) {\n    throw new ToolkitError('unexpected: --output is required');\n  }\n  if (typeof outdir !== 'string') {\n    throw new ToolkitError(`--output takes a string, got ${JSON.stringify(outdir)}`);\n  }\n  try {\n    await fs.mkdirp(outdir);\n  } catch (error: any) {\n    throw new ToolkitError(`Could not create output directory ${outdir} (${error.message})`);\n  }\n\n  await debugFn(`outdir: ${outdir}`);\n  env[cxapi.OUTDIR_ENV] = outdir;\n\n  // Acquire a lock on the output directory\n  const writerLock = await new RWLock(outdir).acquireWrite();\n\n  try {\n    // Send version information\n    env[cxapi.CLI_ASM_VERSION_ENV] = cxschema.Manifest.version();\n    env[cxapi.CLI_VERSION_ENV] = versionNumber();\n\n    await debugFn(format('env:', env));\n\n    const envVariableSizeLimit = os.platform() === 'win32' ? 32760 : 131072;\n    const [smallContext, overflow] = splitBySize(context, spaceAvailableForContext(env, envVariableSizeLimit));\n\n    // Store the safe part in the environment variable\n    env[cxapi.CONTEXT_ENV] = JSON.stringify(smallContext);\n\n    // If there was any overflow, write it to a temporary file\n    let contextOverflowLocation;\n    if (Object.keys(overflow ?? {}).length > 0) {\n      const contextDir = await fs.mkdtemp(path.join(os.tmpdir(), 'cdk-context'));\n      contextOverflowLocation = path.join(contextDir, 'context-overflow.json');\n      fs.writeJSONSync(contextOverflowLocation, overflow);\n      env[cxapi.CONTEXT_OVERFLOW_LOCATION_ENV] = contextOverflowLocation;\n    }\n\n    await exec(commandLine.join(' '));\n\n    const assembly = createAssembly(outdir);\n\n    await contextOverflowCleanup(contextOverflowLocation, assembly, ioHelper);\n\n    return { assembly, lock: await writerLock.convertToReaderLock() };\n  } catch (e) {\n    await writerLock.release();\n    throw e;\n  }\n\n  async function exec(commandAndArgs: string) {\n    try {\n      await new Promise<void>((ok, fail) => {\n        // We use a slightly lower-level interface to:\n        //\n        // - Pass arguments in an array instead of a string, to get around a\n        //   number of quoting issues introduced by the intermediate shell layer\n        //   (which would be different between Linux and Windows).\n        //\n        // - Inherit stderr from controlling terminal. We don't use the captured value\n        //   anyway, and if the subprocess is printing to it for debugging purposes the\n        //   user gets to see it sooner. Plus, capturing doesn't interact nicely with some\n        //   processes like Maven.\n        const proc = childProcess.spawn(commandAndArgs, {\n          stdio: ['ignore', 'inherit', 'inherit'],\n          detached: false,\n          shell: true,\n          env: {\n            ...process.env,\n            ...env,\n          },\n        });\n\n        proc.on('error', fail);\n\n        proc.on('exit', code => {\n          if (code === 0) {\n            return ok();\n          } else {\n            return fail(new ToolkitError(`Subprocess exited with error ${code}`));\n          }\n        });\n      });\n    } catch (e: any) {\n      await debugFn(`failed command: ${commandAndArgs}`);\n      throw e;\n    }\n  }\n}\n\n/**\n * Creates an assembly with error handling\n */\nexport function createAssembly(appDir: string) {\n  try {\n    return new cxapi.CloudAssembly(appDir, {\n      // We sort as we deploy\n      topoSort: false,\n    });\n  } catch (error: any) {\n    if (error.message.includes(cxschema.VERSION_MISMATCH)) {\n      // this means the CLI version is too old.\n      // we instruct the user to upgrade.\n      throw new ToolkitError(`This CDK CLI is not compatible with the CDK library used by your application. Please upgrade the CLI to the latest version.\\n(${error.message})`);\n    }\n    throw error;\n  }\n}\n\n/**\n * If we don't have region/account defined in context, we fall back to the default SDK behavior\n * where region is retrieved from ~/.aws/config and account is based on default credentials provider\n * chain and then STS is queried.\n *\n * This is done opportunistically: for example, if we can't access STS for some reason or the region\n * is not configured, the context value will be 'null' and there could failures down the line. In\n * some cases, synthesis does not require region/account information at all, so that might be perfectly\n * fine in certain scenarios.\n *\n * @param context The context key/value bash.\n */\nexport async function prepareDefaultEnvironment(\n  aws: SdkProvider,\n  debugFn: (msg: string) => Promise<void>,\n): Promise<{ [key: string]: string }> {\n  const env: { [key: string]: string } = { };\n\n  env[cxapi.DEFAULT_REGION_ENV] = aws.defaultRegion;\n  await debugFn(`Setting \"${cxapi.DEFAULT_REGION_ENV}\" environment variable to ${env[cxapi.DEFAULT_REGION_ENV]}`);\n\n  const accountId = (await aws.defaultAccount())?.accountId;\n  if (accountId) {\n    env[cxapi.DEFAULT_ACCOUNT_ENV] = accountId;\n    await debugFn(`Setting \"${cxapi.DEFAULT_ACCOUNT_ENV}\" environment variable to ${env[cxapi.DEFAULT_ACCOUNT_ENV]}`);\n  }\n\n  return env;\n}\n\n/**\n * Settings related to synthesis are read from context.\n * The merging of various configuration sources like cli args or cdk.json has already happened.\n * We now need to set the final values to the context.\n */\nexport async function prepareContext(\n  settings: Settings,\n  context: {[key: string]: any},\n  env: { [key: string]: string | undefined},\n  debugFn: (msg: string) => Promise<void>,\n) {\n  const debugMode: boolean = settings.get(['debug']) ?? true;\n  if (debugMode) {\n    env.CDK_DEBUG = 'true';\n  }\n\n  const pathMetadata: boolean = settings.get(['pathMetadata']) ?? true;\n  if (pathMetadata) {\n    context[cxapi.PATH_METADATA_ENABLE_CONTEXT] = true;\n  }\n\n  const assetMetadata: boolean = settings.get(['assetMetadata']) ?? true;\n  if (assetMetadata) {\n    context[cxapi.ASSET_RESOURCE_METADATA_ENABLED_CONTEXT] = true;\n  }\n\n  const versionReporting: boolean = settings.get(['versionReporting']) ?? true;\n  if (versionReporting) {\n    context[cxapi.ANALYTICS_REPORTING_ENABLED_CONTEXT] = true;\n  }\n  // We need to keep on doing this for framework version from before this flag was deprecated.\n  if (!versionReporting) {\n    context['aws:cdk:disable-version-reporting'] = true;\n  }\n\n  const stagingEnabled = settings.get(['staging']) ?? true;\n  if (!stagingEnabled) {\n    context[cxapi.DISABLE_ASSET_STAGING_CONTEXT] = true;\n  }\n\n  const bundlingStacks = settings.get(['bundlingStacks']) ?? ['**'];\n  context[cxapi.BUNDLING_STACKS] = bundlingStacks;\n\n  await debugFn(format('context:', context));\n\n  return context;\n}\n\n/**\n * Make sure the 'app' is an array\n *\n * If it's a string, split on spaces as a trivial way of tokenizing the command line.\n */\nfunction appToArray(app: any) {\n  return typeof app === 'string' ? app.split(' ') : app;\n}\n\ntype CommandGenerator = (file: string) => string[];\n\n/**\n * Execute the given file with the same 'node' process as is running the current process\n */\nfunction executeNode(scriptFile: string): string[] {\n  return [process.execPath, scriptFile];\n}\n\n/**\n * Mapping of extensions to command-line generators\n */\nconst EXTENSION_MAP = new Map<string, CommandGenerator>([\n  ['.js', executeNode],\n]);\n\n/**\n * Guess the executable from the command-line argument\n *\n * Only do this if the file is NOT marked as executable. If it is,\n * we'll defer to the shebang inside the file itself.\n *\n * If we're on Windows, we ALWAYS take the handler, since it's hard to\n * verify if registry associations have or have not been set up for this\n * file type, so we'll assume the worst and take control.\n */\nexport async function guessExecutable(app: string, debugFn: (msg: string) => Promise<void>) {\n  const commandLine = appToArray(app);\n  if (commandLine.length === 1) {\n    let fstat;\n\n    try {\n      fstat = await fs.stat(commandLine[0]);\n    } catch {\n      await debugFn(`Not a file: '${commandLine[0]}'. Using '${commandLine}' as command-line`);\n      return commandLine;\n    }\n\n    // eslint-disable-next-line no-bitwise\n    const isExecutable = (fstat.mode & fs.constants.X_OK) !== 0;\n    const isWindows = process.platform === 'win32';\n\n    const handler = EXTENSION_MAP.get(path.extname(commandLine[0]));\n    if (handler && (!isExecutable || isWindows)) {\n      return handler(commandLine[0]);\n    }\n  }\n  return commandLine;\n}\n\nasync function contextOverflowCleanup(\n  location: string | undefined,\n  assembly: cxapi.CloudAssembly,\n  ioHelper: IoHelper,\n) {\n  if (location) {\n    fs.removeSync(path.dirname(location));\n\n    const tree = await loadTree(assembly, (msg: string) => ioHelper.notify(IO.DEFAULT_ASSEMBLY_TRACE.msg(msg)));\n    const frameworkDoesNotSupportContextOverflow = some(tree, node => {\n      const fqn = node.constructInfo?.fqn;\n      const version = node.constructInfo?.version;\n      return (fqn === 'aws-cdk-lib.App' && version != null && semver.lte(version, '2.38.0'))\n        || fqn === '@aws-cdk/core.App'; // v1\n    });\n\n    // We're dealing with an old version of the framework here. It is unaware of the temporary\n    // file, which means that it will ignore the context overflow.\n    if (frameworkDoesNotSupportContextOverflow) {\n      await ioHelper.notify(IO.DEFAULT_ASSEMBLY_WARN.msg('Part of the context could not be sent to the application. Please update the AWS CDK library to the latest version.'));\n    }\n  }\n}\n\nexport function spaceAvailableForContext(env: { [key: string]: string }, limit: number) {\n  const size = (value: string) => value != null ? Buffer.byteLength(value) : 0;\n\n  const usedSpace = Object.entries(env)\n    .map(([k, v]) => k === cxapi.CONTEXT_ENV ? size(k) : size(k) + size(v))\n    .reduce((a, b) => a + b, 0);\n\n  return Math.max(0, limit - usedSpace);\n}\n"]}
@@ -165,7 +165,7 @@ async function uploadBodyParameterAndCreateChangeSet(ioHelper, options) {
165
165
  });
166
166
  }
167
167
  catch (e) {
168
- await ioHelper.notify(private_1.IO.DEFAULT_TOOLKIT_DEBUG.msg(e));
168
+ await ioHelper.notify(private_1.IO.DEFAULT_TOOLKIT_DEBUG.msg(String(e)));
169
169
  await ioHelper.notify(private_1.IO.DEFAULT_TOOLKIT_INFO.msg('Could not create a change set, will base the diff on template differences (run again with -v to see the reason)\n'));
170
170
  return undefined;
171
171
  }
@@ -435,4 +435,4 @@ class ParameterValues {
435
435
  }
436
436
  }
437
437
  exports.ParameterValues = ParameterValues;
438
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cfn-api.js","sourceRoot":"","sources":["cfn-api.ts"],"names":[],"mappings":"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","sourcesContent":["import { format } from 'util';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport { SSMPARAM_NO_INVALIDATE } from '@aws-cdk/cx-api';\nimport type {\n  DescribeChangeSetCommandOutput,\n  Parameter,\n  ResourceToImport,\n  Tag,\n} from '@aws-sdk/client-cloudformation';\nimport {\n  ChangeSetStatus,\n} from '@aws-sdk/client-cloudformation';\nimport type { FileManifestEntry } from 'cdk-assets';\nimport { AssetManifest } from 'cdk-assets';\nimport { AssetManifestBuilder } from './asset-manifest-builder';\nimport type { Deployments } from './deployments';\nimport { ToolkitError } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api';\nimport { IO, type IoHelper } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';\nimport type { ICloudFormationClient, SdkProvider } from '../aws-auth';\nimport type { Template, TemplateBodyParameter, TemplateParameter } from '../cloudformation';\nimport { CloudFormationStack, makeBodyParameter } from '../cloudformation';\nimport type { ResourcesToImport } from '../resource-import';\n\n/**\n * Describe a changeset in CloudFormation, regardless of its current state.\n *\n * @param cfn           a CloudFormation client\n * @param stackName     the name of the Stack the ChangeSet belongs to\n * @param changeSetName the name of the ChangeSet\n * @param fetchAll      if true, fetches all pages of the change set description.\n *\n * @returns       CloudFormation information about the ChangeSet\n */\nasync function describeChangeSet(\n  cfn: ICloudFormationClient,\n  stackName: string,\n  changeSetName: string,\n  { fetchAll }: { fetchAll: boolean },\n): Promise<DescribeChangeSetCommandOutput> {\n  const response = await cfn.describeChangeSet({\n    StackName: stackName,\n    ChangeSetName: changeSetName,\n  });\n\n  // If fetchAll is true, traverse all pages from the change set description.\n  while (fetchAll && response.NextToken != null) {\n    const nextPage = await cfn.describeChangeSet({\n      StackName: stackName,\n      ChangeSetName: response.ChangeSetId ?? changeSetName,\n      NextToken: response.NextToken,\n    });\n\n    // Consolidate the changes\n    if (nextPage.Changes != null) {\n      response.Changes = response.Changes != null ? response.Changes.concat(nextPage.Changes) : nextPage.Changes;\n    }\n\n    // Forward the new NextToken\n    response.NextToken = nextPage.NextToken;\n  }\n\n  return response;\n}\n\n/**\n * Waits for a function to return non-+undefined+ before returning.\n *\n * @param valueProvider a function that will return a value that is not +undefined+ once the wait should be over\n * @param timeout     the time to wait between two calls to +valueProvider+\n *\n * @returns       the value that was returned by +valueProvider+\n */\nasync function waitFor<T>(\n  valueProvider: () => Promise<T | null | undefined>,\n  timeout: number = 5000,\n): Promise<T | undefined> {\n  while (true) {\n    const result = await valueProvider();\n    if (result === null) {\n      return undefined;\n    } else if (result !== undefined) {\n      return result;\n    }\n    await new Promise((cb) => setTimeout(cb, timeout));\n  }\n}\n\n/**\n * Waits for a ChangeSet to be available for triggering a StackUpdate.\n *\n * Will return a changeset that is either ready to be executed or has no changes.\n * Will throw in other cases.\n *\n * @param cfn           a CloudFormation client\n * @param stackName     the name of the Stack that the ChangeSet belongs to\n * @param changeSetName the name of the ChangeSet\n * @param fetchAll      if true, fetches all pages of the ChangeSet before returning.\n *\n * @returns       the CloudFormation description of the ChangeSet\n */\nexport async function waitForChangeSet(\n  cfn: ICloudFormationClient,\n  ioHelper: IoHelper,\n  stackName: string,\n  changeSetName: string,\n  { fetchAll }: { fetchAll: boolean },\n): Promise<DescribeChangeSetCommandOutput> {\n  await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(format('Waiting for changeset %s on stack %s to finish creating...', changeSetName, stackName)));\n  const ret = await waitFor(async () => {\n    const description = await describeChangeSet(cfn, stackName, changeSetName, {\n      fetchAll,\n    });\n    // The following doesn't use a switch because tsc will not allow fall-through, UNLESS it is allows\n    // EVERYWHERE that uses this library directly or indirectly, which is undesirable.\n    if (description.Status === 'CREATE_PENDING' || description.Status === 'CREATE_IN_PROGRESS') {\n      await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(format('Changeset %s on stack %s is still creating', changeSetName, stackName)));\n      return undefined;\n    }\n\n    if (description.Status === ChangeSetStatus.CREATE_COMPLETE || changeSetHasNoChanges(description)) {\n      return description;\n    }\n\n    // eslint-disable-next-line max-len\n    throw new ToolkitError(\n      `Failed to create ChangeSet ${changeSetName} on ${stackName}: ${description.Status || 'NO_STATUS'}, ${description.StatusReason || 'no reason provided'}`,\n    );\n  });\n\n  if (!ret) {\n    throw new ToolkitError('Change set took too long to be created; aborting');\n  }\n\n  return ret;\n}\n\nexport type PrepareChangeSetOptions = {\n  stack: cxapi.CloudFormationStackArtifact;\n  deployments: Deployments;\n  uuid: string;\n  willExecute: boolean;\n  sdkProvider: SdkProvider;\n  parameters: { [name: string]: string | undefined };\n  resourcesToImport?: ResourcesToImport;\n}\n\nexport type CreateChangeSetOptions = {\n  cfn: ICloudFormationClient;\n  changeSetName: string;\n  willExecute: boolean;\n  exists: boolean;\n  uuid: string;\n  stack: cxapi.CloudFormationStackArtifact;\n  bodyParameter: TemplateBodyParameter;\n  parameters: { [name: string]: string | undefined };\n  resourcesToImport?: ResourceToImport[];\n  role?: string;\n};\n\n/**\n * Create a changeset for a diff operation\n */\nexport async function createDiffChangeSet(\n  ioHelper: IoHelper,\n  options: PrepareChangeSetOptions,\n): Promise<DescribeChangeSetCommandOutput | undefined> {\n  // `options.stack` has been modified to include any nested stack templates directly inline with its own template, under a special `NestedTemplate` property.\n  // Thus the parent template's Resources section contains the nested template's CDK metadata check, which uses Fn::Equals.\n  // This causes CreateChangeSet to fail with `Template Error: Fn::Equals cannot be partially collapsed`.\n  for (const resource of Object.values(options.stack.template.Resources ?? {})) {\n    if ((resource as any).Type === 'AWS::CloudFormation::Stack') {\n      await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg('This stack contains one or more nested stacks, falling back to template-only diff...'));\n\n      return undefined;\n    }\n  }\n\n  return uploadBodyParameterAndCreateChangeSet(ioHelper, options);\n}\n\n/**\n * Returns all file entries from an AssetManifestArtifact that look like templates.\n *\n * This is used in the `uploadBodyParameterAndCreateChangeSet` function to find\n * all template asset files to build and publish.\n *\n * Returns a tuple of [AssetManifest, FileManifestEntry[]]\n */\nfunction templatesFromAssetManifestArtifact(\n  artifact: cxapi.AssetManifestArtifact,\n): [AssetManifest, FileManifestEntry[]] {\n  const assets: FileManifestEntry[] = [];\n  const fileName = artifact.file;\n  const assetManifest = AssetManifest.fromFile(fileName);\n\n  assetManifest.entries.forEach((entry) => {\n    if (entry.type === 'file') {\n      const source = (entry as FileManifestEntry).source;\n      if (source.path && source.path.endsWith('.template.json')) {\n        assets.push(entry as FileManifestEntry);\n      }\n    }\n  });\n  return [assetManifest, assets];\n}\n\nasync function uploadBodyParameterAndCreateChangeSet(\n  ioHelper: IoHelper,\n  options: PrepareChangeSetOptions,\n): Promise<DescribeChangeSetCommandOutput | undefined> {\n  try {\n    await uploadStackTemplateAssets(options.stack, options.deployments);\n    const env = await options.deployments.envs.accessStackForMutableStackOperations(options.stack);\n\n    const bodyParameter = await makeBodyParameter(\n      ioHelper,\n      options.stack,\n      env.resolvedEnvironment,\n      new AssetManifestBuilder(),\n      env.resources,\n    );\n    const cfn = env.sdk.cloudFormation();\n    const exists = (await CloudFormationStack.lookup(cfn, options.stack.stackName, false)).exists;\n\n    const executionRoleArn = await env.replacePlaceholders(options.stack.cloudFormationExecutionRoleArn);\n    await ioHelper.notify(IO.DEFAULT_TOOLKIT_INFO.msg(\n      'Hold on while we create a read-only change set to get a diff with accurate replacement information (use --no-change-set to use a less accurate but faster template-only diff)\\n',\n    ));\n\n    return await createChangeSet(ioHelper, {\n      cfn,\n      changeSetName: 'cdk-diff-change-set',\n      stack: options.stack,\n      exists,\n      uuid: options.uuid,\n      willExecute: options.willExecute,\n      bodyParameter,\n      parameters: options.parameters,\n      resourcesToImport: options.resourcesToImport,\n      role: executionRoleArn,\n    });\n  } catch (e: any) {\n    await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(e));\n    await ioHelper.notify(IO.DEFAULT_TOOLKIT_INFO.msg(\n      'Could not create a change set, will base the diff on template differences (run again with -v to see the reason)\\n',\n    ));\n\n    return undefined;\n  }\n}\n\n/**\n * Uploads the assets that look like templates for this CloudFormation stack\n *\n * This is necessary for any CloudFormation call that needs the template, it may need\n * to be uploaded to an S3 bucket first. We have to follow the instructions in the\n * asset manifest, because technically that is the only place that knows about\n * bucket and assumed roles and such.\n */\nexport async function uploadStackTemplateAssets(stack: cxapi.CloudFormationStackArtifact, deployments: Deployments) {\n  for (const artifact of stack.dependencies) {\n    // Skip artifact if it is not an Asset Manifest Artifact\n    if (!cxapi.AssetManifestArtifact.isAssetManifestArtifact(artifact)) {\n      continue;\n    }\n\n    const [assetManifest, file_entries] = templatesFromAssetManifestArtifact(artifact);\n    for (const entry of file_entries) {\n      await deployments.buildSingleAsset(artifact, assetManifest, entry, {\n        stack,\n      });\n      await deployments.publishSingleAsset(assetManifest, entry, {\n        stack,\n      });\n    }\n  }\n}\n\nexport async function createChangeSet(\n  ioHelper: IoHelper,\n  options: CreateChangeSetOptions,\n): Promise<DescribeChangeSetCommandOutput> {\n  await cleanupOldChangeset(options.cfn, ioHelper, options.changeSetName, options.stack.stackName);\n\n  await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(`Attempting to create ChangeSet with name ${options.changeSetName} for stack ${options.stack.stackName}`));\n\n  const templateParams = TemplateParameters.fromTemplate(options.stack.template);\n  const stackParams = templateParams.supplyAll(options.parameters);\n\n  const changeSet = await options.cfn.createChangeSet({\n    StackName: options.stack.stackName,\n    ChangeSetName: options.changeSetName,\n    ChangeSetType: options.resourcesToImport ? 'IMPORT' : options.exists ? 'UPDATE' : 'CREATE',\n    Description: `CDK Changeset for diff ${options.uuid}`,\n    ClientToken: `diff${options.uuid}`,\n    TemplateURL: options.bodyParameter.TemplateURL,\n    TemplateBody: options.bodyParameter.TemplateBody,\n    Parameters: stackParams.apiParameters,\n    ResourcesToImport: options.resourcesToImport,\n    RoleARN: options.role,\n    Tags: toCfnTags(options.stack.tags),\n    Capabilities: ['CAPABILITY_IAM', 'CAPABILITY_NAMED_IAM', 'CAPABILITY_AUTO_EXPAND'],\n  });\n\n  await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(format('Initiated creation of changeset: %s; waiting for it to finish creating...', changeSet.Id)));\n  // Fetching all pages if we'll execute, so we can have the correct change count when monitoring.\n  const createdChangeSet = await waitForChangeSet(options.cfn, ioHelper, options.stack.stackName, options.changeSetName, {\n    fetchAll: options.willExecute,\n  });\n  await cleanupOldChangeset(options.cfn, ioHelper, options.changeSetName, options.stack.stackName);\n\n  return createdChangeSet;\n}\n\nfunction toCfnTags(tags: { [id: string]: string }): Tag[] {\n  return Object.entries(tags).map(([k, v]) => ({\n    Key: k,\n    Value: v,\n  }));\n}\n\nasync function cleanupOldChangeset(\n  cfn: ICloudFormationClient,\n  ioHelper: IoHelper,\n  changeSetName: string,\n  stackName: string,\n) {\n  // Delete any existing change sets generated by CDK since change set names must be unique.\n  // The delete request is successful as long as the stack exists (even if the change set does not exist).\n  await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(`Removing existing change set with name ${changeSetName} if it exists`));\n  await cfn.deleteChangeSet({\n    StackName: stackName,\n    ChangeSetName: changeSetName,\n  });\n}\n\n/**\n * Return true if the given change set has no changes\n *\n * This must be determined from the status, not the 'Changes' array on the\n * object; the latter can be empty because no resources were changed, but if\n * there are changes to Outputs, the change set can still be executed.\n */\nexport function changeSetHasNoChanges(description: DescribeChangeSetCommandOutput) {\n  const noChangeErrorPrefixes = [\n    // Error message for a regular template\n    \"The submitted information didn't contain changes.\",\n    // Error message when a Transform is involved (see #10650)\n    'No updates are to be performed.',\n  ];\n\n  return (\n    description.Status === 'FAILED' && noChangeErrorPrefixes.some((p) => (description.StatusReason ?? '').startsWith(p))\n  );\n}\n\n/**\n * Waits for a CloudFormation stack to stabilize in a complete/available state\n * after a delete operation is issued.\n *\n * Fails if the stack is in a FAILED state. Will not fail if the stack was\n * already deleted.\n *\n * @param cfn        a CloudFormation client\n * @param stackName      the name of the stack to wait for after a delete\n *\n * @returns     the CloudFormation description of the stabilized stack after the delete attempt\n */\nexport async function waitForStackDelete(\n  cfn: ICloudFormationClient,\n  ioHelper: IoHelper,\n  stackName: string,\n): Promise<CloudFormationStack | undefined> {\n  const stack = await stabilizeStack(cfn, ioHelper, stackName);\n  if (!stack) {\n    return undefined;\n  }\n\n  const status = stack.stackStatus;\n  if (status.isFailure) {\n    throw new ToolkitError(\n      `The stack named ${stackName} is in a failed state. You may need to delete it from the AWS console : ${status}`,\n    );\n  } else if (status.isDeleted) {\n    return undefined;\n  }\n  return stack;\n}\n\n/**\n * Waits for a CloudFormation stack to stabilize in a complete/available state\n * after an update/create operation is issued.\n *\n * Fails if the stack is in a FAILED state, ROLLBACK state, or DELETED state.\n *\n * @param cfn        a CloudFormation client\n * @param stackName      the name of the stack to wait for after an update\n *\n * @returns     the CloudFormation description of the stabilized stack after the update attempt\n */\nexport async function waitForStackDeploy(\n  cfn: ICloudFormationClient,\n  ioHelper: IoHelper,\n  stackName: string,\n): Promise<CloudFormationStack | undefined> {\n  const stack = await stabilizeStack(cfn, ioHelper, stackName);\n  if (!stack) {\n    return undefined;\n  }\n\n  const status = stack.stackStatus;\n\n  if (status.isCreationFailure) {\n    throw new ToolkitError(\n      `The stack named ${stackName} failed creation, it may need to be manually deleted from the AWS console: ${status}`,\n    );\n  } else if (!status.isDeploySuccess) {\n    throw new ToolkitError(`The stack named ${stackName} failed to deploy: ${status}`);\n  }\n\n  return stack;\n}\n\n/**\n * Wait for a stack to become stable (no longer _IN_PROGRESS), returning it\n */\nexport async function stabilizeStack(\n  cfn: ICloudFormationClient,\n  ioHelper: IoHelper,\n  stackName: string,\n) {\n  await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(format('Waiting for stack %s to finish creating or updating...', stackName)));\n  return waitFor(async () => {\n    const stack = await CloudFormationStack.lookup(cfn, stackName);\n    if (!stack.exists) {\n      await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(format('Stack %s does not exist', stackName)));\n      return null;\n    }\n    const status = stack.stackStatus;\n    if (status.isInProgress) {\n      await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(format('Stack %s has an ongoing operation in progress and is not stable (%s)', stackName, status)));\n      return undefined;\n    } else if (status.isReviewInProgress) {\n      // This may happen if a stack creation operation is interrupted before the ChangeSet execution starts. Recovering\n      // from this would requiring manual intervention (deleting or executing the pending ChangeSet), and failing to do\n      // so will result in an endless wait here (the ChangeSet wont delete or execute itself). Instead of blocking\n      // \"forever\" we proceed as if the stack was existing and stable. If there is a concurrent operation that just\n      // hasn't finished proceeding just yet, either this operation or the concurrent one may fail due to the other one\n      // having made progress. Which is fine. I guess.\n      await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(format('Stack %s is in REVIEW_IN_PROGRESS state. Considering this is a stable status (%s)', stackName, status)));\n    }\n\n    return stack;\n  });\n}\n\n/**\n * The set of (formal) parameters that have been declared in a template\n */\nexport class TemplateParameters {\n  public static fromTemplate(template: Template) {\n    return new TemplateParameters(template.Parameters || {});\n  }\n\n  constructor(private readonly params: Record<string, TemplateParameter>) {\n  }\n\n  /**\n   * Calculate stack parameters to pass from the given desired parameter values\n   *\n   * Will throw if parameters without a Default value or a Previous value are not\n   * supplied.\n   */\n  public supplyAll(updates: Record<string, string | undefined>): ParameterValues {\n    return new ParameterValues(this.params, updates);\n  }\n\n  /**\n   * From the template, the given desired values and the current values, calculate the changes to the stack parameters\n   *\n   * Will take into account parameters already set on the template (will emit\n   * 'UsePreviousValue: true' for those unless the value is changed), and will\n   * throw if parameters without a Default value or a Previous value are not\n   * supplied.\n   */\n  public updateExisting(\n    updates: Record<string, string | undefined>,\n    previousValues: Record<string, string>,\n  ): ParameterValues {\n    return new ParameterValues(this.params, updates, previousValues);\n  }\n}\n\n/**\n * The set of parameters we're going to pass to a Stack\n */\nexport class ParameterValues {\n  public readonly values: Record<string, string> = {};\n  public readonly apiParameters: Parameter[] = [];\n\n  constructor(\n    private readonly formalParams: Record<string, TemplateParameter>,\n    updates: Record<string, string | undefined>,\n    previousValues: Record<string, string> = {},\n  ) {\n    const missingRequired = new Array<string>();\n\n    for (const [key, formalParam] of Object.entries(this.formalParams)) {\n      // Check updates first, then use the previous value (if available), then use\n      // the default (if available).\n      //\n      // If we don't find a parameter value using any of these methods, then that's an error.\n      const updatedValue = updates[key];\n      if (updatedValue !== undefined) {\n        this.values[key] = updatedValue;\n        this.apiParameters.push({\n          ParameterKey: key,\n          ParameterValue: updates[key],\n        });\n        continue;\n      }\n\n      if (key in previousValues) {\n        this.values[key] = previousValues[key];\n        this.apiParameters.push({ ParameterKey: key, UsePreviousValue: true });\n        continue;\n      }\n\n      if (formalParam.Default !== undefined) {\n        this.values[key] = formalParam.Default;\n        continue;\n      }\n\n      // Oh no\n      missingRequired.push(key);\n    }\n\n    if (missingRequired.length > 0) {\n      throw new ToolkitError(`The following CloudFormation Parameters are missing a value: ${missingRequired.join(', ')}`);\n    }\n\n    // Just append all supplied overrides that aren't really expected (this\n    // will fail CFN but maybe people made typos that they want to be notified\n    // of)\n    const unknownParam = ([key, _]: [string, any]) => this.formalParams[key] === undefined;\n    const hasValue = ([_, value]: [string, any]) => !!value;\n    for (const [key, value] of Object.entries(updates).filter(unknownParam).filter(hasValue)) {\n      this.values[key] = value!;\n      this.apiParameters.push({ ParameterKey: key, ParameterValue: value });\n    }\n  }\n\n  /**\n   * Whether this set of parameter updates will change the actual stack values\n   */\n  public hasChanges(currentValues: Record<string, string>): ParameterChanges {\n    // If any of the parameters are SSM parameters, deploying must always happen\n    // because we can't predict what the values will be. We will allow some\n    // parameters to opt out of this check by having a magic string in their description.\n    if (\n      Object.values(this.formalParams).some(\n        (p) => p.Type.startsWith('AWS::SSM::Parameter::') && !p.Description?.includes(SSMPARAM_NO_INVALIDATE),\n      )\n    ) {\n      return 'ssm';\n    }\n\n    // Otherwise we're dirty if:\n    // - any of the existing values are removed, or changed\n    if (Object.entries(currentValues).some(([key, value]) => !(key in this.values) || value !== this.values[key])) {\n      return true;\n    }\n\n    // - any of the values we're setting are new\n    if (Object.keys(this.values).some((key) => !(key in currentValues))) {\n      return true;\n    }\n\n    return false;\n  }\n}\n\nexport type ParameterChanges = boolean | 'ssm';\n"]}
438
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cfn-api.js","sourceRoot":"","sources":["cfn-api.ts"],"names":[],"mappings":"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","sourcesContent":["import { format } from 'util';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport { SSMPARAM_NO_INVALIDATE } from '@aws-cdk/cx-api';\nimport type {\n  DescribeChangeSetCommandOutput,\n  Parameter,\n  ResourceToImport,\n  Tag,\n} from '@aws-sdk/client-cloudformation';\nimport {\n  ChangeSetStatus,\n} from '@aws-sdk/client-cloudformation';\nimport type { FileManifestEntry } from 'cdk-assets';\nimport { AssetManifest } from 'cdk-assets';\nimport { AssetManifestBuilder } from './asset-manifest-builder';\nimport type { Deployments } from './deployments';\nimport { ToolkitError } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api';\nimport { IO, type IoHelper } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';\nimport type { ICloudFormationClient, SdkProvider } from '../aws-auth';\nimport type { Template, TemplateBodyParameter, TemplateParameter } from '../cloudformation';\nimport { CloudFormationStack, makeBodyParameter } from '../cloudformation';\nimport type { ResourcesToImport } from '../resource-import';\n\n/**\n * Describe a changeset in CloudFormation, regardless of its current state.\n *\n * @param cfn           a CloudFormation client\n * @param stackName     the name of the Stack the ChangeSet belongs to\n * @param changeSetName the name of the ChangeSet\n * @param fetchAll      if true, fetches all pages of the change set description.\n *\n * @returns       CloudFormation information about the ChangeSet\n */\nasync function describeChangeSet(\n  cfn: ICloudFormationClient,\n  stackName: string,\n  changeSetName: string,\n  { fetchAll }: { fetchAll: boolean },\n): Promise<DescribeChangeSetCommandOutput> {\n  const response = await cfn.describeChangeSet({\n    StackName: stackName,\n    ChangeSetName: changeSetName,\n  });\n\n  // If fetchAll is true, traverse all pages from the change set description.\n  while (fetchAll && response.NextToken != null) {\n    const nextPage = await cfn.describeChangeSet({\n      StackName: stackName,\n      ChangeSetName: response.ChangeSetId ?? changeSetName,\n      NextToken: response.NextToken,\n    });\n\n    // Consolidate the changes\n    if (nextPage.Changes != null) {\n      response.Changes = response.Changes != null ? response.Changes.concat(nextPage.Changes) : nextPage.Changes;\n    }\n\n    // Forward the new NextToken\n    response.NextToken = nextPage.NextToken;\n  }\n\n  return response;\n}\n\n/**\n * Waits for a function to return non-+undefined+ before returning.\n *\n * @param valueProvider a function that will return a value that is not +undefined+ once the wait should be over\n * @param timeout     the time to wait between two calls to +valueProvider+\n *\n * @returns       the value that was returned by +valueProvider+\n */\nasync function waitFor<T>(\n  valueProvider: () => Promise<T | null | undefined>,\n  timeout: number = 5000,\n): Promise<T | undefined> {\n  while (true) {\n    const result = await valueProvider();\n    if (result === null) {\n      return undefined;\n    } else if (result !== undefined) {\n      return result;\n    }\n    await new Promise((cb) => setTimeout(cb, timeout));\n  }\n}\n\n/**\n * Waits for a ChangeSet to be available for triggering a StackUpdate.\n *\n * Will return a changeset that is either ready to be executed or has no changes.\n * Will throw in other cases.\n *\n * @param cfn           a CloudFormation client\n * @param stackName     the name of the Stack that the ChangeSet belongs to\n * @param changeSetName the name of the ChangeSet\n * @param fetchAll      if true, fetches all pages of the ChangeSet before returning.\n *\n * @returns       the CloudFormation description of the ChangeSet\n */\nexport async function waitForChangeSet(\n  cfn: ICloudFormationClient,\n  ioHelper: IoHelper,\n  stackName: string,\n  changeSetName: string,\n  { fetchAll }: { fetchAll: boolean },\n): Promise<DescribeChangeSetCommandOutput> {\n  await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(format('Waiting for changeset %s on stack %s to finish creating...', changeSetName, stackName)));\n  const ret = await waitFor(async () => {\n    const description = await describeChangeSet(cfn, stackName, changeSetName, {\n      fetchAll,\n    });\n    // The following doesn't use a switch because tsc will not allow fall-through, UNLESS it is allows\n    // EVERYWHERE that uses this library directly or indirectly, which is undesirable.\n    if (description.Status === 'CREATE_PENDING' || description.Status === 'CREATE_IN_PROGRESS') {\n      await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(format('Changeset %s on stack %s is still creating', changeSetName, stackName)));\n      return undefined;\n    }\n\n    if (description.Status === ChangeSetStatus.CREATE_COMPLETE || changeSetHasNoChanges(description)) {\n      return description;\n    }\n\n    // eslint-disable-next-line max-len\n    throw new ToolkitError(\n      `Failed to create ChangeSet ${changeSetName} on ${stackName}: ${description.Status || 'NO_STATUS'}, ${description.StatusReason || 'no reason provided'}`,\n    );\n  });\n\n  if (!ret) {\n    throw new ToolkitError('Change set took too long to be created; aborting');\n  }\n\n  return ret;\n}\n\nexport type PrepareChangeSetOptions = {\n  stack: cxapi.CloudFormationStackArtifact;\n  deployments: Deployments;\n  uuid: string;\n  willExecute: boolean;\n  sdkProvider: SdkProvider;\n  parameters: { [name: string]: string | undefined };\n  resourcesToImport?: ResourcesToImport;\n}\n\nexport type CreateChangeSetOptions = {\n  cfn: ICloudFormationClient;\n  changeSetName: string;\n  willExecute: boolean;\n  exists: boolean;\n  uuid: string;\n  stack: cxapi.CloudFormationStackArtifact;\n  bodyParameter: TemplateBodyParameter;\n  parameters: { [name: string]: string | undefined };\n  resourcesToImport?: ResourceToImport[];\n  role?: string;\n};\n\n/**\n * Create a changeset for a diff operation\n */\nexport async function createDiffChangeSet(\n  ioHelper: IoHelper,\n  options: PrepareChangeSetOptions,\n): Promise<DescribeChangeSetCommandOutput | undefined> {\n  // `options.stack` has been modified to include any nested stack templates directly inline with its own template, under a special `NestedTemplate` property.\n  // Thus the parent template's Resources section contains the nested template's CDK metadata check, which uses Fn::Equals.\n  // This causes CreateChangeSet to fail with `Template Error: Fn::Equals cannot be partially collapsed`.\n  for (const resource of Object.values(options.stack.template.Resources ?? {})) {\n    if ((resource as any).Type === 'AWS::CloudFormation::Stack') {\n      await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg('This stack contains one or more nested stacks, falling back to template-only diff...'));\n\n      return undefined;\n    }\n  }\n\n  return uploadBodyParameterAndCreateChangeSet(ioHelper, options);\n}\n\n/**\n * Returns all file entries from an AssetManifestArtifact that look like templates.\n *\n * This is used in the `uploadBodyParameterAndCreateChangeSet` function to find\n * all template asset files to build and publish.\n *\n * Returns a tuple of [AssetManifest, FileManifestEntry[]]\n */\nfunction templatesFromAssetManifestArtifact(\n  artifact: cxapi.AssetManifestArtifact,\n): [AssetManifest, FileManifestEntry[]] {\n  const assets: FileManifestEntry[] = [];\n  const fileName = artifact.file;\n  const assetManifest = AssetManifest.fromFile(fileName);\n\n  assetManifest.entries.forEach((entry) => {\n    if (entry.type === 'file') {\n      const source = (entry as FileManifestEntry).source;\n      if (source.path && source.path.endsWith('.template.json')) {\n        assets.push(entry as FileManifestEntry);\n      }\n    }\n  });\n  return [assetManifest, assets];\n}\n\nasync function uploadBodyParameterAndCreateChangeSet(\n  ioHelper: IoHelper,\n  options: PrepareChangeSetOptions,\n): Promise<DescribeChangeSetCommandOutput | undefined> {\n  try {\n    await uploadStackTemplateAssets(options.stack, options.deployments);\n    const env = await options.deployments.envs.accessStackForMutableStackOperations(options.stack);\n\n    const bodyParameter = await makeBodyParameter(\n      ioHelper,\n      options.stack,\n      env.resolvedEnvironment,\n      new AssetManifestBuilder(),\n      env.resources,\n    );\n    const cfn = env.sdk.cloudFormation();\n    const exists = (await CloudFormationStack.lookup(cfn, options.stack.stackName, false)).exists;\n\n    const executionRoleArn = await env.replacePlaceholders(options.stack.cloudFormationExecutionRoleArn);\n    await ioHelper.notify(IO.DEFAULT_TOOLKIT_INFO.msg(\n      'Hold on while we create a read-only change set to get a diff with accurate replacement information (use --no-change-set to use a less accurate but faster template-only diff)\\n',\n    ));\n\n    return await createChangeSet(ioHelper, {\n      cfn,\n      changeSetName: 'cdk-diff-change-set',\n      stack: options.stack,\n      exists,\n      uuid: options.uuid,\n      willExecute: options.willExecute,\n      bodyParameter,\n      parameters: options.parameters,\n      resourcesToImport: options.resourcesToImport,\n      role: executionRoleArn,\n    });\n  } catch (e: any) {\n    await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(String(e)));\n    await ioHelper.notify(IO.DEFAULT_TOOLKIT_INFO.msg(\n      'Could not create a change set, will base the diff on template differences (run again with -v to see the reason)\\n',\n    ));\n\n    return undefined;\n  }\n}\n\n/**\n * Uploads the assets that look like templates for this CloudFormation stack\n *\n * This is necessary for any CloudFormation call that needs the template, it may need\n * to be uploaded to an S3 bucket first. We have to follow the instructions in the\n * asset manifest, because technically that is the only place that knows about\n * bucket and assumed roles and such.\n */\nexport async function uploadStackTemplateAssets(stack: cxapi.CloudFormationStackArtifact, deployments: Deployments) {\n  for (const artifact of stack.dependencies) {\n    // Skip artifact if it is not an Asset Manifest Artifact\n    if (!cxapi.AssetManifestArtifact.isAssetManifestArtifact(artifact)) {\n      continue;\n    }\n\n    const [assetManifest, file_entries] = templatesFromAssetManifestArtifact(artifact);\n    for (const entry of file_entries) {\n      await deployments.buildSingleAsset(artifact, assetManifest, entry, {\n        stack,\n      });\n      await deployments.publishSingleAsset(assetManifest, entry, {\n        stack,\n      });\n    }\n  }\n}\n\nexport async function createChangeSet(\n  ioHelper: IoHelper,\n  options: CreateChangeSetOptions,\n): Promise<DescribeChangeSetCommandOutput> {\n  await cleanupOldChangeset(options.cfn, ioHelper, options.changeSetName, options.stack.stackName);\n\n  await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(`Attempting to create ChangeSet with name ${options.changeSetName} for stack ${options.stack.stackName}`));\n\n  const templateParams = TemplateParameters.fromTemplate(options.stack.template);\n  const stackParams = templateParams.supplyAll(options.parameters);\n\n  const changeSet = await options.cfn.createChangeSet({\n    StackName: options.stack.stackName,\n    ChangeSetName: options.changeSetName,\n    ChangeSetType: options.resourcesToImport ? 'IMPORT' : options.exists ? 'UPDATE' : 'CREATE',\n    Description: `CDK Changeset for diff ${options.uuid}`,\n    ClientToken: `diff${options.uuid}`,\n    TemplateURL: options.bodyParameter.TemplateURL,\n    TemplateBody: options.bodyParameter.TemplateBody,\n    Parameters: stackParams.apiParameters,\n    ResourcesToImport: options.resourcesToImport,\n    RoleARN: options.role,\n    Tags: toCfnTags(options.stack.tags),\n    Capabilities: ['CAPABILITY_IAM', 'CAPABILITY_NAMED_IAM', 'CAPABILITY_AUTO_EXPAND'],\n  });\n\n  await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(format('Initiated creation of changeset: %s; waiting for it to finish creating...', changeSet.Id)));\n  // Fetching all pages if we'll execute, so we can have the correct change count when monitoring.\n  const createdChangeSet = await waitForChangeSet(options.cfn, ioHelper, options.stack.stackName, options.changeSetName, {\n    fetchAll: options.willExecute,\n  });\n  await cleanupOldChangeset(options.cfn, ioHelper, options.changeSetName, options.stack.stackName);\n\n  return createdChangeSet;\n}\n\nfunction toCfnTags(tags: { [id: string]: string }): Tag[] {\n  return Object.entries(tags).map(([k, v]) => ({\n    Key: k,\n    Value: v,\n  }));\n}\n\nasync function cleanupOldChangeset(\n  cfn: ICloudFormationClient,\n  ioHelper: IoHelper,\n  changeSetName: string,\n  stackName: string,\n) {\n  // Delete any existing change sets generated by CDK since change set names must be unique.\n  // The delete request is successful as long as the stack exists (even if the change set does not exist).\n  await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(`Removing existing change set with name ${changeSetName} if it exists`));\n  await cfn.deleteChangeSet({\n    StackName: stackName,\n    ChangeSetName: changeSetName,\n  });\n}\n\n/**\n * Return true if the given change set has no changes\n *\n * This must be determined from the status, not the 'Changes' array on the\n * object; the latter can be empty because no resources were changed, but if\n * there are changes to Outputs, the change set can still be executed.\n */\nexport function changeSetHasNoChanges(description: DescribeChangeSetCommandOutput) {\n  const noChangeErrorPrefixes = [\n    // Error message for a regular template\n    \"The submitted information didn't contain changes.\",\n    // Error message when a Transform is involved (see #10650)\n    'No updates are to be performed.',\n  ];\n\n  return (\n    description.Status === 'FAILED' && noChangeErrorPrefixes.some((p) => (description.StatusReason ?? '').startsWith(p))\n  );\n}\n\n/**\n * Waits for a CloudFormation stack to stabilize in a complete/available state\n * after a delete operation is issued.\n *\n * Fails if the stack is in a FAILED state. Will not fail if the stack was\n * already deleted.\n *\n * @param cfn        a CloudFormation client\n * @param stackName      the name of the stack to wait for after a delete\n *\n * @returns     the CloudFormation description of the stabilized stack after the delete attempt\n */\nexport async function waitForStackDelete(\n  cfn: ICloudFormationClient,\n  ioHelper: IoHelper,\n  stackName: string,\n): Promise<CloudFormationStack | undefined> {\n  const stack = await stabilizeStack(cfn, ioHelper, stackName);\n  if (!stack) {\n    return undefined;\n  }\n\n  const status = stack.stackStatus;\n  if (status.isFailure) {\n    throw new ToolkitError(\n      `The stack named ${stackName} is in a failed state. You may need to delete it from the AWS console : ${status}`,\n    );\n  } else if (status.isDeleted) {\n    return undefined;\n  }\n  return stack;\n}\n\n/**\n * Waits for a CloudFormation stack to stabilize in a complete/available state\n * after an update/create operation is issued.\n *\n * Fails if the stack is in a FAILED state, ROLLBACK state, or DELETED state.\n *\n * @param cfn        a CloudFormation client\n * @param stackName      the name of the stack to wait for after an update\n *\n * @returns     the CloudFormation description of the stabilized stack after the update attempt\n */\nexport async function waitForStackDeploy(\n  cfn: ICloudFormationClient,\n  ioHelper: IoHelper,\n  stackName: string,\n): Promise<CloudFormationStack | undefined> {\n  const stack = await stabilizeStack(cfn, ioHelper, stackName);\n  if (!stack) {\n    return undefined;\n  }\n\n  const status = stack.stackStatus;\n\n  if (status.isCreationFailure) {\n    throw new ToolkitError(\n      `The stack named ${stackName} failed creation, it may need to be manually deleted from the AWS console: ${status}`,\n    );\n  } else if (!status.isDeploySuccess) {\n    throw new ToolkitError(`The stack named ${stackName} failed to deploy: ${status}`);\n  }\n\n  return stack;\n}\n\n/**\n * Wait for a stack to become stable (no longer _IN_PROGRESS), returning it\n */\nexport async function stabilizeStack(\n  cfn: ICloudFormationClient,\n  ioHelper: IoHelper,\n  stackName: string,\n) {\n  await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(format('Waiting for stack %s to finish creating or updating...', stackName)));\n  return waitFor(async () => {\n    const stack = await CloudFormationStack.lookup(cfn, stackName);\n    if (!stack.exists) {\n      await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(format('Stack %s does not exist', stackName)));\n      return null;\n    }\n    const status = stack.stackStatus;\n    if (status.isInProgress) {\n      await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(format('Stack %s has an ongoing operation in progress and is not stable (%s)', stackName, status)));\n      return undefined;\n    } else if (status.isReviewInProgress) {\n      // This may happen if a stack creation operation is interrupted before the ChangeSet execution starts. Recovering\n      // from this would requiring manual intervention (deleting or executing the pending ChangeSet), and failing to do\n      // so will result in an endless wait here (the ChangeSet wont delete or execute itself). Instead of blocking\n      // \"forever\" we proceed as if the stack was existing and stable. If there is a concurrent operation that just\n      // hasn't finished proceeding just yet, either this operation or the concurrent one may fail due to the other one\n      // having made progress. Which is fine. I guess.\n      await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(format('Stack %s is in REVIEW_IN_PROGRESS state. Considering this is a stable status (%s)', stackName, status)));\n    }\n\n    return stack;\n  });\n}\n\n/**\n * The set of (formal) parameters that have been declared in a template\n */\nexport class TemplateParameters {\n  public static fromTemplate(template: Template) {\n    return new TemplateParameters(template.Parameters || {});\n  }\n\n  constructor(private readonly params: Record<string, TemplateParameter>) {\n  }\n\n  /**\n   * Calculate stack parameters to pass from the given desired parameter values\n   *\n   * Will throw if parameters without a Default value or a Previous value are not\n   * supplied.\n   */\n  public supplyAll(updates: Record<string, string | undefined>): ParameterValues {\n    return new ParameterValues(this.params, updates);\n  }\n\n  /**\n   * From the template, the given desired values and the current values, calculate the changes to the stack parameters\n   *\n   * Will take into account parameters already set on the template (will emit\n   * 'UsePreviousValue: true' for those unless the value is changed), and will\n   * throw if parameters without a Default value or a Previous value are not\n   * supplied.\n   */\n  public updateExisting(\n    updates: Record<string, string | undefined>,\n    previousValues: Record<string, string>,\n  ): ParameterValues {\n    return new ParameterValues(this.params, updates, previousValues);\n  }\n}\n\n/**\n * The set of parameters we're going to pass to a Stack\n */\nexport class ParameterValues {\n  public readonly values: Record<string, string> = {};\n  public readonly apiParameters: Parameter[] = [];\n\n  constructor(\n    private readonly formalParams: Record<string, TemplateParameter>,\n    updates: Record<string, string | undefined>,\n    previousValues: Record<string, string> = {},\n  ) {\n    const missingRequired = new Array<string>();\n\n    for (const [key, formalParam] of Object.entries(this.formalParams)) {\n      // Check updates first, then use the previous value (if available), then use\n      // the default (if available).\n      //\n      // If we don't find a parameter value using any of these methods, then that's an error.\n      const updatedValue = updates[key];\n      if (updatedValue !== undefined) {\n        this.values[key] = updatedValue;\n        this.apiParameters.push({\n          ParameterKey: key,\n          ParameterValue: updates[key],\n        });\n        continue;\n      }\n\n      if (key in previousValues) {\n        this.values[key] = previousValues[key];\n        this.apiParameters.push({ ParameterKey: key, UsePreviousValue: true });\n        continue;\n      }\n\n      if (formalParam.Default !== undefined) {\n        this.values[key] = formalParam.Default;\n        continue;\n      }\n\n      // Oh no\n      missingRequired.push(key);\n    }\n\n    if (missingRequired.length > 0) {\n      throw new ToolkitError(`The following CloudFormation Parameters are missing a value: ${missingRequired.join(', ')}`);\n    }\n\n    // Just append all supplied overrides that aren't really expected (this\n    // will fail CFN but maybe people made typos that they want to be notified\n    // of)\n    const unknownParam = ([key, _]: [string, any]) => this.formalParams[key] === undefined;\n    const hasValue = ([_, value]: [string, any]) => !!value;\n    for (const [key, value] of Object.entries(updates).filter(unknownParam).filter(hasValue)) {\n      this.values[key] = value!;\n      this.apiParameters.push({ ParameterKey: key, ParameterValue: value });\n    }\n  }\n\n  /**\n   * Whether this set of parameter updates will change the actual stack values\n   */\n  public hasChanges(currentValues: Record<string, string>): ParameterChanges {\n    // If any of the parameters are SSM parameters, deploying must always happen\n    // because we can't predict what the values will be. We will allow some\n    // parameters to opt out of this check by having a magic string in their description.\n    if (\n      Object.values(this.formalParams).some(\n        (p) => p.Type.startsWith('AWS::SSM::Parameter::') && !p.Description?.includes(SSMPARAM_NO_INVALIDATE),\n      )\n    ) {\n      return 'ssm';\n    }\n\n    // Otherwise we're dirty if:\n    // - any of the existing values are removed, or changed\n    if (Object.entries(currentValues).some(([key, value]) => !(key in this.values) || value !== this.values[key])) {\n      return true;\n    }\n\n    // - any of the values we're setting are new\n    if (Object.keys(this.values).some((key) => !(key in currentValues))) {\n      return true;\n    }\n\n    return false;\n  }\n}\n\nexport type ParameterChanges = boolean | 'ssm';\n"]}
@@ -2,3 +2,4 @@ export * from './deployments';
2
2
  export * from './deployment-result';
3
3
  export * from './deployment-method';
4
4
  export * from './asset-manifest-builder';
5
+ export * from './cfn-api';
@@ -18,4 +18,5 @@ __exportStar(require("./deployments"), exports);
18
18
  __exportStar(require("./deployment-result"), exports);
19
19
  __exportStar(require("./deployment-method"), exports);
20
20
  __exportStar(require("./asset-manifest-builder"), exports);
21
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsZ0RBQThCO0FBQzlCLHNEQUFvQztBQUNwQyxzREFBb0M7QUFDcEMsMkRBQXlDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9kZXBsb3ltZW50cyc7XG5leHBvcnQgKiBmcm9tICcuL2RlcGxveW1lbnQtcmVzdWx0JztcbmV4cG9ydCAqIGZyb20gJy4vZGVwbG95bWVudC1tZXRob2QnO1xuZXhwb3J0ICogZnJvbSAnLi9hc3NldC1tYW5pZmVzdC1idWlsZGVyJztcbiJdfQ==
21
+ __exportStar(require("./cfn-api"), exports);
22
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsZ0RBQThCO0FBQzlCLHNEQUFvQztBQUNwQyxzREFBb0M7QUFDcEMsMkRBQXlDO0FBQ3pDLDRDQUEwQiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vZGVwbG95bWVudHMnO1xuZXhwb3J0ICogZnJvbSAnLi9kZXBsb3ltZW50LXJlc3VsdCc7XG5leHBvcnQgKiBmcm9tICcuL2RlcGxveW1lbnQtbWV0aG9kJztcbmV4cG9ydCAqIGZyb20gJy4vYXNzZXQtbWFuaWZlc3QtYnVpbGRlcic7XG5leHBvcnQgKiBmcm9tICcuL2Nmbi1hcGknO1xuIl19