aws-cdk 2.29.0 → 2.31.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CONTRIBUTING.md CHANGED
@@ -89,6 +89,20 @@ You can also run just these tests by executing:
89
89
  yarn integ-cli-no-regression
90
90
  ```
91
91
 
92
+ ##### Warning
93
+
94
+ Since the tests take a long time to run, we run them in parallel in order to minimize running time. Jest does not have
95
+ good support for parallelism, the only thing that exists is `test.concurrent()` and it has a couple of limitations:
96
+
97
+ - It's not possible to only run a subset of tests, all tests will execute (the reason for this is that it will start all
98
+ tests in parallel, but only `await` your selected subset specified with the `-t TESTNAME` option. However, all tests
99
+ are running and Node will not exit until they're all finished).
100
+ - It's not possible to use `beforeEach()` and `afterEach()`.
101
+
102
+ Because of the first limitation, concurrency is only enabled on the build server (via the `JEST_TEST_CONCURRENT`
103
+ environment variable), not locally. Note: tests using `beforeEach()` will appear to work locally, but will fail on the
104
+ build server! Don't use it!
105
+
92
106
  #### Regression
93
107
 
94
108
  Validate that previously tested functionality still works in light of recent changes to the CLI. This is done by fetching the functional tests of the previous published release, and running them against the new CLI code.
package/README.md CHANGED
@@ -27,6 +27,13 @@ Command | Description
27
27
  [`cdk acknowledge`](#cdk-acknowledge) | Acknowledge (and hide) a notice by issue number
28
28
  [`cdk notices`](#cdk-notices) | List all relevant notices for the application
29
29
 
30
+ - [Bundling](#bundling)
31
+ - [MFA Support](#mfa-support)
32
+ - [SSO Support](#sso-support)
33
+ - [Configuration](#configuration)
34
+ - [Running in CI](#running-in-ci)
35
+
36
+
30
37
  This module is part of the [AWS Cloud Development Kit](https://github.com/aws/aws-cdk) project.
31
38
 
32
39
  ## Commands
@@ -714,3 +721,9 @@ The following environment variables affect aws-cdk:
714
721
 
715
722
  - `CDK_DISABLE_VERSION_CHECK`: If set, disable automatic check for newer versions.
716
723
  - `CDK_NEW_BOOTSTRAP`: use the modern bootstrapping stack.
724
+
725
+ ### Running in CI
726
+
727
+ The CLI will attempt to detect whether it is being run in CI by looking for the presence of an
728
+ environment variable `CI=true`. This can be forced by passing the `--ci` flag. By default the CLI
729
+ sends most of its logs to `stderr`, but when `ci=true` it will send the logs to `stdout` instead.
package/build-info.json CHANGED
@@ -1,4 +1,4 @@
1
1
  {
2
- "comment": "Generated at 2022-06-22T22:22:31Z by generate.sh",
3
- "commit": "47d7ec4"
2
+ "comment": "Generated at 2022-07-06T19:57:34Z by generate.sh",
3
+ "commit": "b67950d"
4
4
  }
@@ -165,6 +165,7 @@ export interface DestroyStackOptions {
165
165
  roleArn?: string;
166
166
  quiet?: boolean;
167
167
  force?: boolean;
168
+ ci?: boolean;
168
169
  }
169
170
  export interface StackExistsOptions {
170
171
  stack: cxapi.CloudFormationStackArtifact;
@@ -148,6 +148,7 @@ class CloudFormationDeployments {
148
148
  stack: options.stack,
149
149
  deployName: options.deployName,
150
150
  quiet: options.quiet,
151
+ ci: options.ci,
151
152
  });
152
153
  }
153
154
  async stackExists(options) {
@@ -229,4 +230,4 @@ class CloudFormationDeployments {
229
230
  }
230
231
  }
231
232
  exports.CloudFormationDeployments = CloudFormationDeployments;
232
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloudformation-deployments.js","sourceRoot":"","sources":["cloudformation-deployments.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxapi from '@aws-cdk/cx-api';\nimport { AssetManifest } from 'cdk-assets';\nimport { Tag } from '../cdk-toolkit';\nimport { debug, warning } from '../logging';\nimport { publishAssets } from '../util/asset-publishing';\nimport { Mode } from './aws-auth/credentials';\nimport { ISDK } from './aws-auth/sdk';\nimport { SdkProvider } from './aws-auth/sdk-provider';\nimport { deployStack, DeployStackResult, destroyStack, makeBodyParameterAndUpload } from './deploy-stack';\nimport { loadCurrentTemplateWithNestedStacks, loadCurrentTemplate } from './nested-stack-helpers';\nimport { ToolkitInfo } from './toolkit-info';\nimport { CloudFormationStack, Template, ResourcesToImport, ResourceIdentifierSummaries } from './util/cloudformation';\nimport { StackActivityProgress } from './util/cloudformation/stack-activity-monitor';\nimport { replaceEnvPlaceholders } from './util/placeholders';\n\n/**\n * SDK obtained by assuming the lookup role\n * for a given environment\n */\nexport interface PreparedSdkWithLookupRoleForEnvironment {\n  /**\n   * The SDK for the given environment\n   */\n  readonly sdk: ISDK;\n\n  /**\n   * The resolved environment for the stack\n   * (no more 'unknown-account/unknown-region')\n   */\n  readonly resolvedEnvironment: cxapi.Environment;\n\n  /**\n   * Whether or not the assume role was successful.\n   * If the assume role was not successful (false)\n   * then that means that the 'sdk' returned contains\n   * the default credentials (not the assume role credentials)\n   */\n  readonly didAssumeRole: boolean;\n}\n\n/**\n  * Try to use the bootstrap lookupRole. There are two scenarios that are handled here\n  *  1. The lookup role may not exist (it was added in bootstrap stack version 7)\n  *  2. The lookup role may not have the correct permissions (ReadOnlyAccess was added in\n  *      bootstrap stack version 8)\n  *\n  * In the case of 1 (lookup role doesn't exist) `forEnvironment` will either:\n  *   1. Return the default credentials if the default credentials are for the stack account\n  *   2. Throw an error if the default credentials are not for the stack account.\n  *\n  * If we successfully assume the lookup role we then proceed to 2 and check whether the bootstrap\n  * stack version is valid. If it is not we throw an error which should be handled in the calling\n  * function (and fallback to use a different role, etc)\n  *\n  * If we do not successfully assume the lookup role, but do get back the default credentials\n  * then return those and note that we are returning the default credentials. The calling\n  * function can then decide to use them or fallback to another role.\n  */\nexport async function prepareSdkWithLookupRoleFor(\n  sdkProvider: SdkProvider,\n  stack: cxapi.CloudFormationStackArtifact,\n): Promise<PreparedSdkWithLookupRoleForEnvironment> {\n  const resolvedEnvironment = await sdkProvider.resolveEnvironment(stack.environment);\n\n  // Substitute any placeholders with information about the current environment\n  const arns = await replaceEnvPlaceholders({\n    lookupRoleArn: stack.lookupRole?.arn,\n  }, resolvedEnvironment, sdkProvider);\n\n  // try to assume the lookup role\n  const warningMessage = `Could not assume ${arns.lookupRoleArn}, proceeding anyway.`;\n  const upgradeMessage = `(To get rid of this warning, please upgrade to bootstrap version >= ${stack.lookupRole?.requiresBootstrapStackVersion})`;\n  try {\n    const stackSdk = await sdkProvider.forEnvironment(resolvedEnvironment, Mode.ForReading, {\n      assumeRoleArn: arns.lookupRoleArn,\n      assumeRoleExternalId: stack.lookupRole?.assumeRoleExternalId,\n    });\n\n    // if we succeed in assuming the lookup role, make sure we have the correct bootstrap stack version\n    if (stackSdk.didAssumeRole && stack.lookupRole?.bootstrapStackVersionSsmParameter && stack.lookupRole.requiresBootstrapStackVersion) {\n      const version = await ToolkitInfo.versionFromSsmParameter(stackSdk.sdk, stack.lookupRole.bootstrapStackVersionSsmParameter);\n      if (version < stack.lookupRole.requiresBootstrapStackVersion) {\n        throw new Error(`Bootstrap stack version '${stack.lookupRole.requiresBootstrapStackVersion}' is required, found version '${version}'.`);\n      }\n      // we may not have assumed the lookup role because one was not provided\n      // if that is the case then don't print the upgrade warning\n    } else if (!stackSdk.didAssumeRole && stack.lookupRole?.requiresBootstrapStackVersion) {\n      warning(upgradeMessage);\n    }\n    return { ...stackSdk, resolvedEnvironment };\n  } catch (e) {\n    debug(e);\n    // only print out the warnings if the lookupRole exists AND there is a required\n    // bootstrap version, otherwise the warnings will print `undefined`\n    if (stack.lookupRole && stack.lookupRole.requiresBootstrapStackVersion) {\n      warning(warningMessage);\n      warning(upgradeMessage);\n    }\n    throw (e);\n  }\n}\n\nexport interface DeployStackOptions {\n  /**\n   * Stack to deploy\n   */\n  stack: cxapi.CloudFormationStackArtifact;\n\n  /**\n   * Execution role for the deployment (pass through to CloudFormation)\n   *\n   * @default - Current role\n   */\n  roleArn?: string;\n\n  /**\n   * Topic ARNs to send a message when deployment finishes (pass through to CloudFormation)\n   *\n   * @default - No notifications\n   */\n  notificationArns?: string[];\n\n  /**\n   * Override name under which stack will be deployed\n   *\n   * @default - Use artifact default\n   */\n  deployName?: string;\n\n  /**\n   * Don't show stack deployment events, just wait\n   *\n   * @default false\n   */\n  quiet?: boolean;\n\n  /**\n   * Name of the toolkit stack, if not the default name\n   *\n   * @default 'CDKToolkit'\n   */\n  toolkitStackName?: string;\n\n  /**\n   * List of asset IDs which should NOT be built or uploaded\n   *\n   * @default - Build all assets\n   */\n  reuseAssets?: string[];\n\n  /**\n   * Stack tags (pass through to CloudFormation)\n   */\n  tags?: Tag[];\n\n  /**\n   * Stage the change set but don't execute it\n   *\n   * @default - false\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   * Force deployment, even if the deployed template is identical to the one we are about to deploy.\n   * @default false deployment will be skipped if the template is identical\n   */\n  force?: boolean;\n\n  /**\n   * Extra parameters for CloudFormation\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 true\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   * 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   * List of existing resources to be IMPORTED into the stack, instead of being CREATED\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\nexport interface DestroyStackOptions {\n  stack: cxapi.CloudFormationStackArtifact;\n  deployName?: string;\n  roleArn?: string;\n  quiet?: boolean;\n  force?: boolean;\n}\n\nexport interface StackExistsOptions {\n  stack: cxapi.CloudFormationStackArtifact;\n  deployName?: string;\n}\n\nexport interface ProvisionerProps {\n  sdkProvider: SdkProvider;\n}\n\n/**\n * SDK obtained by assuming the deploy role\n * for a given environment\n */\nexport interface PreparedSdkForEnvironment {\n  /**\n   * The SDK for the given environment\n   */\n  readonly stackSdk: ISDK;\n\n  /**\n   * The resolved environment for the stack\n   * (no more 'unknown-account/unknown-region')\n   */\n  readonly resolvedEnvironment: cxapi.Environment;\n  /**\n   * The Execution Role that should be passed to CloudFormation.\n   *\n   * @default - no execution role is used\n   */\n  readonly cloudFormationRoleArn?: string;\n}\n\n/**\n * Helper class for CloudFormation deployments\n *\n * Looks us the right SDK and Bootstrap stack to deploy a given\n * stack artifact.\n */\nexport class CloudFormationDeployments {\n  private readonly sdkProvider: SdkProvider;\n\n  constructor(props: ProvisionerProps) {\n    this.sdkProvider = props.sdkProvider;\n  }\n\n  public async readCurrentTemplateWithNestedStacks(rootStackArtifact: cxapi.CloudFormationStackArtifact): Promise<Template> {\n    const sdk = (await this.prepareSdkWithLookupOrDeployRole(rootStackArtifact)).stackSdk;\n    return (await loadCurrentTemplateWithNestedStacks(rootStackArtifact, sdk)).deployedTemplate;\n  }\n\n  public async readCurrentTemplate(stackArtifact: cxapi.CloudFormationStackArtifact): Promise<Template> {\n    debug(`Reading existing template for stack ${stackArtifact.displayName}.`);\n    const sdk = (await this.prepareSdkWithLookupOrDeployRole(stackArtifact)).stackSdk;\n    return loadCurrentTemplate(stackArtifact, sdk);\n  }\n\n  public async resourceIdentifierSummaries(\n    stackArtifact: cxapi.CloudFormationStackArtifact,\n    toolkitStackName?: string,\n  ): Promise<ResourceIdentifierSummaries> {\n    debug(`Retrieving template summary for stack ${stackArtifact.displayName}.`);\n    // Currently, needs to use `deploy-role` since it may need to read templates in the staging\n    // bucket which have been encrypted with a KMS key (and lookup-role may not read encrypted things)\n    const { stackSdk, resolvedEnvironment } = await this.prepareSdkFor(stackArtifact, undefined, Mode.ForReading);\n    const cfn = stackSdk.cloudFormation();\n\n    const toolkitInfo = await ToolkitInfo.lookup(resolvedEnvironment, stackSdk, toolkitStackName);\n\n    // Upload the template, if necessary, before passing it to CFN\n    const cfnParam = await makeBodyParameterAndUpload(\n      stackArtifact,\n      resolvedEnvironment,\n      toolkitInfo,\n      this.sdkProvider,\n      stackSdk);\n\n    const response = await cfn.getTemplateSummary(cfnParam).promise();\n    if (!response.ResourceIdentifierSummaries) {\n      debug('GetTemplateSummary API call did not return \"ResourceIdentifierSummaries\"');\n    }\n    return response.ResourceIdentifierSummaries ?? [];\n  }\n\n  public async deployStack(options: DeployStackOptions): Promise<DeployStackResult> {\n    const { stackSdk, resolvedEnvironment, cloudFormationRoleArn } = await this.prepareSdkFor(options.stack, options.roleArn);\n\n    const toolkitInfo = await ToolkitInfo.lookup(resolvedEnvironment, stackSdk, options.toolkitStackName);\n\n    // Publish any assets before doing the actual deploy (do not publish any assets on import operation)\n    if (options.resourcesToImport === undefined) {\n      await this.publishStackAssets(options.stack, toolkitInfo);\n    }\n\n    // Do a verification of the bootstrap stack version\n    await this.validateBootstrapStackVersion(\n      options.stack.stackName,\n      options.stack.requiresBootstrapStackVersion,\n      options.stack.bootstrapStackVersionSsmParameter,\n      toolkitInfo);\n\n    return deployStack({\n      stack: options.stack,\n      resolvedEnvironment,\n      deployName: options.deployName,\n      notificationArns: options.notificationArns,\n      quiet: options.quiet,\n      sdk: stackSdk,\n      sdkProvider: this.sdkProvider,\n      roleArn: cloudFormationRoleArn,\n      reuseAssets: options.reuseAssets,\n      toolkitInfo,\n      tags: options.tags,\n      execute: options.execute,\n      changeSetName: options.changeSetName,\n      force: options.force,\n      parameters: options.parameters,\n      usePreviousParameters: options.usePreviousParameters,\n      progress: options.progress,\n      ci: options.ci,\n      rollback: options.rollback,\n      hotswap: options.hotswap,\n      extraUserAgent: options.extraUserAgent,\n      resourcesToImport: options.resourcesToImport,\n      overrideTemplate: options.overrideTemplate,\n    });\n  }\n\n  public async destroyStack(options: DestroyStackOptions): Promise<void> {\n    const { stackSdk, cloudFormationRoleArn: roleArn } = await this.prepareSdkFor(options.stack, options.roleArn);\n\n    return destroyStack({\n      sdk: stackSdk,\n      roleArn,\n      stack: options.stack,\n      deployName: options.deployName,\n      quiet: options.quiet,\n    });\n  }\n\n  public async stackExists(options: StackExistsOptions): Promise<boolean> {\n    const { stackSdk } = await this.prepareSdkFor(options.stack, undefined, Mode.ForReading);\n    const stack = await CloudFormationStack.lookup(stackSdk.cloudFormation(), options.deployName ?? options.stack.stackName);\n    return stack.exists;\n  }\n\n  private async prepareSdkWithLookupOrDeployRole(stackArtifact: cxapi.CloudFormationStackArtifact): Promise<PreparedSdkForEnvironment> {\n    // try to assume the lookup role\n    try {\n      const result = await prepareSdkWithLookupRoleFor(this.sdkProvider, stackArtifact);\n      if (result.didAssumeRole) {\n        return {\n          resolvedEnvironment: result.resolvedEnvironment,\n          stackSdk: result.sdk,\n        };\n      }\n    } catch { }\n    // fall back to the deploy role\n    return this.prepareSdkFor(stackArtifact, undefined, Mode.ForReading);\n  }\n\n  /**\n   * Get the environment necessary for touching the given stack\n   *\n   * Returns the following:\n   *\n   * - The resolved environment for the stack (no more 'unknown-account/unknown-region')\n   * - SDK loaded with the right credentials for calling `CreateChangeSet`.\n   * - The Execution Role that should be passed to CloudFormation.\n   */\n  private async prepareSdkFor(\n    stack: cxapi.CloudFormationStackArtifact,\n    roleArn?: string,\n    mode = Mode.ForWriting,\n  ): Promise<PreparedSdkForEnvironment> {\n    if (!stack.environment) {\n      throw new Error(`The stack ${stack.displayName} does not have an environment`);\n    }\n\n    const resolvedEnvironment = await this.sdkProvider.resolveEnvironment(stack.environment);\n\n    // Substitute any placeholders with information about the current environment\n    const arns = await replaceEnvPlaceholders({\n      assumeRoleArn: stack.assumeRoleArn,\n\n      // Use the override if given, otherwise use the field from the stack\n      cloudFormationRoleArn: roleArn ?? stack.cloudFormationExecutionRoleArn,\n    }, resolvedEnvironment, this.sdkProvider);\n\n    const stackSdk = await this.sdkProvider.forEnvironment(resolvedEnvironment, mode, {\n      assumeRoleArn: arns.assumeRoleArn,\n      assumeRoleExternalId: stack.assumeRoleExternalId,\n    });\n\n    return {\n      stackSdk: stackSdk.sdk,\n      resolvedEnvironment,\n      cloudFormationRoleArn: arns.cloudFormationRoleArn,\n    };\n  }\n\n  /**\n   * Publish all asset manifests that are referenced by the given stack\n   */\n  private async publishStackAssets(stack: cxapi.CloudFormationStackArtifact, toolkitInfo: ToolkitInfo) {\n    const stackEnv = await this.sdkProvider.resolveEnvironment(stack.environment);\n    const assetArtifacts = stack.dependencies.filter(cxapi.AssetManifestArtifact.isAssetManifestArtifact);\n\n    for (const assetArtifact of assetArtifacts) {\n      await this.validateBootstrapStackVersion(\n        stack.stackName,\n        assetArtifact.requiresBootstrapStackVersion,\n        assetArtifact.bootstrapStackVersionSsmParameter,\n        toolkitInfo);\n\n      const manifest = AssetManifest.fromFile(assetArtifact.file);\n      await publishAssets(manifest, this.sdkProvider, stackEnv);\n    }\n  }\n\n  /**\n   * Validate that the bootstrap stack has the right version for this stack\n   */\n  private async validateBootstrapStackVersion(\n    stackName: string,\n    requiresBootstrapStackVersion: number | undefined,\n    bootstrapStackVersionSsmParameter: string | undefined,\n    toolkitInfo: ToolkitInfo) {\n\n    if (requiresBootstrapStackVersion === undefined) { return; }\n\n    try {\n      await toolkitInfo.validateVersion(requiresBootstrapStackVersion, bootstrapStackVersionSsmParameter);\n    } catch (e) {\n      throw new Error(`${stackName}: ${e.message}`);\n    }\n  }\n}\n"]}
233
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloudformation-deployments.js","sourceRoot":"","sources":["cloudformation-deployments.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxapi from '@aws-cdk/cx-api';\nimport { AssetManifest } from 'cdk-assets';\nimport { Tag } from '../cdk-toolkit';\nimport { debug, warning } from '../logging';\nimport { publishAssets } from '../util/asset-publishing';\nimport { Mode } from './aws-auth/credentials';\nimport { ISDK } from './aws-auth/sdk';\nimport { SdkProvider } from './aws-auth/sdk-provider';\nimport { deployStack, DeployStackResult, destroyStack, makeBodyParameterAndUpload } from './deploy-stack';\nimport { loadCurrentTemplateWithNestedStacks, loadCurrentTemplate } from './nested-stack-helpers';\nimport { ToolkitInfo } from './toolkit-info';\nimport { CloudFormationStack, Template, ResourcesToImport, ResourceIdentifierSummaries } from './util/cloudformation';\nimport { StackActivityProgress } from './util/cloudformation/stack-activity-monitor';\nimport { replaceEnvPlaceholders } from './util/placeholders';\n\n/**\n * SDK obtained by assuming the lookup role\n * for a given environment\n */\nexport interface PreparedSdkWithLookupRoleForEnvironment {\n  /**\n   * The SDK for the given environment\n   */\n  readonly sdk: ISDK;\n\n  /**\n   * The resolved environment for the stack\n   * (no more 'unknown-account/unknown-region')\n   */\n  readonly resolvedEnvironment: cxapi.Environment;\n\n  /**\n   * Whether or not the assume role was successful.\n   * If the assume role was not successful (false)\n   * then that means that the 'sdk' returned contains\n   * the default credentials (not the assume role credentials)\n   */\n  readonly didAssumeRole: boolean;\n}\n\n/**\n  * Try to use the bootstrap lookupRole. There are two scenarios that are handled here\n  *  1. The lookup role may not exist (it was added in bootstrap stack version 7)\n  *  2. The lookup role may not have the correct permissions (ReadOnlyAccess was added in\n  *      bootstrap stack version 8)\n  *\n  * In the case of 1 (lookup role doesn't exist) `forEnvironment` will either:\n  *   1. Return the default credentials if the default credentials are for the stack account\n  *   2. Throw an error if the default credentials are not for the stack account.\n  *\n  * If we successfully assume the lookup role we then proceed to 2 and check whether the bootstrap\n  * stack version is valid. If it is not we throw an error which should be handled in the calling\n  * function (and fallback to use a different role, etc)\n  *\n  * If we do not successfully assume the lookup role, but do get back the default credentials\n  * then return those and note that we are returning the default credentials. The calling\n  * function can then decide to use them or fallback to another role.\n  */\nexport async function prepareSdkWithLookupRoleFor(\n  sdkProvider: SdkProvider,\n  stack: cxapi.CloudFormationStackArtifact,\n): Promise<PreparedSdkWithLookupRoleForEnvironment> {\n  const resolvedEnvironment = await sdkProvider.resolveEnvironment(stack.environment);\n\n  // Substitute any placeholders with information about the current environment\n  const arns = await replaceEnvPlaceholders({\n    lookupRoleArn: stack.lookupRole?.arn,\n  }, resolvedEnvironment, sdkProvider);\n\n  // try to assume the lookup role\n  const warningMessage = `Could not assume ${arns.lookupRoleArn}, proceeding anyway.`;\n  const upgradeMessage = `(To get rid of this warning, please upgrade to bootstrap version >= ${stack.lookupRole?.requiresBootstrapStackVersion})`;\n  try {\n    const stackSdk = await sdkProvider.forEnvironment(resolvedEnvironment, Mode.ForReading, {\n      assumeRoleArn: arns.lookupRoleArn,\n      assumeRoleExternalId: stack.lookupRole?.assumeRoleExternalId,\n    });\n\n    // if we succeed in assuming the lookup role, make sure we have the correct bootstrap stack version\n    if (stackSdk.didAssumeRole && stack.lookupRole?.bootstrapStackVersionSsmParameter && stack.lookupRole.requiresBootstrapStackVersion) {\n      const version = await ToolkitInfo.versionFromSsmParameter(stackSdk.sdk, stack.lookupRole.bootstrapStackVersionSsmParameter);\n      if (version < stack.lookupRole.requiresBootstrapStackVersion) {\n        throw new Error(`Bootstrap stack version '${stack.lookupRole.requiresBootstrapStackVersion}' is required, found version '${version}'.`);\n      }\n      // we may not have assumed the lookup role because one was not provided\n      // if that is the case then don't print the upgrade warning\n    } else if (!stackSdk.didAssumeRole && stack.lookupRole?.requiresBootstrapStackVersion) {\n      warning(upgradeMessage);\n    }\n    return { ...stackSdk, resolvedEnvironment };\n  } catch (e) {\n    debug(e);\n    // only print out the warnings if the lookupRole exists AND there is a required\n    // bootstrap version, otherwise the warnings will print `undefined`\n    if (stack.lookupRole && stack.lookupRole.requiresBootstrapStackVersion) {\n      warning(warningMessage);\n      warning(upgradeMessage);\n    }\n    throw (e);\n  }\n}\n\nexport interface DeployStackOptions {\n  /**\n   * Stack to deploy\n   */\n  stack: cxapi.CloudFormationStackArtifact;\n\n  /**\n   * Execution role for the deployment (pass through to CloudFormation)\n   *\n   * @default - Current role\n   */\n  roleArn?: string;\n\n  /**\n   * Topic ARNs to send a message when deployment finishes (pass through to CloudFormation)\n   *\n   * @default - No notifications\n   */\n  notificationArns?: string[];\n\n  /**\n   * Override name under which stack will be deployed\n   *\n   * @default - Use artifact default\n   */\n  deployName?: string;\n\n  /**\n   * Don't show stack deployment events, just wait\n   *\n   * @default false\n   */\n  quiet?: boolean;\n\n  /**\n   * Name of the toolkit stack, if not the default name\n   *\n   * @default 'CDKToolkit'\n   */\n  toolkitStackName?: string;\n\n  /**\n   * List of asset IDs which should NOT be built or uploaded\n   *\n   * @default - Build all assets\n   */\n  reuseAssets?: string[];\n\n  /**\n   * Stack tags (pass through to CloudFormation)\n   */\n  tags?: Tag[];\n\n  /**\n   * Stage the change set but don't execute it\n   *\n   * @default - false\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   * Force deployment, even if the deployed template is identical to the one we are about to deploy.\n   * @default false deployment will be skipped if the template is identical\n   */\n  force?: boolean;\n\n  /**\n   * Extra parameters for CloudFormation\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 true\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   * 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   * List of existing resources to be IMPORTED into the stack, instead of being CREATED\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\nexport interface DestroyStackOptions {\n  stack: cxapi.CloudFormationStackArtifact;\n  deployName?: string;\n  roleArn?: string;\n  quiet?: boolean;\n  force?: boolean;\n  ci?: boolean;\n}\n\nexport interface StackExistsOptions {\n  stack: cxapi.CloudFormationStackArtifact;\n  deployName?: string;\n}\n\nexport interface ProvisionerProps {\n  sdkProvider: SdkProvider;\n}\n\n/**\n * SDK obtained by assuming the deploy role\n * for a given environment\n */\nexport interface PreparedSdkForEnvironment {\n  /**\n   * The SDK for the given environment\n   */\n  readonly stackSdk: ISDK;\n\n  /**\n   * The resolved environment for the stack\n   * (no more 'unknown-account/unknown-region')\n   */\n  readonly resolvedEnvironment: cxapi.Environment;\n  /**\n   * The Execution Role that should be passed to CloudFormation.\n   *\n   * @default - no execution role is used\n   */\n  readonly cloudFormationRoleArn?: string;\n}\n\n/**\n * Helper class for CloudFormation deployments\n *\n * Looks us the right SDK and Bootstrap stack to deploy a given\n * stack artifact.\n */\nexport class CloudFormationDeployments {\n  private readonly sdkProvider: SdkProvider;\n\n  constructor(props: ProvisionerProps) {\n    this.sdkProvider = props.sdkProvider;\n  }\n\n  public async readCurrentTemplateWithNestedStacks(rootStackArtifact: cxapi.CloudFormationStackArtifact): Promise<Template> {\n    const sdk = (await this.prepareSdkWithLookupOrDeployRole(rootStackArtifact)).stackSdk;\n    return (await loadCurrentTemplateWithNestedStacks(rootStackArtifact, sdk)).deployedTemplate;\n  }\n\n  public async readCurrentTemplate(stackArtifact: cxapi.CloudFormationStackArtifact): Promise<Template> {\n    debug(`Reading existing template for stack ${stackArtifact.displayName}.`);\n    const sdk = (await this.prepareSdkWithLookupOrDeployRole(stackArtifact)).stackSdk;\n    return loadCurrentTemplate(stackArtifact, sdk);\n  }\n\n  public async resourceIdentifierSummaries(\n    stackArtifact: cxapi.CloudFormationStackArtifact,\n    toolkitStackName?: string,\n  ): Promise<ResourceIdentifierSummaries> {\n    debug(`Retrieving template summary for stack ${stackArtifact.displayName}.`);\n    // Currently, needs to use `deploy-role` since it may need to read templates in the staging\n    // bucket which have been encrypted with a KMS key (and lookup-role may not read encrypted things)\n    const { stackSdk, resolvedEnvironment } = await this.prepareSdkFor(stackArtifact, undefined, Mode.ForReading);\n    const cfn = stackSdk.cloudFormation();\n\n    const toolkitInfo = await ToolkitInfo.lookup(resolvedEnvironment, stackSdk, toolkitStackName);\n\n    // Upload the template, if necessary, before passing it to CFN\n    const cfnParam = await makeBodyParameterAndUpload(\n      stackArtifact,\n      resolvedEnvironment,\n      toolkitInfo,\n      this.sdkProvider,\n      stackSdk);\n\n    const response = await cfn.getTemplateSummary(cfnParam).promise();\n    if (!response.ResourceIdentifierSummaries) {\n      debug('GetTemplateSummary API call did not return \"ResourceIdentifierSummaries\"');\n    }\n    return response.ResourceIdentifierSummaries ?? [];\n  }\n\n  public async deployStack(options: DeployStackOptions): Promise<DeployStackResult> {\n    const { stackSdk, resolvedEnvironment, cloudFormationRoleArn } = await this.prepareSdkFor(options.stack, options.roleArn);\n\n    const toolkitInfo = await ToolkitInfo.lookup(resolvedEnvironment, stackSdk, options.toolkitStackName);\n\n    // Publish any assets before doing the actual deploy (do not publish any assets on import operation)\n    if (options.resourcesToImport === undefined) {\n      await this.publishStackAssets(options.stack, toolkitInfo);\n    }\n\n    // Do a verification of the bootstrap stack version\n    await this.validateBootstrapStackVersion(\n      options.stack.stackName,\n      options.stack.requiresBootstrapStackVersion,\n      options.stack.bootstrapStackVersionSsmParameter,\n      toolkitInfo);\n\n    return deployStack({\n      stack: options.stack,\n      resolvedEnvironment,\n      deployName: options.deployName,\n      notificationArns: options.notificationArns,\n      quiet: options.quiet,\n      sdk: stackSdk,\n      sdkProvider: this.sdkProvider,\n      roleArn: cloudFormationRoleArn,\n      reuseAssets: options.reuseAssets,\n      toolkitInfo,\n      tags: options.tags,\n      execute: options.execute,\n      changeSetName: options.changeSetName,\n      force: options.force,\n      parameters: options.parameters,\n      usePreviousParameters: options.usePreviousParameters,\n      progress: options.progress,\n      ci: options.ci,\n      rollback: options.rollback,\n      hotswap: options.hotswap,\n      extraUserAgent: options.extraUserAgent,\n      resourcesToImport: options.resourcesToImport,\n      overrideTemplate: options.overrideTemplate,\n    });\n  }\n\n  public async destroyStack(options: DestroyStackOptions): Promise<void> {\n    const { stackSdk, cloudFormationRoleArn: roleArn } = await this.prepareSdkFor(options.stack, options.roleArn);\n\n    return destroyStack({\n      sdk: stackSdk,\n      roleArn,\n      stack: options.stack,\n      deployName: options.deployName,\n      quiet: options.quiet,\n      ci: options.ci,\n    });\n  }\n\n  public async stackExists(options: StackExistsOptions): Promise<boolean> {\n    const { stackSdk } = await this.prepareSdkFor(options.stack, undefined, Mode.ForReading);\n    const stack = await CloudFormationStack.lookup(stackSdk.cloudFormation(), options.deployName ?? options.stack.stackName);\n    return stack.exists;\n  }\n\n  private async prepareSdkWithLookupOrDeployRole(stackArtifact: cxapi.CloudFormationStackArtifact): Promise<PreparedSdkForEnvironment> {\n    // try to assume the lookup role\n    try {\n      const result = await prepareSdkWithLookupRoleFor(this.sdkProvider, stackArtifact);\n      if (result.didAssumeRole) {\n        return {\n          resolvedEnvironment: result.resolvedEnvironment,\n          stackSdk: result.sdk,\n        };\n      }\n    } catch { }\n    // fall back to the deploy role\n    return this.prepareSdkFor(stackArtifact, undefined, Mode.ForReading);\n  }\n\n  /**\n   * Get the environment necessary for touching the given stack\n   *\n   * Returns the following:\n   *\n   * - The resolved environment for the stack (no more 'unknown-account/unknown-region')\n   * - SDK loaded with the right credentials for calling `CreateChangeSet`.\n   * - The Execution Role that should be passed to CloudFormation.\n   */\n  private async prepareSdkFor(\n    stack: cxapi.CloudFormationStackArtifact,\n    roleArn?: string,\n    mode = Mode.ForWriting,\n  ): Promise<PreparedSdkForEnvironment> {\n    if (!stack.environment) {\n      throw new Error(`The stack ${stack.displayName} does not have an environment`);\n    }\n\n    const resolvedEnvironment = await this.sdkProvider.resolveEnvironment(stack.environment);\n\n    // Substitute any placeholders with information about the current environment\n    const arns = await replaceEnvPlaceholders({\n      assumeRoleArn: stack.assumeRoleArn,\n\n      // Use the override if given, otherwise use the field from the stack\n      cloudFormationRoleArn: roleArn ?? stack.cloudFormationExecutionRoleArn,\n    }, resolvedEnvironment, this.sdkProvider);\n\n    const stackSdk = await this.sdkProvider.forEnvironment(resolvedEnvironment, mode, {\n      assumeRoleArn: arns.assumeRoleArn,\n      assumeRoleExternalId: stack.assumeRoleExternalId,\n    });\n\n    return {\n      stackSdk: stackSdk.sdk,\n      resolvedEnvironment,\n      cloudFormationRoleArn: arns.cloudFormationRoleArn,\n    };\n  }\n\n  /**\n   * Publish all asset manifests that are referenced by the given stack\n   */\n  private async publishStackAssets(stack: cxapi.CloudFormationStackArtifact, toolkitInfo: ToolkitInfo) {\n    const stackEnv = await this.sdkProvider.resolveEnvironment(stack.environment);\n    const assetArtifacts = stack.dependencies.filter(cxapi.AssetManifestArtifact.isAssetManifestArtifact);\n\n    for (const assetArtifact of assetArtifacts) {\n      await this.validateBootstrapStackVersion(\n        stack.stackName,\n        assetArtifact.requiresBootstrapStackVersion,\n        assetArtifact.bootstrapStackVersionSsmParameter,\n        toolkitInfo);\n\n      const manifest = AssetManifest.fromFile(assetArtifact.file);\n      await publishAssets(manifest, this.sdkProvider, stackEnv);\n    }\n  }\n\n  /**\n   * Validate that the bootstrap stack has the right version for this stack\n   */\n  private async validateBootstrapStackVersion(\n    stackName: string,\n    requiresBootstrapStackVersion: number | undefined,\n    bootstrapStackVersionSsmParameter: string | undefined,\n    toolkitInfo: ToolkitInfo) {\n\n    if (requiresBootstrapStackVersion === undefined) { return; }\n\n    try {\n      await toolkitInfo.validateVersion(requiresBootstrapStackVersion, bootstrapStackVersionSsmParameter);\n    } catch (e) {\n      throw new Error(`${stackName}: ${e.message}`);\n    }\n  }\n}\n"]}
@@ -178,6 +178,7 @@ export interface DestroyStackOptions {
178
178
  roleArn?: string;
179
179
  deployName?: string;
180
180
  quiet?: boolean;
181
+ ci?: boolean;
181
182
  }
182
183
  export declare function destroyStack(options: DestroyStackOptions): Promise<void>;
183
184
  export {};
@@ -155,6 +155,7 @@ async function prepareAndExecuteChangeSet(options, cloudFormationStack, stackArt
155
155
  resourcesTotal: cloudFormationStack.exists ? changeSetLength + 1 : changeSetLength,
156
156
  progress: options.progress,
157
157
  changeSetCreationTime: changeSetDescription.CreationTime,
158
+ ci: options.ci,
158
159
  }).start();
159
160
  logging_1.debug('Execution of changeset %s on stack %s has started; waiting for the update to complete...', changeSet.Id, deployName);
160
161
  try {
@@ -251,7 +252,9 @@ async function destroyStack(options) {
251
252
  if (!currentStack.exists) {
252
253
  return;
253
254
  }
254
- const monitor = options.quiet ? undefined : stack_activity_monitor_1.StackActivityMonitor.withDefaultPrinter(cfn, deployName, options.stack).start();
255
+ const monitor = options.quiet ? undefined : stack_activity_monitor_1.StackActivityMonitor.withDefaultPrinter(cfn, deployName, options.stack, {
256
+ ci: options.ci,
257
+ }).start();
255
258
  try {
256
259
  await cfn.deleteStack({ StackName: deployName, RoleARN: options.roleArn }).promise();
257
260
  const destroyedStack = await cloudformation_1.waitForStackDelete(cfn, deployName);
@@ -381,4 +384,4 @@ function suffixWithErrors(msg, errors) {
381
384
  ? `${msg}: ${errors.join(', ')}`
382
385
  : msg;
383
386
  }
384
- //# 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\n  const execute = options.execute ?? true;\n\n  debug('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 changeSetDescription = await waitForChangeSet(cfn, deployName, changeSetName, { fetchAll: execute });\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  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"]}
387
+ //# 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\n  const execute = options.execute ?? true;\n\n  debug('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 changeSetDescription = await waitForChangeSet(cfn, deployName, changeSetName, { fetchAll: execute });\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  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      ci: options.ci,\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  ci?: 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, {\n    ci: options.ci,\n  }).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"]}