aws-cdk 2.38.1 → 2.40.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 (86) hide show
  1. package/README.md +11 -0
  2. package/THIRD_PARTY_LICENSES +83 -3
  3. package/build-info.json +2 -2
  4. package/lib/api/aws-auth/aws-sdk-inifile.js +2 -3
  5. package/lib/api/aws-auth/awscli-compatible.js +5 -7
  6. package/lib/api/aws-auth/sdk-provider.js +18 -12
  7. package/lib/api/aws-auth/sdk.d.ts +20 -0
  8. package/lib/api/aws-auth/sdk.js +40 -5
  9. package/lib/api/bootstrap/bootstrap-environment.js +14 -15
  10. package/lib/api/bootstrap/deploy-bootstrap.js +5 -7
  11. package/lib/api/cloudformation-deployments.d.ts +28 -0
  12. package/lib/api/cloudformation-deployments.js +30 -15
  13. package/lib/api/cxapp/cloud-assembly.js +3 -4
  14. package/lib/api/cxapp/cloud-executable.js +2 -3
  15. package/lib/api/cxapp/exec.js +8 -10
  16. package/lib/api/deploy-stack.js +11 -13
  17. package/lib/api/evaluate-cloudformation-template.js +6 -11
  18. package/lib/api/hotswap/appsync-mapping-templates.js +4 -5
  19. package/lib/api/hotswap/code-build-projects.js +2 -3
  20. package/lib/api/hotswap/common.d.ts +6 -1
  21. package/lib/api/hotswap/common.js +14 -4
  22. package/lib/api/hotswap/ecs-services.js +23 -6
  23. package/lib/api/hotswap/lambda-functions.js +6 -9
  24. package/lib/api/hotswap/s3-bucket-deployments.js +3 -5
  25. package/lib/api/hotswap/stepfunctions-state-machines.js +2 -3
  26. package/lib/api/hotswap-deployments.js +5 -7
  27. package/lib/api/logs/logs-monitor.js +5 -8
  28. package/lib/api/nested-stack-helpers.js +9 -11
  29. package/lib/api/toolkit-info.js +12 -17
  30. package/lib/api/util/cloudformation/stack-activity-monitor.js +17 -27
  31. package/lib/api/util/cloudformation.js +7 -11
  32. package/lib/api/util/placeholders.js +2 -3
  33. package/lib/assets.js +3 -4
  34. package/lib/build.d.ts +6 -0
  35. package/lib/build.js +20 -0
  36. package/lib/cdk-toolkit.d.ts +15 -0
  37. package/lib/cdk-toolkit.js +62 -20
  38. package/lib/cli.js +15 -15
  39. package/lib/context-providers/load-balancers.js +11 -18
  40. package/lib/context-providers/security-groups.js +5 -7
  41. package/lib/deploy.d.ts +7 -0
  42. package/lib/deploy.js +51 -0
  43. package/lib/import.js +8 -13
  44. package/lib/index.js +9378 -2580
  45. package/lib/init-templates/app/go/%name%.template.go +2 -0
  46. package/lib/init-templates/app/go/%name%_test.template.go +1 -1
  47. package/lib/init-templates/app/go/go.template.mod +1 -1
  48. package/lib/init-templates/sample-app/go/%name%.template.go +2 -0
  49. package/lib/init-templates/sample-app/go/%name%_test.template.go +1 -1
  50. package/lib/init-templates/sample-app/go/go.template.mod +1 -1
  51. package/lib/notices.js +11 -17
  52. package/lib/settings.js +4 -5
  53. package/lib/setup-sandbox.js +2 -1
  54. package/lib/util/asset-publishing.d.ts +16 -0
  55. package/lib/util/asset-publishing.js +34 -9
  56. package/lib/util/directories.js +2 -3
  57. package/lib/version.js +2 -2
  58. package/package.json +13 -12
  59. package/test/api/bootstrap2.test.js +2 -3
  60. package/test/api/cloud-executable.test.js +2 -3
  61. package/test/api/cloudformation-deployments.test.js +123 -1
  62. package/test/api/deploy-stack.test.js +4 -7
  63. package/test/api/fake-sts.js +13 -9
  64. package/test/api/hotswap/ecs-services-hotswap-deployments.test.js +113 -1
  65. package/test/api/hotswap/hotswap-deployments.test.js +4 -4
  66. package/test/api/sdk-provider.test.js +22 -11
  67. package/test/aws-sdk-non-public-apis.test.js +2 -3
  68. package/test/build.test.d.ts +1 -0
  69. package/test/build.test.js +31 -0
  70. package/test/cdk-toolkit.test.js +27 -1
  71. package/test/context-providers/load-balancers.test.js +3 -5
  72. package/test/deploy.test.d.ts +1 -0
  73. package/test/deploy.test.js +186 -0
  74. package/test/import.test.js +2 -2
  75. package/test/init.test.js +2 -3
  76. package/test/integ/cli/app/app.js +8 -1
  77. package/test/integ/cli/bootstrapping.integtest.js +9 -20
  78. package/test/integ/cli/cli.integtest.js +57 -34
  79. package/test/integ/github-helpers.js +2 -3
  80. package/test/integ/helpers/aws.js +4 -7
  81. package/test/integ/helpers/cdk.js +21 -30
  82. package/test/integ/helpers/sam.js +10 -17
  83. package/test/util/mock-child_process.js +3 -4
  84. package/test/util/mock-sdk.js +4 -6
  85. package/test/util/mock-toolkitinfo.js +4 -5
  86. package/test/util.js +3 -5
