aws-cdk 2.18.0 → 2.21.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.
Files changed (69) hide show
  1. package/README.md +59 -11
  2. package/THIRD_PARTY_LICENSES +8 -28
  3. package/build-info.json +2 -2
  4. package/does-not-exist.json +1 -1
  5. package/lib/api/aws-auth/account-cache.d.ts +1 -1
  6. package/lib/api/aws-auth/account-cache.js +2 -2
  7. package/lib/api/bootstrap/bootstrap-template.yaml +3 -1
  8. package/lib/api/cloudformation-deployments.d.ts +12 -1
  9. package/lib/api/cloudformation-deployments.js +32 -12
  10. package/lib/api/cxapp/exec.js +7 -2
  11. package/lib/api/deploy-stack.d.ts +24 -0
  12. package/lib/api/deploy-stack.js +46 -8
  13. package/lib/api/evaluate-cloudformation-template.js +3 -2
  14. package/lib/api/hotswap/common.js +2 -2
  15. package/lib/api/util/cloudformation/stack-activity-monitor.d.ts +5 -0
  16. package/lib/api/util/cloudformation/stack-activity-monitor.js +24 -7
  17. package/lib/api/util/cloudformation/stack-status.js +2 -2
  18. package/lib/api/util/cloudformation.d.ts +3 -0
  19. package/lib/api/util/cloudformation.js +1 -1
  20. package/lib/api/util/display.d.ts +3 -0
  21. package/lib/api/util/display.js +23 -15
  22. package/lib/cdk-toolkit.d.ts +44 -21
  23. package/lib/cdk-toolkit.js +68 -1
  24. package/lib/cli.js +47 -5
  25. package/lib/commands/doctor.js +2 -2
  26. package/lib/import.d.ts +151 -0
  27. package/lib/import.js +245 -0
  28. package/lib/index.js +12856 -6484
  29. package/lib/init-templates/v1/app/csharp/README.md +1 -1
  30. package/lib/init-templates/v1/app/fsharp/README.md +1 -1
  31. package/lib/init-templates/v1/app/go/README.md +1 -1
  32. package/lib/init-templates/v1/app/java/README.md +1 -1
  33. package/lib/init-templates/v1/app/javascript/README.md +1 -1
  34. package/lib/init-templates/v1/app/python/README.template.md +1 -1
  35. package/lib/init-templates/v1/app/typescript/README.md +1 -1
  36. package/lib/init-templates/v2/app/csharp/README.md +1 -1
  37. package/lib/init-templates/v2/app/fsharp/README.md +1 -1
  38. package/lib/init-templates/v2/app/go/README.md +1 -1
  39. package/lib/init-templates/v2/app/java/README.md +1 -1
  40. package/lib/init-templates/v2/app/javascript/README.md +1 -1
  41. package/lib/init-templates/v2/app/python/README.template.md +1 -1
  42. package/lib/init-templates/v2/app/typescript/README.md +1 -1
  43. package/lib/notices.d.ts +1 -0
  44. package/lib/notices.js +43 -21
  45. package/lib/settings.js +3 -2
  46. package/lib/util/asset-publishing.d.ts +7 -1
  47. package/lib/util/asset-publishing.js +13 -6
  48. package/lib/version.js +2 -2
  49. package/package.json +14 -14
  50. package/test/api/hotswap/state-machine-hotswap-deployments.test.js +75 -1
  51. package/test/api/util/display.test.d.ts +1 -0
  52. package/test/api/util/display.test.js +54 -0
  53. package/test/bockfs.js +2 -2
  54. package/test/cdk-toolkit.test.js +4 -2
  55. package/test/import.test.d.ts +1 -0
  56. package/test/import.test.js +189 -0
  57. package/test/integ/cli/app/app.js +25 -0
  58. package/test/integ/cli/app/docker/Dockerfile +1 -1
  59. package/test/integ/cli/app/docker/Dockerfile.Custom +1 -1
  60. package/test/integ/cli/cli.integtest.js +33 -1
  61. package/test/integ/cli-regression-patches/v1.64.0/cdk-helpers.js +1 -1
  62. package/test/integ/cli-regression-patches/v1.64.1/cdk-helpers.js +1 -1
  63. package/test/integ/cli-regression-patches/v1.67.0/cdk-helpers.js +1 -1
  64. package/test/integ/helpers/cdk.js +4 -5
  65. package/test/integ/helpers/test-helpers.js +1 -3
  66. package/test/notices.test.js +35 -21
  67. package/test/settings.test.js +10 -1
  68. package/test/util/mock-sdk.d.ts +2 -0
  69. package/test/util/mock-sdk.js +8 -2
@@ -61,7 +61,12 @@ async function execProgram(aws, config) {
61
61
  if (!outdir) {
62
62
  throw new Error('unexpected: --output is required');
63
63
  }
64
- await fs.mkdirp(outdir);
64
+ try {
65
+ await fs.mkdirp(outdir);
66
+ }
67
+ catch (error) {
68
+ throw new Error(`Could not create output directory ${outdir} (${error.message})`);
69
+ }
65
70
  logging_1.debug('outdir:', outdir);
66
71
  env[cxapi.OUTDIR_ENV] = outdir;
67
72
  // Send version information
@@ -189,4 +194,4 @@ async function guessExecutable(commandLine) {
189
194
  }
190
195
  return commandLine;
191
196
  }
192
- //# 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 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 { debug } from '../../logging';\nimport { Configuration, PROJECT_CONFIG, USER_DEFAULTS } from '../../settings';\nimport { versionNumber } from '../../version';\nimport { SdkProvider } from '../aws-auth';\n\n/** Invokes the cloud executable and returns JSON output */\nexport async function execProgram(aws: SdkProvider, config: Configuration): Promise<cxapi.CloudAssembly> {\n  const env: { [key: string]: string } = { };\n\n  const context = config.context.all;\n  await populateDefaultEnvironmentIfNeeded(aws, env);\n\n  const debugMode: boolean = config.settings.get(['debug']) ?? true;\n  if (debugMode) {\n    env.CDK_DEBUG = 'true';\n  }\n\n  const pathMetadata: boolean = config.settings.get(['pathMetadata']) ?? true;\n  if (pathMetadata) {\n    context[cxapi.PATH_METADATA_ENABLE_CONTEXT] = true;\n  }\n\n  const assetMetadata: boolean = config.settings.get(['assetMetadata']) ?? true;\n  if (assetMetadata) {\n    context[cxapi.ASSET_RESOURCE_METADATA_ENABLED_CONTEXT] = true;\n  }\n\n  const versionReporting: boolean = config.settings.get(['versionReporting']) ?? true;\n  if (versionReporting) { context[cxapi.ANALYTICS_REPORTING_ENABLED_CONTEXT] = true; }\n  // We need to keep on doing this for framework version from before this flag was deprecated.\n  if (!versionReporting) { context['aws:cdk:disable-version-reporting'] = true; }\n\n  const stagingEnabled = config.settings.get(['staging']) ?? true;\n  if (!stagingEnabled) {\n    context[cxapi.DISABLE_ASSET_STAGING_CONTEXT] = true;\n  }\n\n  const bundlingStacks = config.settings.get(['bundlingStacks']) ?? ['*'];\n  context[cxapi.BUNDLING_STACKS] = bundlingStacks;\n\n  debug('context:', context);\n  env[cxapi.CONTEXT_ENV] = JSON.stringify(context);\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 Error(`--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    return createAssembly(app);\n  }\n\n  const commandLine = await guessExecutable(appToArray(app));\n\n  const outdir = config.settings.get(['output']);\n  if (!outdir) {\n    throw new Error('unexpected: --output is required');\n  }\n  await fs.mkdirp(outdir);\n\n  debug('outdir:', outdir);\n  env[cxapi.OUTDIR_ENV] = outdir;\n\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  await exec(commandLine.join(' '));\n\n  return createAssembly(outdir);\n\n  function createAssembly(appDir: string) {\n    try {\n      return new cxapi.CloudAssembly(appDir);\n    } catch (error) {\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 Error(`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  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          return fail(new Error(`Subprocess exited with error ${code}`));\n        }\n      });\n    });\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 */\nasync function populateDefaultEnvironmentIfNeeded(aws: SdkProvider, env: { [key: string]: string | undefined}) {\n  env[cxapi.DEFAULT_REGION_ENV] = aws.defaultRegion;\n  debug(`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    debug(`Setting \"${cxapi.DEFAULT_ACCOUNT_ENV}\" environment variable to`, env[cxapi.DEFAULT_ACCOUNT_ENV]);\n  }\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 */\nasync function guessExecutable(commandLine: string[]) {\n  if (commandLine.length === 1) {\n    let fstat;\n\n    try {\n      fstat = await fs.stat(commandLine[0]);\n    } catch (error) {\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"]}
197
+ //# 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 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 { debug } from '../../logging';\nimport { Configuration, PROJECT_CONFIG, USER_DEFAULTS } from '../../settings';\nimport { versionNumber } from '../../version';\nimport { SdkProvider } from '../aws-auth';\n\n/** Invokes the cloud executable and returns JSON output */\nexport async function execProgram(aws: SdkProvider, config: Configuration): Promise<cxapi.CloudAssembly> {\n  const env: { [key: string]: string } = { };\n\n  const context = config.context.all;\n  await populateDefaultEnvironmentIfNeeded(aws, env);\n\n  const debugMode: boolean = config.settings.get(['debug']) ?? true;\n  if (debugMode) {\n    env.CDK_DEBUG = 'true';\n  }\n\n  const pathMetadata: boolean = config.settings.get(['pathMetadata']) ?? true;\n  if (pathMetadata) {\n    context[cxapi.PATH_METADATA_ENABLE_CONTEXT] = true;\n  }\n\n  const assetMetadata: boolean = config.settings.get(['assetMetadata']) ?? true;\n  if (assetMetadata) {\n    context[cxapi.ASSET_RESOURCE_METADATA_ENABLED_CONTEXT] = true;\n  }\n\n  const versionReporting: boolean = config.settings.get(['versionReporting']) ?? true;\n  if (versionReporting) { context[cxapi.ANALYTICS_REPORTING_ENABLED_CONTEXT] = true; }\n  // We need to keep on doing this for framework version from before this flag was deprecated.\n  if (!versionReporting) { context['aws:cdk:disable-version-reporting'] = true; }\n\n  const stagingEnabled = config.settings.get(['staging']) ?? true;\n  if (!stagingEnabled) {\n    context[cxapi.DISABLE_ASSET_STAGING_CONTEXT] = true;\n  }\n\n  const bundlingStacks = config.settings.get(['bundlingStacks']) ?? ['*'];\n  context[cxapi.BUNDLING_STACKS] = bundlingStacks;\n\n  debug('context:', context);\n  env[cxapi.CONTEXT_ENV] = JSON.stringify(context);\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 Error(`--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    return createAssembly(app);\n  }\n\n  const commandLine = await guessExecutable(appToArray(app));\n\n  const outdir = config.settings.get(['output']);\n  if (!outdir) {\n    throw new Error('unexpected: --output is required');\n  }\n  try {\n    await fs.mkdirp(outdir);\n  } catch (error) {\n    throw new Error(`Could not create output directory ${outdir} (${error.message})`);\n  }\n\n  debug('outdir:', outdir);\n  env[cxapi.OUTDIR_ENV] = outdir;\n\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  await exec(commandLine.join(' '));\n\n  return createAssembly(outdir);\n\n  function createAssembly(appDir: string) {\n    try {\n      return new cxapi.CloudAssembly(appDir);\n    } catch (error) {\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 Error(`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  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          return fail(new Error(`Subprocess exited with error ${code}`));\n        }\n      });\n    });\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 */\nasync function populateDefaultEnvironmentIfNeeded(aws: SdkProvider, env: { [key: string]: string | undefined}) {\n  env[cxapi.DEFAULT_REGION_ENV] = aws.defaultRegion;\n  debug(`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    debug(`Setting \"${cxapi.DEFAULT_ACCOUNT_ENV}\" environment variable to`, env[cxapi.DEFAULT_ACCOUNT_ENV]);\n  }\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 */\nasync function guessExecutable(commandLine: string[]) {\n  if (commandLine.length === 1) {\n    let fstat;\n\n    try {\n      fstat = await fs.stat(commandLine[0]);\n    } catch (error) {\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"]}
@@ -2,7 +2,12 @@ import * as cxapi from '@aws-cdk/cx-api';
2
2
  import { Tag } from '../cdk-toolkit';
3
3
  import { ISDK, SdkProvider } from './aws-auth';
4
4
  import { ToolkitInfo } from './toolkit-info';
5
+ import { ResourcesToImport } from './util/cloudformation';
5
6
  import { StackActivityProgress } from './util/cloudformation/stack-activity-monitor';