@@ -31,12 +31,11 @@ class Bootstrapper {
31
31
  *
32
32
  */
33
33
  async legacyBootstrap(environment, sdkProvider, options = {}) {
34
- var _a, _b, _c, _d;
35
- const params = (_a = options.parameters) !== null && _a !== void 0 ? _a : {};
36
- if ((_b = params.trustedAccounts) === null || _b === void 0 ? void 0 : _b.length) {
34
+ const params = options.parameters ?? {};
35
+ if (params.trustedAccounts?.length) {
37
36
  throw new Error('--trust can only be passed for the modern bootstrap experience.');
38
37
  }
39
- if ((_c = params.cloudFormationExecutionPolicies) === null || _c === void 0 ? void 0 : _c.length) {
38
+ if (params.cloudFormationExecutionPolicies?.length) {
40
39
  throw new Error('--cloudformation-execution-policies can only be passed for the modern bootstrap experience.');
41
40
  }
42
41
  if (params.createCustomerMasterKey !== undefined) {
@@ -48,7 +47,7 @@ class Bootstrapper {
48
47
  const current = await deploy_bootstrap_1.BootstrapStack.lookup(sdkProvider, environment, options.toolkitStackName);
49
48
  return current.update(await this.loadTemplate(params), {}, {
50
49
  ...options,
51
- terminationProtection: (_d = options.terminationProtection) !== null && _d !== void 0 ? _d : current.terminationProtection,
50
+ terminationProtection: options.terminationProtection ?? current.terminationProtection,
52
51
  });
53
52
  }
54
53
  /**
@@ -56,8 +55,7 @@ class Bootstrapper {
56
55
  *
57
56
  */
58
57
  async modernBootstrap(environment, sdkProvider, options = {}) {
59
- var _a, _b, _c, _d, _e, _f;
60
- const params = (_a = options.parameters) !== null && _a !== void 0 ? _a : {};
58
+ const params = options.parameters ?? {};
61
59
  const bootstrapTemplate = await this.loadTemplate();
62
60
  const current = await deploy_bootstrap_1.BootstrapStack.lookup(sdkProvider, environment, options.toolkitStackName);
63
61
  if (params.createCustomerMasterKey !== undefined && params.kmsKeyId) {
@@ -71,11 +69,11 @@ class Bootstrapper {
71
69
  // Ideally we'd do this inside the template, but the `Rules` section of CFN
72
70
  // templates doesn't seem to be able to express the conditions that we need
73
71
  // (can't use Fn::Join or reference Conditions) so we do it here instead.
74
- const trustedAccounts = (_b = params.trustedAccounts) !== null && _b !== void 0 ? _b : splitCfnArray(current.parameters.TrustedAccounts);
72
+ const trustedAccounts = params.trustedAccounts ?? splitCfnArray(current.parameters.TrustedAccounts);
75
73
  console_1.info(`Trusted accounts for deployment: ${trustedAccounts.length > 0 ? trustedAccounts.join(', ') : '(none)'}`);
76
- const trustedAccountsForLookup = (_c = params.trustedAccountsForLookup) !== null && _c !== void 0 ? _c : splitCfnArray(current.parameters.TrustedAccountsForLookup);
74
+ const trustedAccountsForLookup = params.trustedAccountsForLookup ?? splitCfnArray(current.parameters.TrustedAccountsForLookup);
77
75
  console_1.info(`Trusted accounts for lookup: ${trustedAccountsForLookup.length > 0 ? trustedAccountsForLookup.join(', ') : '(none)'}`);
78
- const cloudFormationExecutionPolicies = (_d = params.cloudFormationExecutionPolicies) !== null && _d !== void 0 ? _d : splitCfnArray(current.parameters.CloudFormationExecutionPolicies);
76
+ const cloudFormationExecutionPolicies = params.cloudFormationExecutionPolicies ?? splitCfnArray(current.parameters.CloudFormationExecutionPolicies);
79
77
  if (trustedAccounts.length === 0 && cloudFormationExecutionPolicies.length === 0) {
80
78
  // For self-trust it's okay to default to AdministratorAccess, and it improves the usability of bootstrapping a lot.
81
79
  //
@@ -107,9 +105,10 @@ class Bootstrapper {
107
105
  // * undefined if we already had a value in place (reusing what we had)
108
106
  // * '-' if this is the first time we're deploying this stack (or upgrading from old to new bootstrap)
109
107
  const currentKmsKeyId = current.parameters.FileAssetsBucketKmsKeyId;
110
- const kmsKeyId = (_e = params.kmsKeyId) !== null && _e !== void 0 ? _e : (params.createCustomerMasterKey === true ? CREATE_NEW_KEY :
111
- params.createCustomerMasterKey === false || currentKmsKeyId === undefined ? USE_AWS_MANAGED_KEY :
112
- undefined);
108
+ const kmsKeyId = params.kmsKeyId ??
109
+ (params.createCustomerMasterKey === true ? CREATE_NEW_KEY :
110
+ params.createCustomerMasterKey === false || currentKmsKeyId === undefined ? USE_AWS_MANAGED_KEY :
111
+ undefined);
113
112
  return current.update(bootstrapTemplate, {
114
113
  FileAssetsBucketName: params.bucketName,
115
114
  FileAssetsBucketKmsKeyId: kmsKeyId,
@@ -121,7 +120,7 @@ class Bootstrapper {
121
120
  PublicAccessBlockConfiguration: params.publicAccessBlockConfiguration || params.publicAccessBlockConfiguration === undefined ? 'true' : 'false',
122
121
  }, {
123
122
  ...options,
124
- terminationProtection: (_f = options.terminationProtection) !== null && _f !== void 0 ? _f : current.terminationProtection,
123
+ terminationProtection: options.terminationProtection ?? current.terminationProtection,
125
124
  });
126
125
  }
127
126
  async customBootstrap(environment, sdkProvider, options = {}) {
@@ -166,4 +165,4 @@ function splitCfnArray(xs) {
166
165
  }
167
166
  return xs.split(',');
168
167
  }
169
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bootstrap-environment.js","sourceRoot":"","sources":["bootstrap-environment.ts"],"names":[],"mappings":"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","sourcesContent":["import { info } from 'console';\nimport * as path from 'path';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport { warning } from '../../logging';\nimport { loadStructuredFile, toYAML } from '../../serialize';\nimport { rootDir } from '../../util/directories';\nimport { SdkProvider } from '../aws-auth';\nimport { DeployStackResult } from '../deploy-stack';\nimport { BootstrapEnvironmentOptions, BootstrappingParameters } from './bootstrap-props';\nimport { BootstrapStack, bootstrapVersionFromTemplate } from './deploy-bootstrap';\nimport { legacyBootstrapTemplate } from './legacy-template';\n\n/* eslint-disable max-len */\n\nexport type BootstrapSource =\n  { source: 'legacy' }\n  | { source: 'default' }\n  | { source: 'custom'; templateFile: string };\n\n\nexport class Bootstrapper {\n  constructor(private readonly source: BootstrapSource) {\n  }\n\n  public bootstrapEnvironment(environment: cxapi.Environment, sdkProvider: SdkProvider, options: BootstrapEnvironmentOptions = {}): Promise<DeployStackResult> {\n    switch (this.source.source) {\n      case 'legacy':\n        return this.legacyBootstrap(environment, sdkProvider, options);\n      case 'default':\n        return this.modernBootstrap(environment, sdkProvider, options);\n      case 'custom':\n        return this.customBootstrap(environment, sdkProvider, options);\n    }\n  }\n\n  public async showTemplate() {\n    const template = await this.loadTemplate();\n    process.stdout.write(`${toYAML(template)}\\n`);\n  }\n\n  /**\n   * Deploy legacy bootstrap stack\n   *\n   */\n  private async legacyBootstrap(environment: cxapi.Environment, sdkProvider: SdkProvider, options: BootstrapEnvironmentOptions = {}): Promise<DeployStackResult> {\n    const params = options.parameters ?? {};\n\n    if (params.trustedAccounts?.length) {\n      throw new Error('--trust can only be passed for the modern bootstrap experience.');\n    }\n    if (params.cloudFormationExecutionPolicies?.length) {\n      throw new Error('--cloudformation-execution-policies can only be passed for the modern bootstrap experience.');\n    }\n    if (params.createCustomerMasterKey !== undefined) {\n      throw new Error('--bootstrap-customer-key can only be passed for the modern bootstrap experience.');\n    }\n    if (params.qualifier) {\n      throw new Error('--qualifier can only be passed for the modern bootstrap experience.');\n    }\n\n    const current = await BootstrapStack.lookup(sdkProvider, environment, options.toolkitStackName);\n    return current.update(await this.loadTemplate(params), {}, {\n      ...options,\n      terminationProtection: options.terminationProtection ?? current.terminationProtection,\n    });\n  }\n\n  /**\n   * Deploy CI/CD-ready bootstrap stack from template\n   *\n   */\n  private async modernBootstrap(\n    environment: cxapi.Environment,\n    sdkProvider: SdkProvider,\n    options: BootstrapEnvironmentOptions = {}): Promise<DeployStackResult> {\n\n    const params = options.parameters ?? {};\n\n    const bootstrapTemplate = await this.loadTemplate();\n\n    const current = await BootstrapStack.lookup(sdkProvider, environment, options.toolkitStackName);\n\n    if (params.createCustomerMasterKey !== undefined && params.kmsKeyId) {\n      throw new Error('You cannot pass \\'--bootstrap-kms-key-id\\' and \\'--bootstrap-customer-key\\' together. Specify one or the other');\n    }\n\n    // If people re-bootstrap, existing parameter values are reused so that people don't accidentally change the configuration\n    // on their bootstrap stack (this happens automatically in deployStack). However, to do proper validation on the\n    // combined arguments (such that if --trust has been given, --cloudformation-execution-policies is necessary as well)\n    // we need to take this parameter reuse into account.\n    //\n    // Ideally we'd do this inside the template, but the `Rules` section of CFN\n    // templates doesn't seem to be able to express the conditions that we need\n    // (can't use Fn::Join or reference Conditions) so we do it here instead.\n    const trustedAccounts = params.trustedAccounts ?? splitCfnArray(current.parameters.TrustedAccounts);\n    info(`Trusted accounts for deployment: ${trustedAccounts.length > 0 ? trustedAccounts.join(', ') : '(none)'}`);\n\n    const trustedAccountsForLookup = params.trustedAccountsForLookup ?? splitCfnArray(current.parameters.TrustedAccountsForLookup);\n    info(`Trusted accounts for lookup: ${trustedAccountsForLookup.length > 0 ? trustedAccountsForLookup.join(', ') : '(none)'}`);\n\n    const cloudFormationExecutionPolicies = params.cloudFormationExecutionPolicies ?? splitCfnArray(current.parameters.CloudFormationExecutionPolicies);\n    if (trustedAccounts.length === 0 && cloudFormationExecutionPolicies.length === 0) {\n      // For self-trust it's okay to default to AdministratorAccess, and it improves the usability of bootstrapping a lot.\n      //\n      // We don't actually make the implicity policy a physical parameter. The template will infer it instead,\n      // we simply do the UI advertising that behavior here.\n      //\n      // If we DID make it an explicit parameter, we wouldn't be able to tell the difference between whether\n      // we inferred it or whether the user told us, and the sequence:\n      //\n      // $ cdk bootstrap\n      // $ cdk bootstrap --trust 1234\n      //\n      // Would leave AdministratorAccess policies with a trust relationship, without the user explicitly\n      // approving the trust policy.\n      const implicitPolicy = `arn:${await current.partition()}:iam::aws:policy/AdministratorAccess`;\n      warning(`Using default execution policy of '${implicitPolicy}'. Pass '--cloudformation-execution-policies' to customize.`);\n    } else if (cloudFormationExecutionPolicies.length === 0) {\n      throw new Error('Please pass \\'--cloudformation-execution-policies\\' when using \\'--trust\\' to specify deployment permissions. Try a managed policy of the form \\'arn:aws:iam::aws:policy/<PolicyName>\\'.');\n    } else {\n      // Remind people what the current settings are\n      info(`Execution policies: ${cloudFormationExecutionPolicies.join(', ')}`);\n    }\n\n    // * If an ARN is given, that ARN. Otherwise:\n    //   * '-' if customerKey = false\n    //   * '' if customerKey = true\n    //   * if customerKey is also not given\n    //     * undefined if we already had a value in place (reusing what we had)\n    //     * '-' if this is the first time we're deploying this stack (or upgrading from old to new bootstrap)\n    const currentKmsKeyId = current.parameters.FileAssetsBucketKmsKeyId;\n    const kmsKeyId = params.kmsKeyId ??\n      (params.createCustomerMasterKey === true ? CREATE_NEW_KEY :\n        params.createCustomerMasterKey === false || currentKmsKeyId === undefined ? USE_AWS_MANAGED_KEY :\n          undefined);\n\n    return current.update(\n      bootstrapTemplate,\n      {\n        FileAssetsBucketName: params.bucketName,\n        FileAssetsBucketKmsKeyId: kmsKeyId,\n        // Empty array becomes empty string\n        TrustedAccounts: trustedAccounts.join(','),\n        TrustedAccountsForLookup: trustedAccountsForLookup.join(','),\n        CloudFormationExecutionPolicies: cloudFormationExecutionPolicies.join(','),\n        Qualifier: params.qualifier,\n        PublicAccessBlockConfiguration: params.publicAccessBlockConfiguration || params.publicAccessBlockConfiguration === undefined ? 'true' : 'false',\n      }, {\n        ...options,\n        terminationProtection: options.terminationProtection ?? current.terminationProtection,\n      });\n  }\n\n  private async customBootstrap(\n    environment: cxapi.Environment,\n    sdkProvider: SdkProvider,\n    options: BootstrapEnvironmentOptions = {}): Promise<DeployStackResult> {\n\n    // Look at the template, decide whether it's most likely a legacy or modern bootstrap\n    // template, and use the right bootstrapper for that.\n    const version = bootstrapVersionFromTemplate(await this.loadTemplate());\n    if (version === 0) {\n      return this.legacyBootstrap(environment, sdkProvider, options);\n    } else {\n      return this.modernBootstrap(environment, sdkProvider, options);\n    }\n  }\n\n  private async loadTemplate(params: BootstrappingParameters = {}): Promise<any> {\n    switch (this.source.source) {\n      case 'custom':\n        return loadStructuredFile(this.source.templateFile);\n      case 'default':\n        return loadStructuredFile(path.join(rootDir(), 'lib', 'api', 'bootstrap', 'bootstrap-template.yaml'));\n      case 'legacy':\n        return legacyBootstrapTemplate(params);\n    }\n  }\n}\n\n/**\n * Magic parameter value that will cause the bootstrap-template.yml to NOT create a CMK but use the default keyo\n */\nconst USE_AWS_MANAGED_KEY = 'AWS_MANAGED_KEY';\n\n/**\n * Magic parameter value that will cause the bootstrap-template.yml to create a CMK\n */\nconst CREATE_NEW_KEY = '';\n\n/**\n * Split an array-like CloudFormation parameter on ,\n *\n * An empty string is the empty array (instead of `['']`).\n */\nfunction splitCfnArray(xs: string | undefined): string[] {\n  if (xs === '' || xs === undefined) { return []; }\n  return xs.split(',');\n}\n"]}
168
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bootstrap-environment.js","sourceRoot":"","sources":["bootstrap-environment.ts"],"names":[],"mappings":"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","sourcesContent":["import { info } from 'console';\nimport * as path from 'path';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport { warning } from '../../logging';\nimport { loadStructuredFile, toYAML } from '../../serialize';\nimport { rootDir } from '../../util/directories';\nimport { SdkProvider } from '../aws-auth';\nimport { DeployStackResult } from '../deploy-stack';\nimport { BootstrapEnvironmentOptions, BootstrappingParameters } from './bootstrap-props';\nimport { BootstrapStack, bootstrapVersionFromTemplate } from './deploy-bootstrap';\nimport { legacyBootstrapTemplate } from './legacy-template';\n\n/* eslint-disable max-len */\n\nexport type BootstrapSource =\n  { source: 'legacy' }\n  | { source: 'default' }\n  | { source: 'custom'; templateFile: string };\n\n\nexport class Bootstrapper {\n  constructor(private readonly source: BootstrapSource) {\n  }\n\n  public bootstrapEnvironment(environment: cxapi.Environment, sdkProvider: SdkProvider, options: BootstrapEnvironmentOptions = {}): Promise<DeployStackResult> {\n    switch (this.source.source) {\n      case 'legacy':\n        return this.legacyBootstrap(environment, sdkProvider, options);\n      case 'default':\n        return this.modernBootstrap(environment, sdkProvider, options);\n      case 'custom':\n        return this.customBootstrap(environment, sdkProvider, options);\n    }\n  }\n\n  public async showTemplate() {\n    const template = await this.loadTemplate();\n    process.stdout.write(`${toYAML(template)}\\n`);\n  }\n\n  /**\n   * Deploy legacy bootstrap stack\n   *\n   */\n  private async legacyBootstrap(environment: cxapi.Environment, sdkProvider: SdkProvider, options: BootstrapEnvironmentOptions = {}): Promise<DeployStackResult> {\n    const params = options.parameters ?? {};\n\n    if (params.trustedAccounts?.length) {\n      throw new Error('--trust can only be passed for the modern bootstrap experience.');\n    }\n    if (params.cloudFormationExecutionPolicies?.length) {\n      throw new Error('--cloudformation-execution-policies can only be passed for the modern bootstrap experience.');\n    }\n    if (params.createCustomerMasterKey !== undefined) {\n      throw new Error('--bootstrap-customer-key can only be passed for the modern bootstrap experience.');\n    }\n    if (params.qualifier) {\n      throw new Error('--qualifier can only be passed for the modern bootstrap experience.');\n    }\n\n    const current = await BootstrapStack.lookup(sdkProvider, environment, options.toolkitStackName);\n    return current.update(await this.loadTemplate(params), {}, {\n      ...options,\n      terminationProtection: options.terminationProtection ?? current.terminationProtection,\n    });\n  }\n\n  /**\n   * Deploy CI/CD-ready bootstrap stack from template\n   *\n   */\n  private async modernBootstrap(\n    environment: cxapi.Environment,\n    sdkProvider: SdkProvider,\n    options: BootstrapEnvironmentOptions = {}): Promise<DeployStackResult> {\n\n    const params = options.parameters ?? {};\n\n    const bootstrapTemplate = await this.loadTemplate();\n\n    const current = await BootstrapStack.lookup(sdkProvider, environment, options.toolkitStackName);\n\n    if (params.createCustomerMasterKey !== undefined && params.kmsKeyId) {\n      throw new Error('You cannot pass \\'--bootstrap-kms-key-id\\' and \\'--bootstrap-customer-key\\' together. Specify one or the other');\n    }\n\n    // If people re-bootstrap, existing parameter values are reused so that people don't accidentally change the configuration\n    // on their bootstrap stack (this happens automatically in deployStack). However, to do proper validation on the\n    // combined arguments (such that if --trust has been given, --cloudformation-execution-policies is necessary as well)\n    // we need to take this parameter reuse into account.\n    //\n    // Ideally we'd do this inside the template, but the `Rules` section of CFN\n    // templates doesn't seem to be able to express the conditions that we need\n    // (can't use Fn::Join or reference Conditions) so we do it here instead.\n    const trustedAccounts = params.trustedAccounts ?? splitCfnArray(current.parameters.TrustedAccounts);\n    info(`Trusted accounts for deployment: ${trustedAccounts.length > 0 ? trustedAccounts.join(', ') : '(none)'}`);\n\n    const trustedAccountsForLookup = params.trustedAccountsForLookup ?? splitCfnArray(current.parameters.TrustedAccountsForLookup);\n    info(`Trusted accounts for lookup: ${trustedAccountsForLookup.length > 0 ? trustedAccountsForLookup.join(', ') : '(none)'}`);\n\n    const cloudFormationExecutionPolicies = params.cloudFormationExecutionPolicies ?? splitCfnArray(current.parameters.CloudFormationExecutionPolicies);\n    if (trustedAccounts.length === 0 && cloudFormationExecutionPolicies.length === 0) {\n      // For self-trust it's okay to default to AdministratorAccess, and it improves the usability of bootstrapping a lot.\n      //\n      // We don't actually make the implicity policy a physical parameter. The template will infer it instead,\n      // we simply do the UI advertising that behavior here.\n      //\n      // If we DID make it an explicit parameter, we wouldn't be able to tell the difference between whether\n      // we inferred it or whether the user told us, and the sequence:\n      //\n      // $ cdk bootstrap\n      // $ cdk bootstrap --trust 1234\n      //\n      // Would leave AdministratorAccess policies with a trust relationship, without the user explicitly\n      // approving the trust policy.\n      const implicitPolicy = `arn:${await current.partition()}:iam::aws:policy/AdministratorAccess`;\n      warning(`Using default execution policy of '${implicitPolicy}'. Pass '--cloudformation-execution-policies' to customize.`);\n    } else if (cloudFormationExecutionPolicies.length === 0) {\n      throw new Error('Please pass \\'--cloudformation-execution-policies\\' when using \\'--trust\\' to specify deployment permissions. Try a managed policy of the form \\'arn:aws:iam::aws:policy/<PolicyName>\\'.');\n    } else {\n      // Remind people what the current settings are\n      info(`Execution policies: ${cloudFormationExecutionPolicies.join(', ')}`);\n    }\n\n    // * If an ARN is given, that ARN. Otherwise:\n    //   * '-' if customerKey = false\n    //   * '' if customerKey = true\n    //   * if customerKey is also not given\n    //     * undefined if we already had a value in place (reusing what we had)\n    //     * '-' if this is the first time we're deploying this stack (or upgrading from old to new bootstrap)\n    const currentKmsKeyId = current.parameters.FileAssetsBucketKmsKeyId;\n    const kmsKeyId = params.kmsKeyId ??\n      (params.createCustomerMasterKey === true ? CREATE_NEW_KEY :\n        params.createCustomerMasterKey === false || currentKmsKeyId === undefined ? USE_AWS_MANAGED_KEY :\n          undefined);\n\n    return current.update(\n      bootstrapTemplate,\n      {\n        FileAssetsBucketName: params.bucketName,\n        FileAssetsBucketKmsKeyId: kmsKeyId,\n        // Empty array becomes empty string\n        TrustedAccounts: trustedAccounts.join(','),\n        TrustedAccountsForLookup: trustedAccountsForLookup.join(','),\n        CloudFormationExecutionPolicies: cloudFormationExecutionPolicies.join(','),\n        Qualifier: params.qualifier,\n        PublicAccessBlockConfiguration: params.publicAccessBlockConfiguration || params.publicAccessBlockConfiguration === undefined ? 'true' : 'false',\n      }, {\n        ...options,\n        terminationProtection: options.terminationProtection ?? current.terminationProtection,\n      });\n  }\n\n  private async customBootstrap(\n    environment: cxapi.Environment,\n    sdkProvider: SdkProvider,\n    options: BootstrapEnvironmentOptions = {}): Promise<DeployStackResult> {\n\n    // Look at the template, decide whether it's most likely a legacy or modern bootstrap\n    // template, and use the right bootstrapper for that.\n    const version = bootstrapVersionFromTemplate(await this.loadTemplate());\n    if (version === 0) {\n      return this.legacyBootstrap(environment, sdkProvider, options);\n    } else {\n      return this.modernBootstrap(environment, sdkProvider, options);\n    }\n  }\n\n  private async loadTemplate(params: BootstrappingParameters = {}): Promise<any> {\n    switch (this.source.source) {\n      case 'custom':\n        return loadStructuredFile(this.source.templateFile);\n      case 'default':\n        return loadStructuredFile(path.join(rootDir(), 'lib', 'api', 'bootstrap', 'bootstrap-template.yaml'));\n      case 'legacy':\n        return legacyBootstrapTemplate(params);\n    }\n  }\n}\n\n/**\n * Magic parameter value that will cause the bootstrap-template.yml to NOT create a CMK but use the default keyo\n */\nconst USE_AWS_MANAGED_KEY = 'AWS_MANAGED_KEY';\n\n/**\n * Magic parameter value that will cause the bootstrap-template.yml to create a CMK\n */\nconst CREATE_NEW_KEY = '';\n\n/**\n * Split an array-like CloudFormation parameter on ,\n *\n * An empty string is the empty array (instead of `['']`).\n */\nfunction splitCfnArray(xs: string | undefined): string[] {\n  if (xs === '' || xs === undefined) { return []; }\n  return xs.split(',');\n}\n"]}
@@ -34,7 +34,7 @@ class BootstrapStack {
34
34
  this.currentToolkitInfo = currentToolkitInfo;
35
35
  }
36
36
  static async lookup(sdkProvider, environment, toolkitStackName) {
37
- toolkitStackName = toolkitStackName !== null && toolkitStackName !== void 0 ? toolkitStackName : toolkit_info_1.DEFAULT_TOOLKIT_STACK_NAME;
37
+ toolkitStackName = toolkitStackName ?? toolkit_info_1.DEFAULT_TOOLKIT_STACK_NAME;
38
38
  const resolvedEnvironment = await sdkProvider.resolveEnvironment(environment);
39
39
  const sdk = (await sdkProvider.forEnvironment(resolvedEnvironment, aws_auth_1.Mode.ForWriting)).sdk;
40
40
  const currentToolkitInfo = await toolkit_info_1.ToolkitInfo.lookup(resolvedEnvironment, sdk, toolkitStackName);
@@ -53,7 +53,6 @@ class BootstrapStack {
53
53
  * Perform the actual deployment of a bootstrap stack, given a template and some parameters
54
54
  */
55
55
  async update(template, parameters, options) {
56
- var _a;
57
56
  const newVersion = bootstrapVersionFromTemplate(template);
58
57
  if (this.currentToolkitInfo.found && newVersion < this.currentToolkitInfo.version && !options.force) {
59
58
  logging.warning(`Bootstrap stack already at version '${this.currentToolkitInfo.version}'. Not downgrading it to version '${newVersion}' (use --force if you intend to downgrade)`);
@@ -77,7 +76,7 @@ class BootstrapStack {
77
76
  environment: cxapi.EnvironmentUtils.format(this.resolvedEnvironment.account, this.resolvedEnvironment.region),
78
77
  properties: {
79
78
  templateFile,
80
- terminationProtection: (_a = options.terminationProtection) !== null && _a !== void 0 ? _a : false,
79
+ terminationProtection: options.terminationProtection ?? false,
81
80
  },
82
81
  });
83
82
  const assembly = builder.buildAssembly();
@@ -99,10 +98,9 @@ class BootstrapStack {
99
98
  }
100
99
  exports.BootstrapStack = BootstrapStack;
101
100
  function bootstrapVersionFromTemplate(template) {
102
- var _a, _b, _c, _d, _e;
103
101
  const versionSources = [
104
- (_b = (_a = template.Outputs) === null || _a === void 0 ? void 0 : _a[bootstrap_props_1.BOOTSTRAP_VERSION_OUTPUT]) === null || _b === void 0 ? void 0 : _b.Value,
105
- (_e = (_d = (_c = template.Resources) === null || _c === void 0 ? void 0 : _c[bootstrap_props_1.BOOTSTRAP_VERSION_RESOURCE]) === null || _d === void 0 ? void 0 : _d.Properties) === null || _e === void 0 ? void 0 : _e.Value,
102
+ template.Outputs?.[bootstrap_props_1.BOOTSTRAP_VERSION_OUTPUT]?.Value,
103
+ template.Resources?.[bootstrap_props_1.BOOTSTRAP_VERSION_RESOURCE]?.Properties?.Value,
106
104
  ];
107
105
  for (const vs of versionSources) {
108
106
  if (typeof vs === 'number') {
@@ -115,4 +113,4 @@ function bootstrapVersionFromTemplate(template) {
115
113
  return 0;
116
114
  }
117
115
  exports.bootstrapVersionFromTemplate = bootstrapVersionFromTemplate;
118
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"deploy-bootstrap.js","sourceRoot":"","sources":["deploy-bootstrap.ts"],"names":[],"mappings":"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","sourcesContent":["import * as os from 'os';\nimport * as path from 'path';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as fs from 'fs-extra';\nimport * as logging from '../../logging';\nimport { Mode, SdkProvider, ISDK } from '../aws-auth';\nimport { deployStack, DeployStackResult } from '../deploy-stack';\nimport { DEFAULT_TOOLKIT_STACK_NAME, ToolkitInfo } from '../toolkit-info';\nimport { BOOTSTRAP_VERSION_OUTPUT, BootstrapEnvironmentOptions, BOOTSTRAP_VERSION_RESOURCE } from './bootstrap-props';\n\n/**\n * A class to hold state around stack bootstrapping\n *\n * This class exists so we can break bootstrapping into 2 phases:\n *\n * ```ts\n * const current = BootstrapStack.lookup(...);\n * // ...\n * current.update(newTemplate, ...);\n * ```\n *\n * And do something in between the two phases (such as look at the\n * current bootstrap stack and doing something intelligent).\n */\nexport class BootstrapStack {\n  public static async lookup(sdkProvider: SdkProvider, environment: cxapi.Environment, toolkitStackName?: string) {\n    toolkitStackName = toolkitStackName ?? DEFAULT_TOOLKIT_STACK_NAME;\n\n    const resolvedEnvironment = await sdkProvider.resolveEnvironment(environment);\n    const sdk = (await sdkProvider.forEnvironment(resolvedEnvironment, Mode.ForWriting)).sdk;\n\n    const currentToolkitInfo = await ToolkitInfo.lookup(resolvedEnvironment, sdk, toolkitStackName);\n\n    return new BootstrapStack(sdkProvider, sdk, resolvedEnvironment, toolkitStackName, currentToolkitInfo);\n  }\n\n  protected constructor(\n    private readonly sdkProvider: SdkProvider,\n    private readonly sdk: ISDK,\n    private readonly resolvedEnvironment: cxapi.Environment,\n    private readonly toolkitStackName: string,\n    private readonly currentToolkitInfo: ToolkitInfo) {\n  }\n\n  public get parameters(): Record<string, string> {\n    return this.currentToolkitInfo.found ? this.currentToolkitInfo.bootstrapStack.parameters : {};\n  }\n\n  public get terminationProtection() {\n    return this.currentToolkitInfo.found ? this.currentToolkitInfo.bootstrapStack.terminationProtection : undefined;\n  }\n\n  public async partition(): Promise<string> {\n    return (await this.sdk.currentAccount()).partition;\n  }\n\n  /**\n   * Perform the actual deployment of a bootstrap stack, given a template and some parameters\n   */\n  public async update(\n    template: any,\n    parameters: Record<string, string | undefined>,\n    options: Omit<BootstrapEnvironmentOptions, 'parameters'>,\n  ): Promise<DeployStackResult> {\n\n    const newVersion = bootstrapVersionFromTemplate(template);\n    if (this.currentToolkitInfo.found && newVersion < this.currentToolkitInfo.version && !options.force) {\n      logging.warning(`Bootstrap stack already at version '${this.currentToolkitInfo.version}'. Not downgrading it to version '${newVersion}' (use --force if you intend to downgrade)`);\n      if (newVersion === 0) {\n        // A downgrade with 0 as target version means we probably have a new-style bootstrap in the account,\n        // and an old-style bootstrap as current target, which means the user probably forgot to put this flag in.\n        logging.warning('(Did you set the \\'@aws-cdk/core:newStyleStackSynthesis\\' feature flag in cdk.json?)');\n      }\n\n      return {\n        noOp: true,\n        outputs: {},\n        stackArn: this.currentToolkitInfo.bootstrapStack.stackId,\n      };\n    }\n\n    const outdir = await fs.mkdtemp(path.join(os.tmpdir(), 'cdk-bootstrap'));\n    const builder = new cxapi.CloudAssemblyBuilder(outdir);\n    const templateFile = `${this.toolkitStackName}.template.json`;\n    await fs.writeJson(path.join(builder.outdir, templateFile), template, { spaces: 2 });\n\n    builder.addArtifact(this.toolkitStackName, {\n      type: cxschema.ArtifactType.AWS_CLOUDFORMATION_STACK,\n      environment: cxapi.EnvironmentUtils.format(this.resolvedEnvironment.account, this.resolvedEnvironment.region),\n      properties: {\n        templateFile,\n        terminationProtection: options.terminationProtection ?? false,\n      },\n    });\n\n    const assembly = builder.buildAssembly();\n\n    return deployStack({\n      stack: assembly.getStackByName(this.toolkitStackName),\n      resolvedEnvironment: this.resolvedEnvironment,\n      sdk: this.sdk,\n      sdkProvider: this.sdkProvider,\n      force: options.force,\n      roleArn: options.roleArn,\n      tags: options.tags,\n      execute: options.execute,\n      parameters,\n      usePreviousParameters: true,\n      // Obviously we can't need a bootstrap stack to deploy a bootstrap stack\n      toolkitInfo: ToolkitInfo.bootstraplessDeploymentsOnly(this.sdk),\n    });\n  }\n}\n\nexport function bootstrapVersionFromTemplate(template: any): number {\n  const versionSources = [\n    template.Outputs?.[BOOTSTRAP_VERSION_OUTPUT]?.Value,\n    template.Resources?.[BOOTSTRAP_VERSION_RESOURCE]?.Properties?.Value,\n  ];\n\n  for (const vs of versionSources) {\n    if (typeof vs === 'number') { return vs; }\n    if (typeof vs === 'string' && !isNaN(parseInt(vs, 10))) {\n      return parseInt(vs, 10);\n    }\n  }\n  return 0;\n}\n"]}
116
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"deploy-bootstrap.js","sourceRoot":"","sources":["deploy-bootstrap.ts"],"names":[],"mappings":"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","sourcesContent":["import * as os from 'os';\nimport * as path from 'path';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as fs from 'fs-extra';\nimport * as logging from '../../logging';\nimport { Mode, SdkProvider, ISDK } from '../aws-auth';\nimport { deployStack, DeployStackResult } from '../deploy-stack';\nimport { DEFAULT_TOOLKIT_STACK_NAME, ToolkitInfo } from '../toolkit-info';\nimport { BOOTSTRAP_VERSION_OUTPUT, BootstrapEnvironmentOptions, BOOTSTRAP_VERSION_RESOURCE } from './bootstrap-props';\n\n/**\n * A class to hold state around stack bootstrapping\n *\n * This class exists so we can break bootstrapping into 2 phases:\n *\n * ```ts\n * const current = BootstrapStack.lookup(...);\n * // ...\n * current.update(newTemplate, ...);\n * ```\n *\n * And do something in between the two phases (such as look at the\n * current bootstrap stack and doing something intelligent).\n */\nexport class BootstrapStack {\n  public static async lookup(sdkProvider: SdkProvider, environment: cxapi.Environment, toolkitStackName?: string) {\n    toolkitStackName = toolkitStackName ?? DEFAULT_TOOLKIT_STACK_NAME;\n\n    const resolvedEnvironment = await sdkProvider.resolveEnvironment(environment);\n    const sdk = (await sdkProvider.forEnvironment(resolvedEnvironment, Mode.ForWriting)).sdk;\n\n    const currentToolkitInfo = await ToolkitInfo.lookup(resolvedEnvironment, sdk, toolkitStackName);\n\n    return new BootstrapStack(sdkProvider, sdk, resolvedEnvironment, toolkitStackName, currentToolkitInfo);\n  }\n\n  protected constructor(\n    private readonly sdkProvider: SdkProvider,\n    private readonly sdk: ISDK,\n    private readonly resolvedEnvironment: cxapi.Environment,\n    private readonly toolkitStackName: string,\n    private readonly currentToolkitInfo: ToolkitInfo) {\n  }\n\n  public get parameters(): Record<string, string> {\n    return this.currentToolkitInfo.found ? this.currentToolkitInfo.bootstrapStack.parameters : {};\n  }\n\n  public get terminationProtection() {\n    return this.currentToolkitInfo.found ? this.currentToolkitInfo.bootstrapStack.terminationProtection : undefined;\n  }\n\n  public async partition(): Promise<string> {\n    return (await this.sdk.currentAccount()).partition;\n  }\n\n  /**\n   * Perform the actual deployment of a bootstrap stack, given a template and some parameters\n   */\n  public async update(\n    template: any,\n    parameters: Record<string, string | undefined>,\n    options: Omit<BootstrapEnvironmentOptions, 'parameters'>,\n  ): Promise<DeployStackResult> {\n\n    const newVersion = bootstrapVersionFromTemplate(template);\n    if (this.currentToolkitInfo.found && newVersion < this.currentToolkitInfo.version && !options.force) {\n      logging.warning(`Bootstrap stack already at version '${this.currentToolkitInfo.version}'. Not downgrading it to version '${newVersion}' (use --force if you intend to downgrade)`);\n      if (newVersion === 0) {\n        // A downgrade with 0 as target version means we probably have a new-style bootstrap in the account,\n        // and an old-style bootstrap as current target, which means the user probably forgot to put this flag in.\n        logging.warning('(Did you set the \\'@aws-cdk/core:newStyleStackSynthesis\\' feature flag in cdk.json?)');\n      }\n\n      return {\n        noOp: true,\n        outputs: {},\n        stackArn: this.currentToolkitInfo.bootstrapStack.stackId,\n      };\n    }\n\n    const outdir = await fs.mkdtemp(path.join(os.tmpdir(), 'cdk-bootstrap'));\n    const builder = new cxapi.CloudAssemblyBuilder(outdir);\n    const templateFile = `${this.toolkitStackName}.template.json`;\n    await fs.writeJson(path.join(builder.outdir, templateFile), template, { spaces: 2 });\n\n    builder.addArtifact(this.toolkitStackName, {\n      type: cxschema.ArtifactType.AWS_CLOUDFORMATION_STACK,\n      environment: cxapi.EnvironmentUtils.format(this.resolvedEnvironment.account, this.resolvedEnvironment.region),\n      properties: {\n        templateFile,\n        terminationProtection: options.terminationProtection ?? false,\n      },\n    });\n\n    const assembly = builder.buildAssembly();\n\n    return deployStack({\n      stack: assembly.getStackByName(this.toolkitStackName),\n      resolvedEnvironment: this.resolvedEnvironment,\n      sdk: this.sdk,\n      sdkProvider: this.sdkProvider,\n      force: options.force,\n      roleArn: options.roleArn,\n      tags: options.tags,\n      execute: options.execute,\n      parameters,\n      usePreviousParameters: true,\n      // Obviously we can't need a bootstrap stack to deploy a bootstrap stack\n      toolkitInfo: ToolkitInfo.bootstraplessDeploymentsOnly(this.sdk),\n    });\n  }\n}\n\nexport function bootstrapVersionFromTemplate(template: any): number {\n  const versionSources = [\n    template.Outputs?.[BOOTSTRAP_VERSION_OUTPUT]?.Value,\n    template.Resources?.[BOOTSTRAP_VERSION_RESOURCE]?.Properties?.Value,\n  ];\n\n  for (const vs of versionSources) {\n    if (typeof vs === 'number') { return vs; }\n    if (typeof vs === 'string' && !isNaN(parseInt(vs, 10))) {\n      return parseInt(vs, 10);\n    }\n  }\n  return 0;\n}\n"]}
@@ -158,6 +158,30 @@ export interface DeployStackOptions {
158
158
  * @default - Use the stored template
159
159
  */
160
160
  readonly overrideTemplate?: any;
161
+ /**
162
+ * Whether to build assets before publishing.
163
+ *
164
+ * @default true To remain backward compatible.
165
+ */
166
+ readonly buildAssets?: boolean;
167
+ }
168
+ export interface BuildStackAssetsOptions {
169
+ /**
170
+ * Stack with assets to build.
171
+ */
172
+ readonly stack: cxapi.CloudFormationStackArtifact;
173
+ /**
174
+ * Name of the toolkit stack, if not the default name.
175
+ *
176
+ * @default 'CDKToolkit'
177
+ */
178
+ readonly toolkitStackName?: string;
179
+ /**
180
+ * Execution role for the building.
181
+ *
182
+ * @default - Current role
183
+ */
184
+ readonly roleArn?: string;
161
185
  }
162
186
  export interface DestroyStackOptions {
163
187
  stack: cxapi.CloudFormationStackArtifact;
@@ -221,6 +245,10 @@ export declare class CloudFormationDeployments {
221
245
  * - The Execution Role that should be passed to CloudFormation.
222
246
  */
223
247
  private prepareSdkFor;
248
+ /**
249
+ * Build a stack's assets.
250
+ */
251
+ buildStackAssets(options: BuildStackAssetsOptions): Promise<void>;
224
252
  /**
225
253
  * Publish all asset manifests that are referenced by the given stack
226
254
  */
@@ -30,22 +30,21 @@ const placeholders_1 = require("./util/placeholders");
30
30
  * function can then decide to use them or fallback to another role.
31
31
  */
32
32
  async function prepareSdkWithLookupRoleFor(sdkProvider, stack) {
33
- var _a, _b, _c, _d, _e;
34
33
  const resolvedEnvironment = await sdkProvider.resolveEnvironment(stack.environment);
35
34
  // Substitute any placeholders with information about the current environment
36
35
  const arns = await placeholders_1.replaceEnvPlaceholders({
37
- lookupRoleArn: (_a = stack.lookupRole) === null || _a === void 0 ? void 0 : _a.arn,
36
+ lookupRoleArn: stack.lookupRole?.arn,
38
37
  }, resolvedEnvironment, sdkProvider);
39
38
  // try to assume the lookup role
40
39
  const warningMessage = `Could not assume ${arns.lookupRoleArn}, proceeding anyway.`;
41
- const upgradeMessage = `(To get rid of this warning, please upgrade to bootstrap version >= ${(_b = stack.lookupRole) === null || _b === void 0 ? void 0 : _b.requiresBootstrapStackVersion})`;
40
+ const upgradeMessage = `(To get rid of this warning, please upgrade to bootstrap version >= ${stack.lookupRole?.requiresBootstrapStackVersion})`;
42
41
  try {
43
42
  const stackSdk = await sdkProvider.forEnvironment(resolvedEnvironment, credentials_1.Mode.ForReading, {
44
43
  assumeRoleArn: arns.lookupRoleArn,
45
- assumeRoleExternalId: (_c = stack.lookupRole) === null || _c === void 0 ? void 0 : _c.assumeRoleExternalId,
44
+ assumeRoleExternalId: stack.lookupRole?.assumeRoleExternalId,
46
45
  });
47
46
  // if we succeed in assuming the lookup role, make sure we have the correct bootstrap stack version
48
- if (stackSdk.didAssumeRole && ((_d = stack.lookupRole) === null || _d === void 0 ? void 0 : _d.bootstrapStackVersionSsmParameter) && stack.lookupRole.requiresBootstrapStackVersion) {
47
+ if (stackSdk.didAssumeRole && stack.lookupRole?.bootstrapStackVersionSsmParameter && stack.lookupRole.requiresBootstrapStackVersion) {
49
48
  const version = await toolkit_info_1.ToolkitInfo.versionFromSsmParameter(stackSdk.sdk, stack.lookupRole.bootstrapStackVersionSsmParameter);
50
49
  if (version < stack.lookupRole.requiresBootstrapStackVersion) {
51
50
  throw new Error(`Bootstrap stack version '${stack.lookupRole.requiresBootstrapStackVersion}' is required, found version '${version}'.`);
@@ -53,7 +52,7 @@ async function prepareSdkWithLookupRoleFor(sdkProvider, stack) {
53
52
  // we may not have assumed the lookup role because one was not provided
54
53
  // if that is the case then don't print the upgrade warning
55
54
  }
56
- else if (!stackSdk.didAssumeRole && ((_e = stack.lookupRole) === null || _e === void 0 ? void 0 : _e.requiresBootstrapStackVersion)) {
55
+ else if (!stackSdk.didAssumeRole && stack.lookupRole?.requiresBootstrapStackVersion) {
57
56
  logging_1.warning(upgradeMessage);
58
57
  }
59
58
  return { ...stackSdk, resolvedEnvironment };
@@ -90,7 +89,6 @@ class CloudFormationDeployments {
90
89
  return nested_stack_helpers_1.loadCurrentTemplate(stackArtifact, sdk);
91
90
  }
92
91
  async resourceIdentifierSummaries(stackArtifact, toolkitStackName) {
93
- var _a;
94
92
  logging_1.debug(`Retrieving template summary for stack ${stackArtifact.displayName}.`);
95
93
  // Currently, needs to use `deploy-role` since it may need to read templates in the staging
96
94
  // bucket which have been encrypted with a KMS key (and lookup-role may not read encrypted things)
@@ -103,14 +101,16 @@ class CloudFormationDeployments {
103
101
  if (!response.ResourceIdentifierSummaries) {
104
102
  logging_1.debug('GetTemplateSummary API call did not return "ResourceIdentifierSummaries"');
105
103
  }
106
- return (_a = response.ResourceIdentifierSummaries) !== null && _a !== void 0 ? _a : [];
104
+ return response.ResourceIdentifierSummaries ?? [];
107
105
  }
108
106
  async deployStack(options) {
109
107
  const { stackSdk, resolvedEnvironment, cloudFormationRoleArn } = await this.prepareSdkFor(options.stack, options.roleArn);
110
108
  const toolkitInfo = await toolkit_info_1.ToolkitInfo.lookup(resolvedEnvironment, stackSdk, options.toolkitStackName);
111
109
  // Publish any assets before doing the actual deploy (do not publish any assets on import operation)
112
110
  if (options.resourcesToImport === undefined) {
113
- await this.publishStackAssets(options.stack, toolkitInfo);
111
+ await this.publishStackAssets(options.stack, toolkitInfo, {
112
+ buildAssets: options.buildAssets ?? true,
113
+ });
114
114
  }
115
115
  // Do a verification of the bootstrap stack version
116
116
  await this.validateBootstrapStackVersion(options.stack.stackName, options.stack.requiresBootstrapStackVersion, options.stack.bootstrapStackVersionSsmParameter, toolkitInfo);
@@ -152,9 +152,8 @@ class CloudFormationDeployments {
152
152
  });
153
153
  }
154
154
  async stackExists(options) {
155
- var _a;
156
155
  const { stackSdk } = await this.prepareSdkFor(options.stack, undefined, credentials_1.Mode.ForReading);
157
- const stack = await cloudformation_1.CloudFormationStack.lookup(stackSdk.cloudFormation(), (_a = options.deployName) !== null && _a !== void 0 ? _a : options.stack.stackName);
156
+ const stack = await cloudformation_1.CloudFormationStack.lookup(stackSdk.cloudFormation(), options.deployName ?? options.stack.stackName);
158
157
  return stack.exists;
159
158
  }
160
159
  async prepareSdkWithLookupOrDeployRole(stackArtifact) {
@@ -190,7 +189,7 @@ class CloudFormationDeployments {
190
189
  const arns = await placeholders_1.replaceEnvPlaceholders({
191
190
  assumeRoleArn: stack.assumeRoleArn,
192
191
  // Use the override if given, otherwise use the field from the stack
193
- cloudFormationRoleArn: roleArn !== null && roleArn !== void 0 ? roleArn : stack.cloudFormationExecutionRoleArn,
192
+ cloudFormationRoleArn: roleArn ?? stack.cloudFormationExecutionRoleArn,
194
193
  }, resolvedEnvironment, this.sdkProvider);
195
194
  const stackSdk = await this.sdkProvider.forEnvironment(resolvedEnvironment, mode, {
196
195
  assumeRoleArn: arns.assumeRoleArn,
@@ -202,16 +201,32 @@ class CloudFormationDeployments {
202
201
  cloudFormationRoleArn: arns.cloudFormationRoleArn,
203
202
  };
204
203
  }
204
+ /**
205
+ * Build a stack's assets.
206
+ */
207
+ async buildStackAssets(options) {
208
+ const { stackSdk, resolvedEnvironment } = await this.prepareSdkFor(options.stack, options.roleArn);
209
+ const toolkitInfo = await toolkit_info_1.ToolkitInfo.lookup(resolvedEnvironment, stackSdk, options.toolkitStackName);
210
+ const stackEnv = await this.sdkProvider.resolveEnvironment(options.stack.environment);
211
+ const assetArtifacts = options.stack.dependencies.filter(cxapi.AssetManifestArtifact.isAssetManifestArtifact);
212
+ for (const assetArtifact of assetArtifacts) {
213
+ await this.validateBootstrapStackVersion(options.stack.stackName, assetArtifact.requiresBootstrapStackVersion, assetArtifact.bootstrapStackVersionSsmParameter, toolkitInfo);
214
+ const manifest = cdk_assets_1.AssetManifest.fromFile(assetArtifact.file);
215
+ await asset_publishing_1.buildAssets(manifest, this.sdkProvider, stackEnv);
216
+ }
217
+ }
205
218
  /**
206
219
  * Publish all asset manifests that are referenced by the given stack
207
220
  */
208
- async publishStackAssets(stack, toolkitInfo) {
221
+ async publishStackAssets(stack, toolkitInfo, options = {}) {
209
222
  const stackEnv = await this.sdkProvider.resolveEnvironment(stack.environment);
210
223
  const assetArtifacts = stack.dependencies.filter(cxapi.AssetManifestArtifact.isAssetManifestArtifact);
211
224
  for (const assetArtifact of assetArtifacts) {
212
225
  await this.validateBootstrapStackVersion(stack.stackName, assetArtifact.requiresBootstrapStackVersion, assetArtifact.bootstrapStackVersionSsmParameter, toolkitInfo);
213
226
  const manifest = cdk_assets_1.AssetManifest.fromFile(assetArtifact.file);
214
- await asset_publishing_1.publishAssets(manifest, this.sdkProvider, stackEnv);
227
+ await asset_publishing_1.publishAssets(manifest, this.sdkProvider, stackEnv, {
228
+ buildAssets: options.buildAssets ?? true,
229
+ });
215
230
  }
216
231
  }
217
232
  /**
@@ -230,4 +245,4 @@ class CloudFormationDeployments {
230
245
  }
231
246
  }
232
247
  exports.CloudFormationDeployments = CloudFormationDeployments;
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(\n    rootStackArtifact: cxapi.CloudFormationStackArtifact,\n    retrieveProcessedTemplate: boolean = false,\n  ): Promise<Template> {\n    const sdk = (await this.prepareSdkWithLookupOrDeployRole(rootStackArtifact)).stackSdk;\n    return (await loadCurrentTemplateWithNestedStacks(rootStackArtifact, sdk, retrieveProcessedTemplate)).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"]}
248
+ //# 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 { buildAssets, 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  /**\n   * Whether to build assets before publishing.\n   *\n   * @default true To remain backward compatible.\n   */\n  readonly buildAssets?: boolean;\n}\n\nexport interface BuildStackAssetsOptions {\n  /**\n   * Stack with assets to build.\n   */\n  readonly stack: cxapi.CloudFormationStackArtifact;\n\n  /**\n   * Name of the toolkit stack, if not the default name.\n   *\n   * @default 'CDKToolkit'\n   */\n  readonly toolkitStackName?: string;\n\n  /**\n   * Execution role for the building.\n   *\n   * @default - Current role\n   */\n  readonly roleArn?: string;\n}\n\ninterface PublishStackAssetsOptions {\n  /**\n   * Whether to build assets before publishing.\n   *\n   * @default true To remain backward compatible.\n   */\n  readonly buildAssets?: boolean;\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(\n    rootStackArtifact: cxapi.CloudFormationStackArtifact,\n    retrieveProcessedTemplate: boolean = false,\n  ): Promise<Template> {\n    const sdk = (await this.prepareSdkWithLookupOrDeployRole(rootStackArtifact)).stackSdk;\n    return (await loadCurrentTemplateWithNestedStacks(rootStackArtifact, sdk, retrieveProcessedTemplate)).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        buildAssets: options.buildAssets ?? true,\n      });\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   * Build a stack's assets.\n   */\n  public async buildStackAssets(options: BuildStackAssetsOptions) {\n    const { stackSdk, resolvedEnvironment } = await this.prepareSdkFor(options.stack, options.roleArn);\n    const toolkitInfo = await ToolkitInfo.lookup(resolvedEnvironment, stackSdk, options.toolkitStackName);\n\n    const stackEnv = await this.sdkProvider.resolveEnvironment(options.stack.environment);\n    const assetArtifacts = options.stack.dependencies.filter(cxapi.AssetManifestArtifact.isAssetManifestArtifact);\n\n    for (const assetArtifact of assetArtifacts) {\n      await this.validateBootstrapStackVersion(\n        options.stack.stackName,\n        assetArtifact.requiresBootstrapStackVersion,\n        assetArtifact.bootstrapStackVersionSsmParameter,\n        toolkitInfo);\n\n      const manifest = AssetManifest.fromFile(assetArtifact.file);\n      await buildAssets(manifest, this.sdkProvider, stackEnv);\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, options: PublishStackAssetsOptions = {}) {\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        buildAssets: options.buildAssets ?? true,\n      });\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"]}