7
+ declare type TemplateBodyParameter = {
8
+ TemplateBody?: string;
9
+ TemplateURL?: string;
10
+ };
6
11
  export interface DeployStackResult {
7
12
  readonly noOp: boolean;
8
13
  readonly outputs: {
@@ -144,8 +149,26 @@ export interface DeployStackOptions {
144
149
  * @default - nothing extra is appended to the User-Agent header
145
150
  */
146
151
  readonly extraUserAgent?: string;
152
+ /**
153
+ * If set, change set of type IMPORT will be created, and resourcesToImport
154
+ * passed to it.
155
+ */
156
+ readonly resourcesToImport?: ResourcesToImport;
157
+ /**
158
+ * If present, use this given template instead of the stored one
159
+ *
160
+ * @default - Use the stored template
161
+ */
162
+ readonly overrideTemplate?: any;
147
163
  }
148
164
  export declare function deployStack(options: DeployStackOptions): Promise<DeployStackResult>;
165
+ /**
166
+ * Prepare a body parameter for CFN, performing the upload
167
+ *
168
+ * Return it as-is if it is small enough to pass in the API call,
169
+ * upload to S3 and return the coordinates if it is not.
170
+ */
171
+ export declare function makeBodyParameterAndUpload(stack: cxapi.CloudFormationStackArtifact, resolvedEnvironment: cxapi.Environment, toolkitInfo: ToolkitInfo, sdkProvider: SdkProvider, sdk: ISDK, overrideTemplate?: any): Promise<TemplateBodyParameter>;
149
172
  export interface DestroyStackOptions {
150
173
  /**
151
174
  * The stack to be destroyed
@@ -157,3 +180,4 @@ export interface DestroyStackOptions {
157
180
  quiet?: boolean;
158
181
  }
159
182
  export declare function destroyStack(options: DestroyStackOptions): Promise<void>;
183
+ export {};
@@ -1,8 +1,9 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.destroyStack = exports.deployStack = void 0;
3
+ exports.destroyStack = exports.makeBodyParameterAndUpload = exports.deployStack = void 0;
4
4
  const cxapi = require("@aws-cdk/cx-api");
5
5
  const chalk = require("chalk");
6
+ const fs = require("fs-extra");
6
7
  const uuid = require("uuid");
7
8
  const assets_1 = require("../assets");
8
9
  const logging_1 = require("../logging");
@@ -61,7 +62,7 @@ async function deployStack(options) {
61
62
  else {
62
63
  logging_1.debug(`${deployName}: deploying...`);
63
64
  }
64
- const bodyParameter = await makeBodyParameter(stackArtifact, options.resolvedEnvironment, legacyAssets, options.toolkitInfo, options.sdk);
65
+ const bodyParameter = await makeBodyParameter(stackArtifact, options.resolvedEnvironment, legacyAssets, options.toolkitInfo, options.sdk, options.overrideTemplate);
65
66
  await asset_publishing_1.publishAssets(legacyAssets.toManifest(stackArtifact.assembly.directory), options.sdkProvider, stackEnv);
66
67
  if (options.hotswap) {
67
68
  // attempt to short-circuit the deployment if possible
@@ -107,7 +108,8 @@ async function prepareAndExecuteChangeSet(options, cloudFormationStack, stackArt
107
108
  const changeSet = await cfn.createChangeSet({
108
109
  StackName: deployName,
109
110
  ChangeSetName: changeSetName,
110
- ChangeSetType: update ? 'UPDATE' : 'CREATE',
111
+ ChangeSetType: options.resourcesToImport ? 'IMPORT' : update ? 'UPDATE' : 'CREATE',
112
+ ResourcesToImport: options.resourcesToImport,
111
113
  Description: `CDK Changeset for execution ${executionId}`,
112
114
  TemplateBody: bodyParameter.TemplateBody,
113
115
  TemplateURL: bodyParameter.TemplateURL,
@@ -162,6 +164,9 @@ async function prepareAndExecuteChangeSet(options, cloudFormationStack, stackArt
162
164
  }
163
165
  cloudFormationStack = finalStack;
164
166
  }
167
+ catch (e) {
168
+ throw new Error(suffixWithErrors(e.message, monitor === null || monitor === void 0 ? void 0 : monitor.errors));
169
+ }
165
170
  finally {
166
171
  await (monitor === null || monitor === void 0 ? void 0 : monitor.stop());
167
172
  }
@@ -185,13 +190,13 @@ async function prepareAndExecuteChangeSet(options, cloudFormationStack, stackArt
185
190
  * @param stack the synthesized stack that provides the CloudFormation template
186
191
  * @param toolkitInfo information about the toolkit stack
187
192
  */
188
- async function makeBodyParameter(stack, resolvedEnvironment, assetManifest, toolkitInfo, sdk) {
193
+ async function makeBodyParameter(stack, resolvedEnvironment, assetManifest, toolkitInfo, sdk, overrideTemplate) {
189
194
  // If the template has already been uploaded to S3, just use it from there.
190
- if (stack.stackTemplateAssetObjectUrl) {
195
+ if (stack.stackTemplateAssetObjectUrl && !overrideTemplate) {
191
196
  return { TemplateURL: restUrlFromManifest(stack.stackTemplateAssetObjectUrl, resolvedEnvironment, sdk) };
192
197
  }
193
198
  // Otherwise, pass via API call (if small) or upload here (if large)
194
- const templateJson = serialize_1.toYAML(stack.template);
199
+ const templateJson = serialize_1.toYAML(overrideTemplate !== null && overrideTemplate !== void 0 ? overrideTemplate : stack.template);
195
200
  if (templateJson.length <= LARGE_TEMPLATE_SIZE_KB * 1024) {
196
201
  return { TemplateBody: templateJson };
197
202
  }
@@ -203,8 +208,14 @@ async function makeBodyParameter(stack, resolvedEnvironment, assetManifest, tool
203
208
  }
204
209
  const templateHash = content_hash_1.contentHash(templateJson);
205
210
  const key = `cdk/${stack.id}/${templateHash}.yml`;
211
+ let templateFile = stack.templateFile;
212
+ if (overrideTemplate) {
213
+ // Add a variant of this template
214
+ templateFile = `${stack.templateFile}-${templateHash}.yaml`;
215
+ await fs.writeFile(templateFile, templateJson, { encoding: 'utf-8' });
216
+ }
206
217
  assetManifest.addFileAsset(templateHash, {
207
- path: stack.templateFile,
218
+ path: templateFile,
208
219
  }, {
209
220
  bucketName: toolkitInfo.bucketName,
210
221
  objectKey: key,
@@ -213,6 +224,25 @@ async function makeBodyParameter(stack, resolvedEnvironment, assetManifest, tool
213
224
  logging_1.debug('Storing template in S3 at:', templateURL);
214
225
  return { TemplateURL: templateURL };
215
226
  }
227
+ /**
228
+ * Prepare a body parameter for CFN, performing the upload
229
+ *
230
+ * Return it as-is if it is small enough to pass in the API call,
231
+ * upload to S3 and return the coordinates if it is not.
232
+ */
233
+ async function makeBodyParameterAndUpload(stack, resolvedEnvironment, toolkitInfo, sdkProvider, sdk, overrideTemplate) {
234
+ // We don't have access to the actual asset manifest here, so pretend that the
235
+ // stack doesn't have a pre-published URL.
236
+ const forceUploadStack = Object.create(stack, {
237
+ stackTemplateAssetObjectUrl: { value: undefined },
238
+ });
239
+ const builder = new asset_manifest_builder_1.AssetManifestBuilder();
240
+ const bodyparam = await makeBodyParameter(forceUploadStack, resolvedEnvironment, builder, toolkitInfo, sdk, overrideTemplate);
241
+ const manifest = builder.toManifest(stack.assembly.directory);
242
+ await asset_publishing_1.publishAssets(manifest, sdkProvider, resolvedEnvironment, { quiet: true });
243
+ return bodyparam;
244
+ }
245
+ exports.makeBodyParameterAndUpload = makeBodyParameterAndUpload;
216
246
  async function destroyStack(options) {
217
247
  const deployName = options.deployName || options.stack.stackName;
218
248
  const cfn = options.sdk.cloudFormation();
@@ -228,6 +258,9 @@ async function destroyStack(options) {
228
258
  throw new Error(`Failed to destroy ${deployName}: ${destroyedStack.stackStatus}`);
229
259
  }
230
260
  }
261
+ catch (e) {
262
+ throw new Error(suffixWithErrors(e.message, monitor === null || monitor === void 0 ? void 0 : monitor.errors));
263
+ }
231
264
  finally {
232
265
  if (monitor) {
233
266
  await monitor.stop();
@@ -342,4 +375,9 @@ function restUrlFromManifest(url, environment, sdk) {
342
375
  const urlSuffix = sdk.getEndpointSuffix(environment.region);
343
376
  return `https://s3.${environment.region}.${urlSuffix}/${bucketName}/${objectKey}`;
344
377
  }
345
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"deploy-stack.js","sourceRoot":"","sources":["deploy-stack.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxapi from '@aws-cdk/cx-api';\nimport * as chalk from 'chalk';\nimport * as uuid from 'uuid';\nimport { addMetadataAssetsToManifest } from '../assets';\nimport { Tag } from '../cdk-toolkit';\nimport { debug, error, print } from '../logging';\nimport { toYAML } from '../serialize';\nimport { AssetManifestBuilder } from '../util/asset-manifest-builder';\nimport { publishAssets } from '../util/asset-publishing';\nimport { contentHash } from '../util/content-hash';\nimport { ISDK, SdkProvider } from './aws-auth';\nimport { CfnEvaluationException } from './evaluate-cloudformation-template';\nimport { tryHotswapDeployment } from './hotswap-deployments';\nimport { ICON } from './hotswap/common';\nimport { ToolkitInfo } from './toolkit-info';\nimport {\n  changeSetHasNoChanges, CloudFormationStack, TemplateParameters, waitForChangeSet,\n  waitForStackDeploy, waitForStackDelete, ParameterValues, ParameterChanges,\n} from './util/cloudformation';\nimport { StackActivityMonitor, StackActivityProgress } from './util/cloudformation/stack-activity-monitor';\n\ntype TemplateBodyParameter = {\n  TemplateBody?: string\n  TemplateURL?: string\n};\n\nexport interface DeployStackResult {\n  readonly noOp: boolean;\n  readonly outputs: { [name: string]: string };\n  readonly stackArn: string;\n}\n\nexport interface DeployStackOptions {\n  /**\n   * The stack to be deployed\n   */\n  stack: cxapi.CloudFormationStackArtifact;\n\n  /**\n   * The environment to deploy this stack in\n   *\n   * The environment on the stack artifact may be unresolved, this one\n   * must be resolved.\n   */\n  resolvedEnvironment: cxapi.Environment;\n\n  /**\n   * The SDK to use for deploying the stack\n   *\n   * Should have been initialized with the correct role with which\n   * stack operations should be performed.\n   */\n  sdk: ISDK;\n\n  /**\n   * SDK provider (seeded with default credentials)\n   *\n   * Will exclusively be used to assume publishing credentials (which must\n   * start out from current credentials regardless of whether we've assumed an\n   * action role to touch the stack or not).\n   *\n   * Used for the following purposes:\n   *\n   * - Publish legacy assets.\n   * - Upload large CloudFormation templates to the staging bucket.\n   */\n  sdkProvider: SdkProvider;\n\n  /**\n   * Information about the bootstrap stack found in the target environment\n   */\n  toolkitInfo: ToolkitInfo;\n\n  /**\n   * Role to pass to CloudFormation to execute the change set\n   *\n   * @default - Role specified on stack, otherwise current\n   */\n  roleArn?: string;\n\n  /**\n   * Notification ARNs to pass to CloudFormation to notify when the change set has completed\n   *\n   * @default - No notifications\n   */\n  notificationArns?: string[];\n\n  /**\n   * Name to deploy the stack under\n   *\n   * @default - Name from assembly\n   */\n  deployName?: string;\n\n  /**\n   * Quiet or verbose deployment\n   *\n   * @default false\n   */\n  quiet?: boolean;\n\n  /**\n   * List of asset IDs which shouldn't be built\n   *\n   * @default - Build all assets\n   */\n  reuseAssets?: string[];\n\n  /**\n   * Tags to pass to CloudFormation to add to stack\n   *\n   * @default - No tags\n   */\n  tags?: Tag[];\n\n  /**\n   * Whether to execute the changeset or leave it in review.\n   *\n   * @default true\n   */\n  execute?: boolean;\n\n  /**\n   * Optional name to use for the CloudFormation change set.\n   * If not provided, a name will be generated automatically.\n   */\n  changeSetName?: string;\n\n  /**\n   * The collection of extra parameters\n   * (in addition to those used for assets)\n   * to pass to the deployed template.\n   * Note that parameters with `undefined` or empty values will be ignored,\n   * and not passed to the template.\n   *\n   * @default - no additional parameters will be passed to the template\n   */\n  parameters?: { [name: string]: string | undefined };\n\n  /**\n   * Use previous values for unspecified parameters\n   *\n   * If not set, all parameters must be specified for every deployment.\n   *\n   * @default false\n   */\n  usePreviousParameters?: boolean;\n\n  /**\n   * Display mode for stack deployment progress.\n   *\n   * @default StackActivityProgress.Bar stack events will be displayed for\n   *   the resource currently being deployed.\n   */\n  progress?: StackActivityProgress;\n\n  /**\n   * Deploy even if the deployed template is identical to the one we are about to deploy.\n   * @default false\n   */\n  force?: boolean;\n\n  /**\n   * Whether we are on a CI system\n   *\n   * @default false\n   */\n  readonly ci?: boolean;\n\n  /**\n   * Rollback failed deployments\n   *\n   * @default true\n   */\n  readonly rollback?: boolean;\n\n  /*\n   * Whether to perform a 'hotswap' deployment.\n   * A 'hotswap' deployment will attempt to short-circuit CloudFormation\n   * and update the affected resources like Lambda functions directly.\n   *\n   * @default - false for regular deployments, true for 'watch' deployments\n   */\n  readonly hotswap?: boolean;\n\n  /**\n   * The extra string to append to the User-Agent header when performing AWS SDK calls.\n   *\n   * @default - nothing extra is appended to the User-Agent header\n   */\n  readonly extraUserAgent?: string;\n}\n\nconst LARGE_TEMPLATE_SIZE_KB = 50;\n\nexport async function deployStack(options: DeployStackOptions): Promise<DeployStackResult> {\n  const stackArtifact = options.stack;\n\n  const stackEnv = options.resolvedEnvironment;\n\n  options.sdk.appendCustomUserAgent(options.extraUserAgent);\n  const cfn = options.sdk.cloudFormation();\n  const deployName = options.deployName || stackArtifact.stackName;\n  let cloudFormationStack = await CloudFormationStack.lookup(cfn, deployName);\n\n  if (cloudFormationStack.stackStatus.isCreationFailure) {\n    debug(`Found existing stack ${deployName} that had previously failed creation. Deleting it before attempting to re-create it.`);\n    await cfn.deleteStack({ StackName: deployName }).promise();\n    const deletedStack = await waitForStackDelete(cfn, deployName);\n    if (deletedStack && deletedStack.stackStatus.name !== 'DELETE_COMPLETE') {\n      throw new Error(`Failed deleting stack ${deployName} that had previously failed creation (current state: ${deletedStack.stackStatus})`);\n    }\n    // Update variable to mark that the stack does not exist anymore, but avoid\n    // doing an actual lookup in CloudFormation (which would be silly to do if\n    // we just deleted it).\n    cloudFormationStack = CloudFormationStack.doesNotExist(cfn, deployName);\n  }\n\n  // Detect \"legacy\" assets (which remain in the metadata) and publish them via\n  // an ad-hoc asset manifest, while passing their locations via template\n  // parameters.\n  const legacyAssets = new AssetManifestBuilder();\n  const assetParams = await addMetadataAssetsToManifest(stackArtifact, legacyAssets, options.toolkitInfo, options.reuseAssets);\n\n  const finalParameterValues = { ...options.parameters, ...assetParams };\n\n  const templateParams = TemplateParameters.fromTemplate(stackArtifact.template);\n  const stackParams = options.usePreviousParameters\n    ? templateParams.updateExisting(finalParameterValues, cloudFormationStack.parameters)\n    : templateParams.supplyAll(finalParameterValues);\n\n  if (await canSkipDeploy(options, cloudFormationStack, stackParams.hasChanges(cloudFormationStack.parameters))) {\n    debug(`${deployName}: skipping deployment (use --force to override)`);\n    // if we can skip deployment and we are performing a hotswap, let the user know\n    // that no hotswap deployment happened\n    if (options.hotswap) {\n      print(`\\n ${ICON} %s\\n`, chalk.bold('hotswap deployment skipped - no changes were detected (use --force to override)'));\n    }\n    return {\n      noOp: true,\n      outputs: cloudFormationStack.outputs,\n      stackArn: cloudFormationStack.stackId,\n    };\n  } else {\n    debug(`${deployName}: deploying...`);\n  }\n\n  const bodyParameter = await makeBodyParameter(stackArtifact, options.resolvedEnvironment, legacyAssets, options.toolkitInfo, options.sdk);\n  await publishAssets(legacyAssets.toManifest(stackArtifact.assembly.directory), options.sdkProvider, stackEnv);\n\n  if (options.hotswap) {\n    // attempt to short-circuit the deployment if possible\n    try {\n      const hotswapDeploymentResult = await tryHotswapDeployment(options.sdkProvider, assetParams, cloudFormationStack, stackArtifact);\n      if (hotswapDeploymentResult) {\n        return hotswapDeploymentResult;\n      }\n      print('Could not perform a hotswap deployment, as the stack %s contains non-Asset changes', stackArtifact.displayName);\n    } catch (e) {\n      if (!(e instanceof CfnEvaluationException)) {\n        throw e;\n      }\n      print('Could not perform a hotswap deployment, because the CloudFormation template could not be resolved: %s', e.message);\n    }\n    print('Falling back to doing a full deployment');\n  }\n\n  // could not short-circuit the deployment, perform a full CFN deploy instead\n  return prepareAndExecuteChangeSet(options, cloudFormationStack, stackArtifact, stackParams, bodyParameter);\n}\n\nasync function prepareAndExecuteChangeSet(\n  options: DeployStackOptions, cloudFormationStack: CloudFormationStack,\n  stackArtifact: cxapi.CloudFormationStackArtifact, stackParams: ParameterValues, bodyParameter: TemplateBodyParameter,\n): Promise<DeployStackResult> {\n  // if we got here, and hotswap is enabled, that means changes couldn't be hotswapped,\n  // and we had to fall back on a full deployment. Note that fact in our User-Agent\n  if (options.hotswap) {\n    options.sdk.appendCustomUserAgent('cdk-hotswap/fallback');\n  }\n\n  const cfn = options.sdk.cloudFormation();\n  const deployName = options.deployName ?? stackArtifact.stackName;\n\n  const changeSetName = options.changeSetName ?? 'cdk-deploy-change-set';\n  if (cloudFormationStack.exists) {\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    debug(`Removing existing change set with name ${changeSetName} if it exists`);\n    await cfn.deleteChangeSet({ StackName: deployName, ChangeSetName: changeSetName }).promise();\n  }\n\n  const update = cloudFormationStack.exists && cloudFormationStack.stackStatus.name !== 'REVIEW_IN_PROGRESS';\n\n  debug(`Attempting to create ChangeSet with name ${changeSetName} to ${update ? 'update' : 'create'} stack ${deployName}`);\n  print('%s: creating CloudFormation changeset...', chalk.bold(deployName));\n  const executionId = uuid.v4();\n  const changeSet = await cfn.createChangeSet({\n    StackName: deployName,\n    ChangeSetName: changeSetName,\n    ChangeSetType: update ? 'UPDATE' : 'CREATE',\n    Description: `CDK Changeset for execution ${executionId}`,\n    TemplateBody: bodyParameter.TemplateBody,\n    TemplateURL: bodyParameter.TemplateURL,\n    Parameters: stackParams.apiParameters,\n    RoleARN: options.roleArn,\n    NotificationARNs: options.notificationArns,\n    Capabilities: ['CAPABILITY_IAM', 'CAPABILITY_NAMED_IAM', 'CAPABILITY_AUTO_EXPAND'],\n    Tags: options.tags,\n  }).promise();\n  debug('Initiated creation of changeset: %s; waiting for it to finish creating...', changeSet.Id);\n  const changeSetDescription = await waitForChangeSet(cfn, deployName, changeSetName);\n\n  // Update termination protection only if it has changed.\n  const terminationProtection = stackArtifact.terminationProtection ?? false;\n  if (!!cloudFormationStack.terminationProtection !== terminationProtection) {\n    debug('Updating termination protection from %s to %s for stack %s', cloudFormationStack.terminationProtection, terminationProtection, deployName);\n    await cfn.updateTerminationProtection({\n      StackName: deployName,\n      EnableTerminationProtection: terminationProtection,\n    }).promise();\n    debug('Termination protection updated to %s for stack %s', terminationProtection, deployName);\n  }\n\n  if (changeSetHasNoChanges(changeSetDescription)) {\n    debug('No changes are to be performed on %s.', deployName);\n    if (options.execute) {\n      debug('Deleting empty change set %s', changeSet.Id);\n      await cfn.deleteChangeSet({ StackName: deployName, ChangeSetName: changeSetName }).promise();\n    }\n    return { noOp: true, outputs: cloudFormationStack.outputs, stackArn: changeSet.StackId! };\n  }\n\n  const execute = options.execute === undefined ? true : options.execute;\n  if (execute) {\n    debug('Initiating execution of changeset %s on stack %s', changeSet.Id, deployName);\n\n    const shouldDisableRollback = options.rollback === false;\n    // Do a bit of contortions to only pass the `DisableRollback` flag if it's true. That way,\n    // CloudFormation won't balk at the unrecognized option in regions where the feature is not available yet.\n    const disableRollback = shouldDisableRollback ? { DisableRollback: true } : undefined;\n\n    await cfn.executeChangeSet({ StackName: deployName, ChangeSetName: changeSetName, ...disableRollback }).promise();\n\n    // eslint-disable-next-line max-len\n    const changeSetLength: number = (changeSetDescription.Changes ?? []).length;\n    const monitor = options.quiet ? undefined : StackActivityMonitor.withDefaultPrinter(cfn, deployName, stackArtifact, {\n      // +1 for the extra event emitted from updates.\n      resourcesTotal: cloudFormationStack.exists ? changeSetLength + 1 : changeSetLength,\n      progress: options.progress,\n      changeSetCreationTime: changeSetDescription.CreationTime,\n    }).start();\n    debug('Execution of changeset %s on stack %s has started; waiting for the update to complete...', changeSet.Id, deployName);\n    try {\n      const finalStack = await waitForStackDeploy(cfn, deployName);\n\n      // This shouldn't really happen, but catch it anyway. You never know.\n      if (!finalStack) { throw new Error('Stack deploy failed (the stack disappeared while we were deploying it)'); }\n      cloudFormationStack = finalStack;\n    } finally {\n      await monitor?.stop();\n    }\n    debug('Stack %s has completed updating', deployName);\n  } else {\n    print('Changeset %s created and waiting in review for manual execution (--no-execute)', changeSet.Id);\n  }\n\n  return { noOp: false, outputs: cloudFormationStack.outputs, stackArn: changeSet.StackId! };\n}\n\n/**\n * Prepares the body parameter for +CreateChangeSet+.\n *\n * If the template is small enough to be inlined into the API call, just return\n * it immediately.\n *\n * Otherwise, add it to the asset manifest to get uploaded to the staging\n * bucket and return its coordinates. If there is no staging bucket, an error\n * is thrown.\n *\n * @param stack     the synthesized stack that provides the CloudFormation template\n * @param toolkitInfo information about the toolkit stack\n */\nasync function makeBodyParameter(\n  stack: cxapi.CloudFormationStackArtifact,\n  resolvedEnvironment: cxapi.Environment,\n  assetManifest: AssetManifestBuilder,\n  toolkitInfo: ToolkitInfo,\n  sdk: ISDK): Promise<TemplateBodyParameter> {\n\n  // If the template has already been uploaded to S3, just use it from there.\n  if (stack.stackTemplateAssetObjectUrl) {\n    return { TemplateURL: restUrlFromManifest(stack.stackTemplateAssetObjectUrl, resolvedEnvironment, sdk) };\n  }\n\n  // Otherwise, pass via API call (if small) or upload here (if large)\n  const templateJson = toYAML(stack.template);\n\n  if (templateJson.length <= LARGE_TEMPLATE_SIZE_KB * 1024) {\n    return { TemplateBody: templateJson };\n  }\n\n  if (!toolkitInfo.found) {\n    error(\n      `The template for stack \"${stack.displayName}\" is ${Math.round(templateJson.length / 1024)}KiB. ` +\n      `Templates larger than ${LARGE_TEMPLATE_SIZE_KB}KiB must be uploaded to S3.\\n` +\n      'Run the following command in order to setup an S3 bucket in this environment, and then re-deploy:\\n\\n',\n      chalk.blue(`\\t$ cdk bootstrap ${resolvedEnvironment.name}\\n`));\n\n    throw new Error('Template too large to deploy (\"cdk bootstrap\" is required)');\n  }\n\n  const templateHash = contentHash(templateJson);\n  const key = `cdk/${stack.id}/${templateHash}.yml`;\n\n  assetManifest.addFileAsset(templateHash, {\n    path: stack.templateFile,\n  }, {\n    bucketName: toolkitInfo.bucketName,\n    objectKey: key,\n  });\n\n  const templateURL = `${toolkitInfo.bucketUrl}/${key}`;\n  debug('Storing template in S3 at:', templateURL);\n  return { TemplateURL: templateURL };\n}\n\nexport interface DestroyStackOptions {\n  /**\n   * The stack to be destroyed\n   */\n  stack: cxapi.CloudFormationStackArtifact;\n\n  sdk: ISDK;\n  roleArn?: string;\n  deployName?: string;\n  quiet?: boolean;\n}\n\nexport async function destroyStack(options: DestroyStackOptions) {\n  const deployName = options.deployName || options.stack.stackName;\n  const cfn = options.sdk.cloudFormation();\n\n  const currentStack = await CloudFormationStack.lookup(cfn, deployName);\n  if (!currentStack.exists) {\n    return;\n  }\n  const monitor = options.quiet ? undefined : StackActivityMonitor.withDefaultPrinter(cfn, deployName, options.stack).start();\n\n  try {\n    await cfn.deleteStack({ StackName: deployName, RoleARN: options.roleArn }).promise();\n    const destroyedStack = await waitForStackDelete(cfn, deployName);\n    if (destroyedStack && destroyedStack.stackStatus.name !== 'DELETE_COMPLETE') {\n      throw new Error(`Failed to destroy ${deployName}: ${destroyedStack.stackStatus}`);\n    }\n  } finally {\n    if (monitor) { await monitor.stop(); }\n  }\n}\n\n/**\n * Checks whether we can skip deployment\n *\n * We do this in a complicated way by preprocessing (instead of just\n * looking at the changeset), because if there are nested stacks involved\n * the changeset will always show the nested stacks as needing to be\n * updated, and the deployment will take a long time to in effect not\n * do anything.\n */\nasync function canSkipDeploy(\n  deployStackOptions: DeployStackOptions,\n  cloudFormationStack: CloudFormationStack,\n  parameterChanges: ParameterChanges): Promise<boolean> {\n\n  const deployName = deployStackOptions.deployName || deployStackOptions.stack.stackName;\n  debug(`${deployName}: checking if we can skip deploy`);\n\n  // Forced deploy\n  if (deployStackOptions.force) {\n    debug(`${deployName}: forced deployment`);\n    return false;\n  }\n\n  // Creating changeset only (default true), never skip\n  if (deployStackOptions.execute === false) {\n    debug(`${deployName}: --no-execute, always creating change set`);\n    return false;\n  }\n\n  // No existing stack\n  if (!cloudFormationStack.exists) {\n    debug(`${deployName}: no existing stack`);\n    return false;\n  }\n\n  // Template has changed (assets taken into account here)\n  if (JSON.stringify(deployStackOptions.stack.template) !== JSON.stringify(await cloudFormationStack.template())) {\n    debug(`${deployName}: template has changed`);\n    return false;\n  }\n\n  // Tags have changed\n  if (!compareTags(cloudFormationStack.tags, deployStackOptions.tags ?? [])) {\n    debug(`${deployName}: tags have changed`);\n    return false;\n  }\n\n  // Termination protection has been updated\n  if (!!deployStackOptions.stack.terminationProtection !== !!cloudFormationStack.terminationProtection) {\n    debug(`${deployName}: termination protection has been updated`);\n    return false;\n  }\n\n  // Parameters have changed\n  if (parameterChanges) {\n    if (parameterChanges === 'ssm') {\n      debug(`${deployName}: some parameters come from SSM so we have to assume they may have changed`);\n    } else {\n      debug(`${deployName}: parameters have changed`);\n    }\n    return false;\n  }\n\n  // Existing stack is in a failed state\n  if (cloudFormationStack.stackStatus.isFailure) {\n    debug(`${deployName}: stack is in a failure state`);\n    return false;\n  }\n\n  // We can skip deploy\n  return true;\n}\n\n/**\n * Compares two list of tags, returns true if identical.\n */\nfunction compareTags(a: Tag[], b: Tag[]): boolean {\n  if (a.length !== b.length) {\n    return false;\n  }\n\n  for (const aTag of a) {\n    const bTag = b.find(tag => tag.Key === aTag.Key);\n\n    if (!bTag || bTag.Value !== aTag.Value) {\n      return false;\n    }\n  }\n\n  return true;\n}\n\n/**\n * Format an S3 URL in the manifest for use with CloudFormation\n *\n * Replaces environment placeholders (which this field may contain),\n * and reformats s3://.../... urls into S3 REST URLs (which CloudFormation\n * expects)\n */\nfunction restUrlFromManifest(url: string, environment: cxapi.Environment, sdk: ISDK): string {\n  const doNotUseMarker = '**DONOTUSE**';\n  // This URL may contain placeholders, so still substitute those.\n  url = cxapi.EnvironmentPlaceholders.replace(url, {\n    accountId: environment.account,\n    region: environment.region,\n    partition: doNotUseMarker,\n  });\n\n  // Yes, this is extremely crude, but we don't actually need this so I'm not inclined to spend\n  // a lot of effort trying to thread the right value to this location.\n  if (url.indexOf(doNotUseMarker) > -1) {\n    throw new Error('Cannot use \\'${AWS::Partition}\\' in the \\'stackTemplateAssetObjectUrl\\' field');\n  }\n\n  const s3Url = url.match(/s3:\\/\\/([^/]+)\\/(.*)$/);\n  if (!s3Url) { return url; }\n\n  // We need to pass an 'https://s3.REGION.amazonaws.com[.cn]/bucket/object' URL to CloudFormation, but we\n  // got an 's3://bucket/object' URL instead. Construct the rest API URL here.\n  const bucketName = s3Url[1];\n  const objectKey = s3Url[2];\n\n  const urlSuffix: string = sdk.getEndpointSuffix(environment.region);\n  return `https://s3.${environment.region}.${urlSuffix}/${bucketName}/${objectKey}`;\n}\n"]}
378
+ function suffixWithErrors(msg, errors) {
379
+ return errors && errors.length > 0
380
+ ? `${msg}: ${errors.join(', ')}`
381
+ : msg;
382
+ }
383
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"deploy-stack.js","sourceRoot":"","sources":["deploy-stack.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxapi from '@aws-cdk/cx-api';\nimport * as chalk from 'chalk';\nimport * as fs from 'fs-extra';\nimport * as uuid from 'uuid';\nimport { addMetadataAssetsToManifest } from '../assets';\nimport { Tag } from '../cdk-toolkit';\nimport { debug, error, print } from '../logging';\nimport { toYAML } from '../serialize';\nimport { AssetManifestBuilder } from '../util/asset-manifest-builder';\nimport { publishAssets } from '../util/asset-publishing';\nimport { contentHash } from '../util/content-hash';\nimport { ISDK, SdkProvider } from './aws-auth';\nimport { CfnEvaluationException } from './evaluate-cloudformation-template';\nimport { tryHotswapDeployment } from './hotswap-deployments';\nimport { ICON } from './hotswap/common';\nimport { ToolkitInfo } from './toolkit-info';\nimport {\n  changeSetHasNoChanges, CloudFormationStack, TemplateParameters, waitForChangeSet,\n  waitForStackDeploy, waitForStackDelete, ParameterValues, ParameterChanges, ResourcesToImport,\n} from './util/cloudformation';\nimport { StackActivityMonitor, StackActivityProgress } from './util/cloudformation/stack-activity-monitor';\n\ntype TemplateBodyParameter = {\n  TemplateBody?: string\n  TemplateURL?: string\n};\n\nexport interface DeployStackResult {\n  readonly noOp: boolean;\n  readonly outputs: { [name: string]: string };\n  readonly stackArn: string;\n}\n\nexport interface DeployStackOptions {\n  /**\n   * The stack to be deployed\n   */\n  stack: cxapi.CloudFormationStackArtifact;\n\n  /**\n   * The environment to deploy this stack in\n   *\n   * The environment on the stack artifact may be unresolved, this one\n   * must be resolved.\n   */\n  resolvedEnvironment: cxapi.Environment;\n\n  /**\n   * The SDK to use for deploying the stack\n   *\n   * Should have been initialized with the correct role with which\n   * stack operations should be performed.\n   */\n  sdk: ISDK;\n\n  /**\n   * SDK provider (seeded with default credentials)\n   *\n   * Will exclusively be used to assume publishing credentials (which must\n   * start out from current credentials regardless of whether we've assumed an\n   * action role to touch the stack or not).\n   *\n   * Used for the following purposes:\n   *\n   * - Publish legacy assets.\n   * - Upload large CloudFormation templates to the staging bucket.\n   */\n  sdkProvider: SdkProvider;\n\n  /**\n   * Information about the bootstrap stack found in the target environment\n   */\n  toolkitInfo: ToolkitInfo;\n\n  /**\n   * Role to pass to CloudFormation to execute the change set\n   *\n   * @default - Role specified on stack, otherwise current\n   */\n  roleArn?: string;\n\n  /**\n   * Notification ARNs to pass to CloudFormation to notify when the change set has completed\n   *\n   * @default - No notifications\n   */\n  notificationArns?: string[];\n\n  /**\n   * Name to deploy the stack under\n   *\n   * @default - Name from assembly\n   */\n  deployName?: string;\n\n  /**\n   * Quiet or verbose deployment\n   *\n   * @default false\n   */\n  quiet?: boolean;\n\n  /**\n   * List of asset IDs which shouldn't be built\n   *\n   * @default - Build all assets\n   */\n  reuseAssets?: string[];\n\n  /**\n   * Tags to pass to CloudFormation to add to stack\n   *\n   * @default - No tags\n   */\n  tags?: Tag[];\n\n  /**\n   * Whether to execute the changeset or leave it in review.\n   *\n   * @default true\n   */\n  execute?: boolean;\n\n  /**\n   * Optional name to use for the CloudFormation change set.\n   * If not provided, a name will be generated automatically.\n   */\n  changeSetName?: string;\n\n  /**\n   * The collection of extra parameters\n   * (in addition to those used for assets)\n   * to pass to the deployed template.\n   * Note that parameters with `undefined` or empty values will be ignored,\n   * and not passed to the template.\n   *\n   * @default - no additional parameters will be passed to the template\n   */\n  parameters?: { [name: string]: string | undefined };\n\n  /**\n   * Use previous values for unspecified parameters\n   *\n   * If not set, all parameters must be specified for every deployment.\n   *\n   * @default false\n   */\n  usePreviousParameters?: boolean;\n\n  /**\n   * Display mode for stack deployment progress.\n   *\n   * @default StackActivityProgress.Bar stack events will be displayed for\n   *   the resource currently being deployed.\n   */\n  progress?: StackActivityProgress;\n\n  /**\n   * Deploy even if the deployed template is identical to the one we are about to deploy.\n   * @default false\n   */\n  force?: boolean;\n\n  /**\n   * Whether we are on a CI system\n   *\n   * @default false\n   */\n  readonly ci?: boolean;\n\n  /**\n   * Rollback failed deployments\n   *\n   * @default true\n   */\n  readonly rollback?: boolean;\n\n  /*\n   * Whether to perform a 'hotswap' deployment.\n   * A 'hotswap' deployment will attempt to short-circuit CloudFormation\n   * and update the affected resources like Lambda functions directly.\n   *\n   * @default - false for regular deployments, true for 'watch' deployments\n   */\n  readonly hotswap?: boolean;\n\n  /**\n   * The extra string to append to the User-Agent header when performing AWS SDK calls.\n   *\n   * @default - nothing extra is appended to the User-Agent header\n   */\n  readonly extraUserAgent?: string;\n\n  /**\n   * If set, change set of type IMPORT will be created, and resourcesToImport\n   * passed to it.\n   */\n  readonly resourcesToImport?: ResourcesToImport;\n\n  /**\n   * If present, use this given template instead of the stored one\n   *\n   * @default - Use the stored template\n   */\n  readonly overrideTemplate?: any;\n}\n\nconst LARGE_TEMPLATE_SIZE_KB = 50;\n\nexport async function deployStack(options: DeployStackOptions): Promise<DeployStackResult> {\n  const stackArtifact = options.stack;\n\n  const stackEnv = options.resolvedEnvironment;\n\n  options.sdk.appendCustomUserAgent(options.extraUserAgent);\n  const cfn = options.sdk.cloudFormation();\n  const deployName = options.deployName || stackArtifact.stackName;\n  let cloudFormationStack = await CloudFormationStack.lookup(cfn, deployName);\n\n  if (cloudFormationStack.stackStatus.isCreationFailure) {\n    debug(`Found existing stack ${deployName} that had previously failed creation. Deleting it before attempting to re-create it.`);\n    await cfn.deleteStack({ StackName: deployName }).promise();\n    const deletedStack = await waitForStackDelete(cfn, deployName);\n    if (deletedStack && deletedStack.stackStatus.name !== 'DELETE_COMPLETE') {\n      throw new Error(`Failed deleting stack ${deployName} that had previously failed creation (current state: ${deletedStack.stackStatus})`);\n    }\n    // Update variable to mark that the stack does not exist anymore, but avoid\n    // doing an actual lookup in CloudFormation (which would be silly to do if\n    // we just deleted it).\n    cloudFormationStack = CloudFormationStack.doesNotExist(cfn, deployName);\n  }\n\n  // Detect \"legacy\" assets (which remain in the metadata) and publish them via\n  // an ad-hoc asset manifest, while passing their locations via template\n  // parameters.\n  const legacyAssets = new AssetManifestBuilder();\n  const assetParams = await addMetadataAssetsToManifest(stackArtifact, legacyAssets, options.toolkitInfo, options.reuseAssets);\n\n  const finalParameterValues = { ...options.parameters, ...assetParams };\n\n  const templateParams = TemplateParameters.fromTemplate(stackArtifact.template);\n  const stackParams = options.usePreviousParameters\n    ? templateParams.updateExisting(finalParameterValues, cloudFormationStack.parameters)\n    : templateParams.supplyAll(finalParameterValues);\n\n  if (await canSkipDeploy(options, cloudFormationStack, stackParams.hasChanges(cloudFormationStack.parameters))) {\n    debug(`${deployName}: skipping deployment (use --force to override)`);\n    // if we can skip deployment and we are performing a hotswap, let the user know\n    // that no hotswap deployment happened\n    if (options.hotswap) {\n      print(`\\n ${ICON} %s\\n`, chalk.bold('hotswap deployment skipped - no changes were detected (use --force to override)'));\n    }\n    return {\n      noOp: true,\n      outputs: cloudFormationStack.outputs,\n      stackArn: cloudFormationStack.stackId,\n    };\n  } else {\n    debug(`${deployName}: deploying...`);\n  }\n\n  const bodyParameter = await makeBodyParameter(\n    stackArtifact,\n    options.resolvedEnvironment,\n    legacyAssets,\n    options.toolkitInfo,\n    options.sdk,\n    options.overrideTemplate);\n  await publishAssets(legacyAssets.toManifest(stackArtifact.assembly.directory), options.sdkProvider, stackEnv);\n\n  if (options.hotswap) {\n    // attempt to short-circuit the deployment if possible\n    try {\n      const hotswapDeploymentResult = await tryHotswapDeployment(options.sdkProvider, assetParams, cloudFormationStack, stackArtifact);\n      if (hotswapDeploymentResult) {\n        return hotswapDeploymentResult;\n      }\n      print('Could not perform a hotswap deployment, as the stack %s contains non-Asset changes', stackArtifact.displayName);\n    } catch (e) {\n      if (!(e instanceof CfnEvaluationException)) {\n        throw e;\n      }\n      print('Could not perform a hotswap deployment, because the CloudFormation template could not be resolved: %s', e.message);\n    }\n    print('Falling back to doing a full deployment');\n  }\n\n  // could not short-circuit the deployment, perform a full CFN deploy instead\n  return prepareAndExecuteChangeSet(options, cloudFormationStack, stackArtifact, stackParams, bodyParameter);\n}\n\nasync function prepareAndExecuteChangeSet(\n  options: DeployStackOptions, cloudFormationStack: CloudFormationStack,\n  stackArtifact: cxapi.CloudFormationStackArtifact, stackParams: ParameterValues, bodyParameter: TemplateBodyParameter,\n): Promise<DeployStackResult> {\n  // if we got here, and hotswap is enabled, that means changes couldn't be hotswapped,\n  // and we had to fall back on a full deployment. Note that fact in our User-Agent\n  if (options.hotswap) {\n    options.sdk.appendCustomUserAgent('cdk-hotswap/fallback');\n  }\n\n  const cfn = options.sdk.cloudFormation();\n  const deployName = options.deployName ?? stackArtifact.stackName;\n\n  const changeSetName = options.changeSetName ?? 'cdk-deploy-change-set';\n  if (cloudFormationStack.exists) {\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    debug(`Removing existing change set with name ${changeSetName} if it exists`);\n    await cfn.deleteChangeSet({ StackName: deployName, ChangeSetName: changeSetName }).promise();\n  }\n\n  const update = cloudFormationStack.exists && cloudFormationStack.stackStatus.name !== 'REVIEW_IN_PROGRESS';\n\n  debug(`Attempting to create ChangeSet with name ${changeSetName} to ${update ? 'update' : 'create'} stack ${deployName}`);\n  print('%s: creating CloudFormation changeset...', chalk.bold(deployName));\n  const executionId = uuid.v4();\n  const changeSet = await cfn.createChangeSet({\n    StackName: deployName,\n    ChangeSetName: changeSetName,\n    ChangeSetType: options.resourcesToImport ? 'IMPORT' : update ? 'UPDATE' : 'CREATE',\n    ResourcesToImport: options.resourcesToImport,\n    Description: `CDK Changeset for execution ${executionId}`,\n    TemplateBody: bodyParameter.TemplateBody,\n    TemplateURL: bodyParameter.TemplateURL,\n    Parameters: stackParams.apiParameters,\n    RoleARN: options.roleArn,\n    NotificationARNs: options.notificationArns,\n    Capabilities: ['CAPABILITY_IAM', 'CAPABILITY_NAMED_IAM', 'CAPABILITY_AUTO_EXPAND'],\n    Tags: options.tags,\n  }).promise();\n  debug('Initiated creation of changeset: %s; waiting for it to finish creating...', changeSet.Id);\n  const changeSetDescription = await waitForChangeSet(cfn, deployName, changeSetName);\n\n  // Update termination protection only if it has changed.\n  const terminationProtection = stackArtifact.terminationProtection ?? false;\n  if (!!cloudFormationStack.terminationProtection !== terminationProtection) {\n    debug('Updating termination protection from %s to %s for stack %s', cloudFormationStack.terminationProtection, terminationProtection, deployName);\n    await cfn.updateTerminationProtection({\n      StackName: deployName,\n      EnableTerminationProtection: terminationProtection,\n    }).promise();\n    debug('Termination protection updated to %s for stack %s', terminationProtection, deployName);\n  }\n\n  if (changeSetHasNoChanges(changeSetDescription)) {\n    debug('No changes are to be performed on %s.', deployName);\n    if (options.execute) {\n      debug('Deleting empty change set %s', changeSet.Id);\n      await cfn.deleteChangeSet({ StackName: deployName, ChangeSetName: changeSetName }).promise();\n    }\n    return { noOp: true, outputs: cloudFormationStack.outputs, stackArn: changeSet.StackId! };\n  }\n\n  const execute = options.execute === undefined ? true : options.execute;\n  if (execute) {\n    debug('Initiating execution of changeset %s on stack %s', changeSet.Id, deployName);\n\n    const shouldDisableRollback = options.rollback === false;\n    // Do a bit of contortions to only pass the `DisableRollback` flag if it's true. That way,\n    // CloudFormation won't balk at the unrecognized option in regions where the feature is not available yet.\n    const disableRollback = shouldDisableRollback ? { DisableRollback: true } : undefined;\n\n    await cfn.executeChangeSet({ StackName: deployName, ChangeSetName: changeSetName, ...disableRollback }).promise();\n\n    // eslint-disable-next-line max-len\n    const changeSetLength: number = (changeSetDescription.Changes ?? []).length;\n    const monitor = options.quiet ? undefined : StackActivityMonitor.withDefaultPrinter(cfn, deployName, stackArtifact, {\n      // +1 for the extra event emitted from updates.\n      resourcesTotal: cloudFormationStack.exists ? changeSetLength + 1 : changeSetLength,\n      progress: options.progress,\n      changeSetCreationTime: changeSetDescription.CreationTime,\n    }).start();\n    debug('Execution of changeset %s on stack %s has started; waiting for the update to complete...', changeSet.Id, deployName);\n    try {\n      const finalStack = await waitForStackDeploy(cfn, deployName);\n\n      // This shouldn't really happen, but catch it anyway. You never know.\n      if (!finalStack) { throw new Error('Stack deploy failed (the stack disappeared while we were deploying it)'); }\n      cloudFormationStack = finalStack;\n    } catch (e) {\n      throw new Error(suffixWithErrors(e.message, monitor?.errors));\n    } finally {\n      await monitor?.stop();\n    }\n    debug('Stack %s has completed updating', deployName);\n  } else {\n    print('Changeset %s created and waiting in review for manual execution (--no-execute)', changeSet.Id);\n  }\n\n  return { noOp: false, outputs: cloudFormationStack.outputs, stackArn: changeSet.StackId! };\n}\n\n/**\n * Prepares the body parameter for +CreateChangeSet+.\n *\n * If the template is small enough to be inlined into the API call, just return\n * it immediately.\n *\n * Otherwise, add it to the asset manifest to get uploaded to the staging\n * bucket and return its coordinates. If there is no staging bucket, an error\n * is thrown.\n *\n * @param stack     the synthesized stack that provides the CloudFormation template\n * @param toolkitInfo information about the toolkit stack\n */\nasync function makeBodyParameter(\n  stack: cxapi.CloudFormationStackArtifact,\n  resolvedEnvironment: cxapi.Environment,\n  assetManifest: AssetManifestBuilder,\n  toolkitInfo: ToolkitInfo,\n  sdk: ISDK,\n  overrideTemplate?: any,\n): Promise<TemplateBodyParameter> {\n\n  // If the template has already been uploaded to S3, just use it from there.\n  if (stack.stackTemplateAssetObjectUrl && !overrideTemplate) {\n    return { TemplateURL: restUrlFromManifest(stack.stackTemplateAssetObjectUrl, resolvedEnvironment, sdk) };\n  }\n\n  // Otherwise, pass via API call (if small) or upload here (if large)\n  const templateJson = toYAML(overrideTemplate ?? stack.template);\n\n  if (templateJson.length <= LARGE_TEMPLATE_SIZE_KB * 1024) {\n    return { TemplateBody: templateJson };\n  }\n\n  if (!toolkitInfo.found) {\n    error(\n      `The template for stack \"${stack.displayName}\" is ${Math.round(templateJson.length / 1024)}KiB. ` +\n      `Templates larger than ${LARGE_TEMPLATE_SIZE_KB}KiB must be uploaded to S3.\\n` +\n      'Run the following command in order to setup an S3 bucket in this environment, and then re-deploy:\\n\\n',\n      chalk.blue(`\\t$ cdk bootstrap ${resolvedEnvironment.name}\\n`));\n\n    throw new Error('Template too large to deploy (\"cdk bootstrap\" is required)');\n  }\n\n  const templateHash = contentHash(templateJson);\n  const key = `cdk/${stack.id}/${templateHash}.yml`;\n\n  let templateFile = stack.templateFile;\n  if (overrideTemplate) {\n    // Add a variant of this template\n    templateFile = `${stack.templateFile}-${templateHash}.yaml`;\n    await fs.writeFile(templateFile, templateJson, { encoding: 'utf-8' });\n  }\n\n  assetManifest.addFileAsset(templateHash, {\n    path: templateFile,\n  }, {\n    bucketName: toolkitInfo.bucketName,\n    objectKey: key,\n  });\n\n  const templateURL = `${toolkitInfo.bucketUrl}/${key}`;\n  debug('Storing template in S3 at:', templateURL);\n  return { TemplateURL: templateURL };\n}\n\n/**\n * Prepare a body parameter for CFN, performing the upload\n *\n * Return it as-is if it is small enough to pass in the API call,\n * upload to S3 and return the coordinates if it is not.\n */\nexport async function makeBodyParameterAndUpload(\n  stack: cxapi.CloudFormationStackArtifact,\n  resolvedEnvironment: cxapi.Environment,\n  toolkitInfo: ToolkitInfo,\n  sdkProvider: SdkProvider,\n  sdk: ISDK,\n  overrideTemplate?: any): Promise<TemplateBodyParameter> {\n\n  // We don't have access to the actual asset manifest here, so pretend that the\n  // stack doesn't have a pre-published URL.\n  const forceUploadStack = Object.create(stack, {\n    stackTemplateAssetObjectUrl: { value: undefined },\n  });\n\n  const builder = new AssetManifestBuilder();\n  const bodyparam = await makeBodyParameter(forceUploadStack, resolvedEnvironment, builder, toolkitInfo, sdk, overrideTemplate);\n  const manifest = builder.toManifest(stack.assembly.directory);\n  await publishAssets(manifest, sdkProvider, resolvedEnvironment, { quiet: true });\n  return bodyparam;\n}\n\nexport interface DestroyStackOptions {\n  /**\n   * The stack to be destroyed\n   */\n  stack: cxapi.CloudFormationStackArtifact;\n\n  sdk: ISDK;\n  roleArn?: string;\n  deployName?: string;\n  quiet?: boolean;\n}\n\nexport async function destroyStack(options: DestroyStackOptions) {\n  const deployName = options.deployName || options.stack.stackName;\n  const cfn = options.sdk.cloudFormation();\n\n  const currentStack = await CloudFormationStack.lookup(cfn, deployName);\n  if (!currentStack.exists) {\n    return;\n  }\n  const monitor = options.quiet ? undefined : StackActivityMonitor.withDefaultPrinter(cfn, deployName, options.stack).start();\n\n  try {\n    await cfn.deleteStack({ StackName: deployName, RoleARN: options.roleArn }).promise();\n    const destroyedStack = await waitForStackDelete(cfn, deployName);\n    if (destroyedStack && destroyedStack.stackStatus.name !== 'DELETE_COMPLETE') {\n      throw new Error(`Failed to destroy ${deployName}: ${destroyedStack.stackStatus}`);\n    }\n  } catch (e) {\n    throw new Error(suffixWithErrors(e.message, monitor?.errors));\n  } finally {\n    if (monitor) { await monitor.stop(); }\n  }\n}\n\n/**\n * Checks whether we can skip deployment\n *\n * We do this in a complicated way by preprocessing (instead of just\n * looking at the changeset), because if there are nested stacks involved\n * the changeset will always show the nested stacks as needing to be\n * updated, and the deployment will take a long time to in effect not\n * do anything.\n */\nasync function canSkipDeploy(\n  deployStackOptions: DeployStackOptions,\n  cloudFormationStack: CloudFormationStack,\n  parameterChanges: ParameterChanges): Promise<boolean> {\n\n  const deployName = deployStackOptions.deployName || deployStackOptions.stack.stackName;\n  debug(`${deployName}: checking if we can skip deploy`);\n\n  // Forced deploy\n  if (deployStackOptions.force) {\n    debug(`${deployName}: forced deployment`);\n    return false;\n  }\n\n  // Creating changeset only (default true), never skip\n  if (deployStackOptions.execute === false) {\n    debug(`${deployName}: --no-execute, always creating change set`);\n    return false;\n  }\n\n  // No existing stack\n  if (!cloudFormationStack.exists) {\n    debug(`${deployName}: no existing stack`);\n    return false;\n  }\n\n  // Template has changed (assets taken into account here)\n  if (JSON.stringify(deployStackOptions.stack.template) !== JSON.stringify(await cloudFormationStack.template())) {\n    debug(`${deployName}: template has changed`);\n    return false;\n  }\n\n  // Tags have changed\n  if (!compareTags(cloudFormationStack.tags, deployStackOptions.tags ?? [])) {\n    debug(`${deployName}: tags have changed`);\n    return false;\n  }\n\n  // Termination protection has been updated\n  if (!!deployStackOptions.stack.terminationProtection !== !!cloudFormationStack.terminationProtection) {\n    debug(`${deployName}: termination protection has been updated`);\n    return false;\n  }\n\n  // Parameters have changed\n  if (parameterChanges) {\n    if (parameterChanges === 'ssm') {\n      debug(`${deployName}: some parameters come from SSM so we have to assume they may have changed`);\n    } else {\n      debug(`${deployName}: parameters have changed`);\n    }\n    return false;\n  }\n\n  // Existing stack is in a failed state\n  if (cloudFormationStack.stackStatus.isFailure) {\n    debug(`${deployName}: stack is in a failure state`);\n    return false;\n  }\n\n  // We can skip deploy\n  return true;\n}\n\n/**\n * Compares two list of tags, returns true if identical.\n */\nfunction compareTags(a: Tag[], b: Tag[]): boolean {\n  if (a.length !== b.length) {\n    return false;\n  }\n\n  for (const aTag of a) {\n    const bTag = b.find(tag => tag.Key === aTag.Key);\n\n    if (!bTag || bTag.Value !== aTag.Value) {\n      return false;\n    }\n  }\n\n  return true;\n}\n\n/**\n * Format an S3 URL in the manifest for use with CloudFormation\n *\n * Replaces environment placeholders (which this field may contain),\n * and reformats s3://.../... urls into S3 REST URLs (which CloudFormation\n * expects)\n */\nfunction restUrlFromManifest(url: string, environment: cxapi.Environment, sdk: ISDK): string {\n  const doNotUseMarker = '**DONOTUSE**';\n  // This URL may contain placeholders, so still substitute those.\n  url = cxapi.EnvironmentPlaceholders.replace(url, {\n    accountId: environment.account,\n    region: environment.region,\n    partition: doNotUseMarker,\n  });\n\n  // Yes, this is extremely crude, but we don't actually need this so I'm not inclined to spend\n  // a lot of effort trying to thread the right value to this location.\n  if (url.indexOf(doNotUseMarker) > -1) {\n    throw new Error('Cannot use \\'${AWS::Partition}\\' in the \\'stackTemplateAssetObjectUrl\\' field');\n  }\n\n  const s3Url = url.match(/s3:\\/\\/([^/]+)\\/(.*)$/);\n  if (!s3Url) { return url; }\n\n  // We need to pass an 'https://s3.REGION.amazonaws.com[.cn]/bucket/object' URL to CloudFormation, but we\n  // got an 's3://bucket/object' URL instead. Construct the rest API URL here.\n  const bucketName = s3Url[1];\n  const objectKey = s3Url[2];\n\n  const urlSuffix: string = sdk.getEndpointSuffix(environment.region);\n  return `https://s3.${environment.region}.${urlSuffix}/${bucketName}/${objectKey}`;\n}\n\nfunction suffixWithErrors(msg: string, errors?: string[]) {\n  return errors && errors.length > 0\n    ? `${msg}: ${errors.join(', ')}`\n    : msg;\n}\n"]}