aws-cdk 2.14.0 → 2.15.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 (35) hide show
  1. package/README.md +2 -1
  2. package/build-info.json +2 -2
  3. package/does-not-exist.json +1 -0
  4. package/lib/api/cloudformation-deployments.d.ts +1 -6
  5. package/lib/api/cloudformation-deployments.js +6 -72
  6. package/lib/api/evaluate-cloudformation-template.d.ts +9 -2
  7. package/lib/api/evaluate-cloudformation-template.js +14 -2
  8. package/lib/api/hotswap-deployments.js +34 -7
  9. package/lib/api/logs/find-cloudwatch-logs.js +2 -2
  10. package/lib/api/nested-stack-helpers.d.ts +28 -0
  11. package/lib/api/nested-stack-helpers.js +97 -0
  12. package/lib/cli.js +2 -2
  13. package/lib/index.js +1139 -389
  14. package/lib/notices.js +16 -7
  15. package/package.json +9 -9
  16. package/test/api/hotswap/hotswap-deployments.test.js +38 -2
  17. package/test/api/hotswap/hotswap-test-setup.d.ts +6 -4
  18. package/test/api/hotswap/hotswap-test-setup.js +46 -6
  19. package/test/api/hotswap/nested-stacks-hotswap.test.d.ts +1 -0
  20. package/test/api/hotswap/nested-stacks-hotswap.test.js +840 -0
  21. package/test/api/hotswap/state-machine-hotswap-deployments.test.js +2 -2
  22. package/test/integ/cli/cli.integtest.js +2 -2
  23. package/test/nested-stack-templates/one-lambda-one-stack-stack-with-asset-parameters.nested.template.json +52 -0
  24. package/test/nested-stack-templates/one-lambda-one-stack-stack.nested.template.json +26 -0
  25. package/test/nested-stack-templates/one-lambda-stack-with-asset-parameters.nested.template.json +29 -0
  26. package/test/nested-stack-templates/one-lambda-stack.nested.template.json +17 -0
  27. package/test/nested-stack-templates/one-lambda-version-stack.nested.template.json +20 -0
  28. package/test/{diff-nested-stacks-templates → nested-stack-templates}/one-output-one-param-stack.nested.template.json +0 -0
  29. package/test/{diff-nested-stacks-templates → nested-stack-templates}/one-resource-one-stack-stack.nested.template.json +0 -0
  30. package/test/{diff-nested-stacks-templates → nested-stack-templates}/one-resource-stack.nested.template.json +0 -0
  31. package/test/{diff-nested-stacks-templates → nested-stack-templates}/one-resource-two-stacks-stack.nested.template.json +0 -0
  32. package/test/nested-stack-templates/one-unnamed-lambda-stack.nested.template.json +16 -0
  33. package/test/nested-stack-templates/one-unnamed-lambda-two-stacks-stack.nested.template.json +34 -0
  34. package/test/notices.test.js +26 -3
  35. package/test/util.js +2 -2
@@ -13,6 +13,7 @@ const ecs_services_1 = require("./hotswap/ecs-services");
13
13
  const lambda_functions_1 = require("./hotswap/lambda-functions");
14
14
  const s3_bucket_deployments_1 = require("./hotswap/s3-bucket-deployments");
15
15
  const stepfunctions_state_machines_1 = require("./hotswap/stepfunctions-state-machines");
16
+ const nested_stack_helpers_1 = require("./nested-stack-helpers");
16
17
  /**
17
18
  * Perform a hotswap deployment,
18
19
  * short-circuiting CloudFormation if possible.
@@ -31,7 +32,7 @@ async function tryHotswapDeployment(sdkProvider, assetParams, cloudFormationStac
31
32
  // We fetch it lazily, to save a service call, in case all hotswapped resources have their physical names set.
32
33
  const listStackResources = new evaluate_cloudformation_template_1.LazyListStackResources(sdk, stackArtifact.stackName);
33
34
  const evaluateCfnTemplate = new evaluate_cloudformation_template_1.EvaluateCloudFormationTemplate({
34
- stackArtifact,
35
+ template: stackArtifact.template,
35
36
  parameters: assetParams,
36
37
  account: resolvedEnv.account,
37
38
  region: resolvedEnv.region,
@@ -39,9 +40,9 @@ async function tryHotswapDeployment(sdkProvider, assetParams, cloudFormationStac
39
40
  urlSuffix: (region) => sdk.getEndpointSuffix(region),
40
41
  listStackResources,
41
42
  });
42
- const currentTemplate = await cloudFormationStack.template();
43
- const stackChanges = cfn_diff.diffTemplate(currentTemplate, stackArtifact.template);
44
- const hotswappableChanges = await findAllHotswappableChanges(stackChanges, evaluateCfnTemplate);
43
+ const currentTemplate = await nested_stack_helpers_1.loadCurrentTemplateWithNestedStacks(stackArtifact, sdk);
44
+ const stackChanges = cfn_diff.diffTemplate(currentTemplate.deployedTemplate, stackArtifact.template);
45
+ const hotswappableChanges = await findAllHotswappableChanges(stackChanges, evaluateCfnTemplate, sdk, currentTemplate.nestedStackNames);
45
46
  if (!hotswappableChanges) {
46
47
  // this means there were changes to the template that cannot be short-circuited
47
48
  return undefined;
@@ -51,12 +52,22 @@ async function tryHotswapDeployment(sdkProvider, assetParams, cloudFormationStac
51
52
  return { noOp: hotswappableChanges.length === 0, stackArn: cloudFormationStack.stackId, outputs: cloudFormationStack.outputs };
52
53
  }
53
54
  exports.tryHotswapDeployment = tryHotswapDeployment;
54
- async function findAllHotswappableChanges(stackChanges, evaluateCfnTemplate) {
55
+ async function findAllHotswappableChanges(stackChanges, evaluateCfnTemplate, sdk, nestedStackNames) {
56
+ var _a, _b;
55
57
  const resourceDifferences = getStackResourceDifferences(stackChanges);
56
58
  let foundNonHotswappableChange = false;
57
59
  const promises = [];
60
+ const hotswappableResources = new Array();
58
61
  // gather the results of the detector functions
59
62
  for (const [logicalId, change] of Object.entries(resourceDifferences)) {
63
+ if (((_a = change.newValue) === null || _a === void 0 ? void 0 : _a.Type) === 'AWS::CloudFormation::Stack' && ((_b = change.oldValue) === null || _b === void 0 ? void 0 : _b.Type) === 'AWS::CloudFormation::Stack') {
64
+ const nestedHotswappableResources = await findNestedHotswappableChanges(logicalId, change, nestedStackNames, evaluateCfnTemplate, sdk);
65
+ if (!nestedHotswappableResources) {
66
+ return undefined;
67
+ }
68
+ hotswappableResources.push(...nestedHotswappableResources);
69
+ continue;
70
+ }
60
71
  const resourceHotswapEvaluation = isCandidateForHotswapping(change);
61
72
  if (resourceHotswapEvaluation === common_1.ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT) {
62
73
  foundNonHotswappableChange = true;
@@ -81,7 +92,6 @@ async function findAllHotswappableChanges(stackChanges, evaluateCfnTemplate) {
81
92
  const hotswapDetectionResults = await Promise.all(detectorResultPromises);
82
93
  changesDetectionResults.push(hotswapDetectionResults);
83
94
  }
84
- const hotswappableResources = new Array();
85
95
  for (const hotswapDetectionResults of changesDetectionResults) {
86
96
  const perChangeHotswappableResources = new Array();
87
97
  for (const result of hotswapDetectionResults) {
@@ -147,6 +157,19 @@ function filterDict(dict, func) {
147
157
  return acc;
148
158
  }, {});
149
159
  }
160
+ /** Finds any hotswappable changes in all nested stacks. */
161
+ async function findNestedHotswappableChanges(logicalId, change, nestedStackNames, evaluateCfnTemplate, sdk) {
162
+ var _a, _b, _c, _d, _e, _f, _g, _h;
163
+ const nestedStackName = nestedStackNames[logicalId].nestedStackPhysicalName;
164
+ // the stack name could not be found in CFN, so this is a newly created nested stack
165
+ if (!nestedStackName) {
166
+ return undefined;
167
+ }
168
+ const nestedStackParameters = await evaluateCfnTemplate.evaluateCfnExpression((_b = (_a = change.newValue) === null || _a === void 0 ? void 0 : _a.Properties) === null || _b === void 0 ? void 0 : _b.Parameters);
169
+ const evaluateNestedCfnTemplate = evaluateCfnTemplate.createNestedEvaluateCloudFormationTemplate(new evaluate_cloudformation_template_1.LazyListStackResources(sdk, nestedStackName), (_d = (_c = change.newValue) === null || _c === void 0 ? void 0 : _c.Properties) === null || _d === void 0 ? void 0 : _d.NestedTemplate, nestedStackParameters);
170
+ const nestedDiff = cfn_diff.diffTemplate((_f = (_e = change.oldValue) === null || _e === void 0 ? void 0 : _e.Properties) === null || _f === void 0 ? void 0 : _f.NestedTemplate, (_h = (_g = change.newValue) === null || _g === void 0 ? void 0 : _g.Properties) === null || _h === void 0 ? void 0 : _h.NestedTemplate);
171
+ return findAllHotswappableChanges(nestedDiff, evaluateNestedCfnTemplate, sdk, nestedStackNames[logicalId].nestedChildStackNames);
172
+ }
150
173
  /** Returns 'true' if a pair of changes is for the same resource. */
151
174
  function changesAreForSameResource(oldChange, newChange) {
152
175
  return oldChange.oldResourceType === newChange.newResourceType &&
@@ -174,6 +197,10 @@ function isCandidateForHotswapping(change) {
174
197
  if (!change.newValue || !change.oldValue) {
175
198
  return common_1.ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;
176
199
  }
200
+ // a resource has had its type changed
201
+ if (change.newValue.Type !== change.oldValue.Type) {
202
+ return common_1.ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;
203
+ }
177
204
  // Ignore Metadata changes
178
205
  if (change.newValue.Type === 'AWS::CDK::Metadata') {
179
206
  return common_1.ChangeHotswapImpact.IRRELEVANT;
@@ -206,4 +233,4 @@ async function applyHotswappableChange(sdk, hotswapOperation) {
206
233
  sdk.removeCustomUserAgent(customUserAgent);
207
234
  }
208
235
  }
209
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hotswap-deployments.js","sourceRoot":"","sources":["hotswap-deployments.ts"],"names":[],"mappings":";;;AAAA,yDAAyD;AAEzD,+BAA+B;AAC/B,wCAAmC;AACnC,yCAAqD;AAErD,yFAA4G;AAC5G,mFAAkF;AAClF,uEAAqF;AACrF,6CAAiI;AACjI,yDAAwE;AACxE,iEAAgF;AAChF,2EAAyF;AACzF,yFAA0F;AAG1F;;;;;;GAMG;AACI,KAAK,UAAU,oBAAoB,CACxC,WAAwB,EAAE,WAAsC,EAChE,mBAAwC,EAAE,aAAgD;IAE1F,2FAA2F;IAC3F,MAAM,WAAW,GAAG,MAAM,WAAW,CAAC,kBAAkB,CAAC,aAAa,CAAC,WAAW,CAAC,CAAC;IACpF,8GAA8G;IAC9G,kGAAkG;IAClG,MAAM,GAAG,GAAG,CAAC,MAAM,WAAW,CAAC,cAAc,CAAC,WAAW,EAAE,eAAI,CAAC,UAAU,CAAC,CAAC,CAAC,GAAG,CAAC;IACjF,sCAAsC;IACtC,sGAAsG;IACtG,8GAA8G;IAC9G,MAAM,kBAAkB,GAAG,IAAI,yDAAsB,CAAC,GAAG,EAAE,aAAa,CAAC,SAAS,CAAC,CAAC;IACpF,MAAM,mBAAmB,GAAG,IAAI,iEAA8B,CAAC;QAC7D,aAAa;QACb,UAAU,EAAE,WAAW;QACvB,OAAO,EAAE,WAAW,CAAC,OAAO;QAC5B,MAAM,EAAE,WAAW,CAAC,MAAM;QAC1B,SAAS,EAAE,CAAC,MAAM,GAAG,CAAC,cAAc,EAAE,CAAC,CAAC,SAAS;QACjD,SAAS,EAAE,CAAC,MAAM,EAAE,EAAE,CAAC,GAAG,CAAC,iBAAiB,CAAC,MAAM,CAAC;QACpD,kBAAkB;KACnB,CAAC,CAAC;IAEH,MAAM,eAAe,GAAG,MAAM,mBAAmB,CAAC,QAAQ,EAAE,CAAC;IAC7D,MAAM,YAAY,GAAG,QAAQ,CAAC,YAAY,CAAC,eAAe,EAAE,aAAa,CAAC,QAAQ,CAAC,CAAC;IACpF,MAAM,mBAAmB,GAAG,MAAM,0BAA0B,CAAC,YAAY,EAAE,mBAAmB,CAAC,CAAC;IAChG,IAAI,CAAC,mBAAmB,EAAE;QACxB,+EAA+E;QAC/E,OAAO,SAAS,CAAC;KAClB;IAED,sCAAsC;IACtC,MAAM,2BAA2B,CAAC,GAAG,EAAE,mBAAmB,CAAC,CAAC;IAE5D,OAAO,EAAE,IAAI,EAAE,mBAAmB,CAAC,MAAM,KAAK,CAAC,EAAE,QAAQ,EAAE,mBAAmB,CAAC,OAAO,EAAE,OAAO,EAAE,mBAAmB,CAAC,OAAO,EAAE,CAAC;AACjI,CAAC;AAnCD,oDAmCC;AAED,KAAK,UAAU,0BAA0B,CACvC,YAAmC,EAAE,mBAAmD;IAExF,MAAM,mBAAmB,GAAG,2BAA2B,CAAC,YAAY,CAAC,CAAC;IAEtE,IAAI,0BAA0B,GAAG,KAAK,CAAC;IACvC,MAAM,QAAQ,GAA+C,EAAE,CAAC;IAChE,+CAA+C;IAC/C,KAAK,MAAM,CAAC,SAAS,EAAE,MAAM,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,mBAAmB,CAAC,EAAE;QACrE,MAAM,yBAAyB,GAAG,yBAAyB,CAAC,MAAM,CAAC,CAAC;QAEpE,IAAI,yBAAyB,KAAK,4BAAmB,CAAC,wBAAwB,EAAE;YAC9E,0BAA0B,GAAG,IAAI,CAAC;SACnC;aAAM,IAAI,yBAAyB,KAAK,4BAAmB,CAAC,UAAU,EAAE;YACvE,sDAAsD;SACvD;aAAM;YACL,QAAQ,CAAC,IAAI,CAAC;gBACZ,qDAAkC,CAAC,SAAS,EAAE,yBAAyB,EAAE,mBAAmB,CAAC;gBAC7F,+DAAgC,CAAC,SAAS,EAAE,yBAAyB,EAAE,mBAAmB,CAAC;gBAC3F,6CAA8B,CAAC,SAAS,EAAE,yBAAyB,EAAE,mBAAmB,CAAC;gBACzF,8DAAsC,CAAC,SAAS,EAAE,yBAAyB,EAAE,mBAAmB,CAAC;gBACjG,0DAAoC,CAAC,SAAS,EAAE,yBAAyB,EAAE,mBAAmB,CAAC;gBAC/F,uDAA2B,CAAC,SAAS,EAAE,yBAAyB,EAAE,mBAAmB,CAAC;aACvF,CAAC,CAAC;SACJ;KACF;IAED,+BAA+B;IAC/B,MAAM,uBAAuB,GAAsC,EAAE,CAAC;IACtE,KAAK,MAAM,sBAAsB,IAAI,QAAQ,EAAE;QAC7C,MAAM,uBAAuB,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,sBAAsB,CAAC,CAAC;QAC1E,uBAAuB,CAAC,IAAI,CAAC,uBAAuB,CAAC,CAAC;KACvD;IAED,MAAM,qBAAqB,GAAG,IAAI,KAAK,EAAoB,CAAC;IAC5D,KAAK,MAAM,uBAAuB,IAAI,uBAAuB,EAAE;QAC7D,MAAM,8BAA8B,GAAG,IAAI,KAAK,EAAoB,CAAC;QAErE,KAAK,MAAM,MAAM,IAAI,uBAAuB,EAAE;YAC5C,IAAI,OAAO,MAAM,KAAK,QAAQ,EAAE;gBAC9B,8BAA8B,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;aAC7C;SACF;QAED,mDAAmD;QACnD,IAAI,8BAA8B,CAAC,MAAM,GAAG,CAAC,EAAE;YAC7C,qBAAqB,CAAC,IAAI,CAAC,GAAG,8BAA8B,CAAC,CAAC;YAC9D,SAAS;SACV;QAED,6FAA6F;QAC7F,mGAAmG;QACnG,2CAA2C;QAC3C,IAAI,CAAC,uBAAuB,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,4BAAmB,CAAC,UAAU,CAAC,EAAE;YAChF,0BAA0B,GAAG,IAAI,CAAC;SACnC;KACF;IAED,OAAO,0BAA0B,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,qBAAqB,CAAC;AACxE,CAAC;AAED;;;;GAIG;AACH,SAAS,2BAA2B,CAAC,YAAmC;IACtE,iEAAiE;IACjE,iGAAiG;IACjG,MAAM,kBAAkB,GAAqD,YAAY,CAAC,SAAS,CAAC,OAAO,CAAC;IAC5G,MAAM,iBAAiB,GAAG,UAAU,CAAC,kBAAkB,EAAE,SAAS,CAAC,EAAE,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;IAC3F,MAAM,oBAAoB,GAAG,UAAU,CAAC,kBAAkB,EAAE,SAAS,CAAC,EAAE,CAAC,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;IAC/F,KAAK,MAAM,CAAC,KAAK,EAAE,gBAAgB,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,oBAAoB,CAAC,EAAE;QAC5E,IAAI,gBAAgB,CAAC,UAAU,EAAE;YAC/B,MAAM,SAAS,GAAG,gBAAgB,CAAC;YACnC,yCAAyC;YACzC,MAAM,sBAAsB,GAAG,MAAM,CAAC,OAAO,CAAC,iBAAiB,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE,SAAS,CAAC,EAAE,EAAE;gBACvF,OAAO,yBAAyB,CAAC,SAAS,EAAE,SAAS,CAAC,CAAC;YACzD,CAAC,CAAC,CAAC;YACH,2DAA2D;YAC3D,IAAI,sBAAsB,EAAE;gBAC1B,MAAM,CAAC,YAAY,EAAE,qBAAqB,CAAC,GAAG,sBAAsB,CAAC;gBACrE,oBAAoB,CAAC,KAAK,CAAC,GAAG,oBAAoB,CAAC,qBAAqB,EAAE,SAAS,CAAC,CAAC;gBACrF,uDAAuD;gBACvD,OAAO,iBAAiB,CAAC,YAAY,CAAC,CAAC;aACxC;SACF;KACF;IACD,6DAA6D;IAC7D,sCAAsC;IACtC,uDAAuD;IACvD,OAAO;QACL,GAAG,iBAAiB;QACpB,GAAG,oBAAoB;KACxB,CAAC;AACJ,CAAC;AAED,yHAAyH;AACzH,SAAS,UAAU,CAAI,IAA0B,EAAE,IAAuB;IACxE,OAAO,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,GAAG,EAAE,CAAC,CAAC,EAAE,EAAE;QACnD,IAAI,IAAI,CAAC,CAAC,CAAC,EAAE;YACX,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;SACd;QACD,OAAO,GAAG,CAAC;IACb,CAAC,EAAE,EAA0B,CAAC,CAAC;AACjC,CAAC;AAED,oEAAoE;AACpE,SAAS,yBAAyB,CAAC,SAAsC,EAAE,SAAsC;IAC/G,OAAO,SAAS,CAAC,eAAe,KAAK,SAAS,CAAC,eAAe;QAC1D,oGAAoG;QACpG,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,aAAa,CAAC,KAAK,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC;AAC1F,CAAC;AAED,SAAS,oBAAoB,CAC3B,SAAsC,EACtC,SAAsC;IAEtC,OAAO,IAAI,QAAQ,CAAC,kBAAkB;IACpC,2GAA2G;IAC3G,SAAS,CAAC,QAAQ,EAClB,SAAS,CAAC,QAAQ,EAClB;QACE,YAAY,EAAE;YACZ,OAAO,EAAE,SAAS,CAAC,eAAe;YAClC,OAAO,EAAE,SAAS,CAAC,eAAe;SACnC;QACD,aAAa,EAAG,SAAiB,CAAC,aAAa;QAC/C,UAAU,EAAG,SAAiB,CAAC,UAAU;KAC1C,CACF,CAAC;AACJ,CAAC;AAED;;;GAGG;AACH,SAAS,yBAAyB,CAAC,MAAmC;IACpE,+FAA+F;IAC/F,IAAI,CAAC,MAAM,CAAC,QAAQ,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE;QACxC,OAAO,4BAAmB,CAAC,wBAAwB,CAAC;KACrD;IAED,0BAA0B;IAC1B,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,KAAK,oBAAoB,EAAE;QACjD,OAAO,4BAAmB,CAAC,UAAU,CAAC;KACvC;IAED,OAAO;QACL,QAAQ,EAAE,MAAM,CAAC,QAAQ;QACzB,eAAe,EAAE,MAAM,CAAC,eAAe;KACxC,CAAC;AACJ,CAAC;AAED,KAAK,UAAU,2BAA2B,CAAC,GAAS,EAAE,mBAAuC;IAC3F,eAAK,CAAC,KAAK,aAAI,yBAAyB,CAAC,CAAC;IAC1C,OAAO,OAAO,CAAC,GAAG,CAAC,mBAAmB,CAAC,GAAG,CAAC,gBAAgB,CAAC,EAAE;QAC5D,OAAO,uBAAuB,CAAC,GAAG,EAAE,gBAAgB,CAAC,CAAC;IACxD,CAAC,CAAC,CAAC,CAAC;AACN,CAAC;AAED,KAAK,UAAU,uBAAuB,CAAC,GAAS,EAAE,gBAAkC;IAClF,8EAA8E;IAC9E,MAAM,eAAe,GAAG,uBAAuB,gBAAgB,CAAC,OAAO,EAAE,CAAC;IAC1E,GAAG,CAAC,qBAAqB,CAAC,eAAe,CAAC,CAAC;IAE3C,IAAI;QACF,KAAK,MAAM,IAAI,IAAI,gBAAgB,CAAC,aAAa,EAAE;YACjD,eAAK,CAAC,MAAM,aAAI,KAAK,EAAE,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;SAC1C;QACD,OAAO,MAAM,gBAAgB,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;KAC1C;YAAS;QACR,KAAK,MAAM,IAAI,IAAI,gBAAgB,CAAC,aAAa,EAAE;YACjD,eAAK,CAAC,GAAG,aAAI,QAAQ,EAAE,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,KAAK,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC;SACtE;QACD,GAAG,CAAC,qBAAqB,CAAC,eAAe,CAAC,CAAC;KAC5C;AACH,CAAC","sourcesContent":["import * as cfn_diff from '@aws-cdk/cloudformation-diff';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as chalk from 'chalk';\nimport { print } from '../logging';\nimport { ISDK, Mode, SdkProvider } from './aws-auth';\nimport { DeployStackResult } from './deploy-stack';\nimport { EvaluateCloudFormationTemplate, LazyListStackResources } from './evaluate-cloudformation-template';\nimport { isHotswappableAppSyncChange } from './hotswap/appsync-mapping-templates';\nimport { isHotswappableCodeBuildProjectChange } from './hotswap/code-build-projects';\nimport { ICON, ChangeHotswapImpact, ChangeHotswapResult, HotswapOperation, HotswappableChangeCandidate } from './hotswap/common';\nimport { isHotswappableEcsServiceChange } from './hotswap/ecs-services';\nimport { isHotswappableLambdaFunctionChange } from './hotswap/lambda-functions';\nimport { isHotswappableS3BucketDeploymentChange } from './hotswap/s3-bucket-deployments';\nimport { isHotswappableStateMachineChange } from './hotswap/stepfunctions-state-machines';\nimport { CloudFormationStack } from './util/cloudformation';\n\n/**\n * Perform a hotswap deployment,\n * short-circuiting CloudFormation if possible.\n * If it's not possible to short-circuit the deployment\n * (because the CDK Stack contains changes that cannot be deployed without CloudFormation),\n * returns `undefined`.\n */\nexport async function tryHotswapDeployment(\n  sdkProvider: SdkProvider, assetParams: { [key: string]: string },\n  cloudFormationStack: CloudFormationStack, stackArtifact: cxapi.CloudFormationStackArtifact,\n): Promise<DeployStackResult | undefined> {\n  // resolve the environment, so we can substitute things like AWS::Region in CFN expressions\n  const resolvedEnv = await sdkProvider.resolveEnvironment(stackArtifact.environment);\n  // create a new SDK using the CLI credentials, because the default one will not work for new-style synthesis -\n  // it assumes the bootstrap deploy Role, which doesn't have permissions to update Lambda functions\n  const sdk = (await sdkProvider.forEnvironment(resolvedEnv, Mode.ForWriting)).sdk;\n  // The current resources of the Stack.\n  // We need them to figure out the physical name of a resource in case it wasn't specified by the user.\n  // We fetch it lazily, to save a service call, in case all hotswapped resources have their physical names set.\n  const listStackResources = new LazyListStackResources(sdk, stackArtifact.stackName);\n  const evaluateCfnTemplate = new EvaluateCloudFormationTemplate({\n    stackArtifact,\n    parameters: assetParams,\n    account: resolvedEnv.account,\n    region: resolvedEnv.region,\n    partition: (await sdk.currentAccount()).partition,\n    urlSuffix: (region) => sdk.getEndpointSuffix(region),\n    listStackResources,\n  });\n\n  const currentTemplate = await cloudFormationStack.template();\n  const stackChanges = cfn_diff.diffTemplate(currentTemplate, stackArtifact.template);\n  const hotswappableChanges = await findAllHotswappableChanges(stackChanges, evaluateCfnTemplate);\n  if (!hotswappableChanges) {\n    // this means there were changes to the template that cannot be short-circuited\n    return undefined;\n  }\n\n  // apply the short-circuitable changes\n  await applyAllHotswappableChanges(sdk, hotswappableChanges);\n\n  return { noOp: hotswappableChanges.length === 0, stackArn: cloudFormationStack.stackId, outputs: cloudFormationStack.outputs };\n}\n\nasync function findAllHotswappableChanges(\n  stackChanges: cfn_diff.TemplateDiff, evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<HotswapOperation[] | undefined> {\n  const resourceDifferences = getStackResourceDifferences(stackChanges);\n\n  let foundNonHotswappableChange = false;\n  const promises: Array<Array<Promise<ChangeHotswapResult>>> = [];\n  // gather the results of the detector functions\n  for (const [logicalId, change] of Object.entries(resourceDifferences)) {\n    const resourceHotswapEvaluation = isCandidateForHotswapping(change);\n\n    if (resourceHotswapEvaluation === ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT) {\n      foundNonHotswappableChange = true;\n    } else if (resourceHotswapEvaluation === ChangeHotswapImpact.IRRELEVANT) {\n      // empty 'if' just for flow-aware typing to kick in...\n    } else {\n      promises.push([\n        isHotswappableLambdaFunctionChange(logicalId, resourceHotswapEvaluation, evaluateCfnTemplate),\n        isHotswappableStateMachineChange(logicalId, resourceHotswapEvaluation, evaluateCfnTemplate),\n        isHotswappableEcsServiceChange(logicalId, resourceHotswapEvaluation, evaluateCfnTemplate),\n        isHotswappableS3BucketDeploymentChange(logicalId, resourceHotswapEvaluation, evaluateCfnTemplate),\n        isHotswappableCodeBuildProjectChange(logicalId, resourceHotswapEvaluation, evaluateCfnTemplate),\n        isHotswappableAppSyncChange(logicalId, resourceHotswapEvaluation, evaluateCfnTemplate),\n      ]);\n    }\n  }\n\n  // resolve all detector results\n  const changesDetectionResults: Array<Array<ChangeHotswapResult>> = [];\n  for (const detectorResultPromises of promises) {\n    const hotswapDetectionResults = await Promise.all(detectorResultPromises);\n    changesDetectionResults.push(hotswapDetectionResults);\n  }\n\n  const hotswappableResources = new Array<HotswapOperation>();\n  for (const hotswapDetectionResults of changesDetectionResults) {\n    const perChangeHotswappableResources = new Array<HotswapOperation>();\n\n    for (const result of hotswapDetectionResults) {\n      if (typeof result !== 'string') {\n        perChangeHotswappableResources.push(result);\n      }\n    }\n\n    // if we found any hotswappable changes, return now\n    if (perChangeHotswappableResources.length > 0) {\n      hotswappableResources.push(...perChangeHotswappableResources);\n      continue;\n    }\n\n    // no hotswappable changes found, so at least one IRRELEVANT means we can ignore this change;\n    // otherwise, all answers are REQUIRES_FULL_DEPLOYMENT, so this means we can't hotswap this change,\n    // and have to do a full deployment instead\n    if (!hotswapDetectionResults.some(hdr => hdr === ChangeHotswapImpact.IRRELEVANT)) {\n      foundNonHotswappableChange = true;\n    }\n  }\n\n  return foundNonHotswappableChange ? undefined : hotswappableResources;\n}\n\n/**\n * Returns all changes to resources in the given Stack.\n *\n * @param stackChanges the collection of all changes to a given Stack\n */\nfunction getStackResourceDifferences(stackChanges: cfn_diff.TemplateDiff): { [logicalId: string]: cfn_diff.ResourceDifference } {\n  // we need to collapse logical ID rename changes into one change,\n  // as they are represented in stackChanges as a pair of two changes: one addition and one removal\n  const allResourceChanges: { [logId: string]: cfn_diff.ResourceDifference } = stackChanges.resources.changes;\n  const allRemovalChanges = filterDict(allResourceChanges, resChange => resChange.isRemoval);\n  const allNonRemovalChanges = filterDict(allResourceChanges, resChange => !resChange.isRemoval);\n  for (const [logId, nonRemovalChange] of Object.entries(allNonRemovalChanges)) {\n    if (nonRemovalChange.isAddition) {\n      const addChange = nonRemovalChange;\n      // search for an identical removal change\n      const identicalRemovalChange = Object.entries(allRemovalChanges).find(([_, remChange]) => {\n        return changesAreForSameResource(remChange, addChange);\n      });\n      // if we found one, then this means this is a rename change\n      if (identicalRemovalChange) {\n        const [removedLogId, removedResourceChange] = identicalRemovalChange;\n        allNonRemovalChanges[logId] = makeRenameDifference(removedResourceChange, addChange);\n        // delete the removal change that forms the rename pair\n        delete allRemovalChanges[removedLogId];\n      }\n    }\n  }\n  // the final result are all of the remaining removal changes,\n  // plus all of the non-removal changes\n  // (we saved the rename changes in that object already)\n  return {\n    ...allRemovalChanges,\n    ...allNonRemovalChanges,\n  };\n}\n\n/** Filters an object with string keys based on whether the callback returns 'true' for the given value in the object. */\nfunction filterDict<T>(dict: { [key: string]: T }, func: (t: T) => boolean): { [key: string]: T } {\n  return Object.entries(dict).reduce((acc, [key, t]) => {\n    if (func(t)) {\n      acc[key] = t;\n    }\n    return acc;\n  }, {} as { [key: string]: T });\n}\n\n/** Returns 'true' if a pair of changes is for the same resource. */\nfunction changesAreForSameResource(oldChange: cfn_diff.ResourceDifference, newChange: cfn_diff.ResourceDifference): boolean {\n  return oldChange.oldResourceType === newChange.newResourceType &&\n      // this isn't great, but I don't want to bring in something like underscore just for this comparison\n      JSON.stringify(oldChange.oldProperties) === JSON.stringify(newChange.newProperties);\n}\n\nfunction makeRenameDifference(\n  remChange: cfn_diff.ResourceDifference,\n  addChange: cfn_diff.ResourceDifference,\n): cfn_diff.ResourceDifference {\n  return new cfn_diff.ResourceDifference(\n    // we have to fill in the old value, because otherwise this will be classified as a non-hotswappable change\n    remChange.oldValue,\n    addChange.newValue,\n    {\n      resourceType: {\n        oldType: remChange.oldResourceType,\n        newType: addChange.newResourceType,\n      },\n      propertyDiffs: (addChange as any).propertyDiffs,\n      otherDiffs: (addChange as any).otherDiffs,\n    },\n  );\n}\n\n/**\n * returns `ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT` if a resource was deleted, or a change that we cannot short-circuit occured.\n * Returns `ChangeHotswapImpact.IRRELEVANT` if a change that does not impact shortcircuiting occured, such as a metadata change.\n */\nfunction isCandidateForHotswapping(change: cfn_diff.ResourceDifference): HotswappableChangeCandidate | ChangeHotswapImpact {\n  // a resource has been removed OR a resource has been added; we can't short-circuit that change\n  if (!change.newValue || !change.oldValue) {\n    return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n  }\n\n  // Ignore Metadata changes\n  if (change.newValue.Type === 'AWS::CDK::Metadata') {\n    return ChangeHotswapImpact.IRRELEVANT;\n  }\n\n  return {\n    newValue: change.newValue,\n    propertyUpdates: change.propertyUpdates,\n  };\n}\n\nasync function applyAllHotswappableChanges(sdk: ISDK, hotswappableChanges: HotswapOperation[]): Promise<void[]> {\n  print(`\\n${ICON} hotswapping resources:`);\n  return Promise.all(hotswappableChanges.map(hotswapOperation => {\n    return applyHotswappableChange(sdk, hotswapOperation);\n  }));\n}\n\nasync function applyHotswappableChange(sdk: ISDK, hotswapOperation: HotswapOperation): Promise<any> {\n  // note the type of service that was successfully hotswapped in the User-Agent\n  const customUserAgent = `cdk-hotswap/success-${hotswapOperation.service}`;\n  sdk.appendCustomUserAgent(customUserAgent);\n\n  try {\n    for (const name of hotswapOperation.resourceNames) {\n      print(`   ${ICON} %s`, chalk.bold(name));\n    }\n    return await hotswapOperation.apply(sdk);\n  } finally {\n    for (const name of hotswapOperation.resourceNames) {\n      print(`${ICON} %s %s`, chalk.bold(name), chalk.green('hotswapped!'));\n    }\n    sdk.removeCustomUserAgent(customUserAgent);\n  }\n}\n"]}
236
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hotswap-deployments.js","sourceRoot":"","sources":["hotswap-deployments.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cfn_diff from '@aws-cdk/cloudformation-diff';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as chalk from 'chalk';\nimport { print } from '../logging';\nimport { ISDK, Mode, SdkProvider } from './aws-auth';\nimport { DeployStackResult } from './deploy-stack';\nimport { EvaluateCloudFormationTemplate, LazyListStackResources } from './evaluate-cloudformation-template';\nimport { isHotswappableAppSyncChange } from './hotswap/appsync-mapping-templates';\nimport { isHotswappableCodeBuildProjectChange } from './hotswap/code-build-projects';\nimport { ICON, ChangeHotswapImpact, ChangeHotswapResult, HotswapOperation, HotswappableChangeCandidate } from './hotswap/common';\nimport { isHotswappableEcsServiceChange } from './hotswap/ecs-services';\nimport { isHotswappableLambdaFunctionChange } from './hotswap/lambda-functions';\nimport { isHotswappableS3BucketDeploymentChange } from './hotswap/s3-bucket-deployments';\nimport { isHotswappableStateMachineChange } from './hotswap/stepfunctions-state-machines';\nimport { loadCurrentTemplateWithNestedStacks, NestedStackNames } from './nested-stack-helpers';\nimport { CloudFormationStack } from './util/cloudformation';\n\n/**\n * Perform a hotswap deployment,\n * short-circuiting CloudFormation if possible.\n * If it's not possible to short-circuit the deployment\n * (because the CDK Stack contains changes that cannot be deployed without CloudFormation),\n * returns `undefined`.\n */\nexport async function tryHotswapDeployment(\n  sdkProvider: SdkProvider, assetParams: { [key: string]: string },\n  cloudFormationStack: CloudFormationStack, stackArtifact: cxapi.CloudFormationStackArtifact,\n): Promise<DeployStackResult | undefined> {\n  // resolve the environment, so we can substitute things like AWS::Region in CFN expressions\n  const resolvedEnv = await sdkProvider.resolveEnvironment(stackArtifact.environment);\n  // create a new SDK using the CLI credentials, because the default one will not work for new-style synthesis -\n  // it assumes the bootstrap deploy Role, which doesn't have permissions to update Lambda functions\n  const sdk = (await sdkProvider.forEnvironment(resolvedEnv, Mode.ForWriting)).sdk;\n  // The current resources of the Stack.\n  // We need them to figure out the physical name of a resource in case it wasn't specified by the user.\n  // We fetch it lazily, to save a service call, in case all hotswapped resources have their physical names set.\n  const listStackResources = new LazyListStackResources(sdk, stackArtifact.stackName);\n  const evaluateCfnTemplate = new EvaluateCloudFormationTemplate({\n    template: stackArtifact.template,\n    parameters: assetParams,\n    account: resolvedEnv.account,\n    region: resolvedEnv.region,\n    partition: (await sdk.currentAccount()).partition,\n    urlSuffix: (region) => sdk.getEndpointSuffix(region),\n    listStackResources,\n  });\n\n  const currentTemplate = await loadCurrentTemplateWithNestedStacks(stackArtifact, sdk);\n  const stackChanges = cfn_diff.diffTemplate(currentTemplate.deployedTemplate, stackArtifact.template);\n  const hotswappableChanges = await findAllHotswappableChanges(\n    stackChanges, evaluateCfnTemplate, sdk, currentTemplate.nestedStackNames,\n  );\n\n  if (!hotswappableChanges) {\n    // this means there were changes to the template that cannot be short-circuited\n    return undefined;\n  }\n\n  // apply the short-circuitable changes\n  await applyAllHotswappableChanges(sdk, hotswappableChanges);\n\n  return { noOp: hotswappableChanges.length === 0, stackArn: cloudFormationStack.stackId, outputs: cloudFormationStack.outputs };\n}\n\nasync function findAllHotswappableChanges(\n  stackChanges: cfn_diff.TemplateDiff,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  sdk: ISDK,\n  nestedStackNames: { [nestedStackName: string]: NestedStackNames },\n): Promise<HotswapOperation[] | undefined> {\n  const resourceDifferences = getStackResourceDifferences(stackChanges);\n\n  let foundNonHotswappableChange = false;\n  const promises: Array<Array<Promise<ChangeHotswapResult>>> = [];\n  const hotswappableResources = new Array<HotswapOperation>();\n\n  // gather the results of the detector functions\n  for (const [logicalId, change] of Object.entries(resourceDifferences)) {\n    if (change.newValue?.Type === 'AWS::CloudFormation::Stack' && change.oldValue?.Type === 'AWS::CloudFormation::Stack') {\n      const nestedHotswappableResources = await findNestedHotswappableChanges(logicalId, change, nestedStackNames, evaluateCfnTemplate, sdk);\n      if (!nestedHotswappableResources) {\n        return undefined;\n      }\n      hotswappableResources.push(...nestedHotswappableResources);\n      continue;\n    }\n\n    const resourceHotswapEvaluation = isCandidateForHotswapping(change);\n\n    if (resourceHotswapEvaluation === ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT) {\n      foundNonHotswappableChange = true;\n    } else if (resourceHotswapEvaluation === ChangeHotswapImpact.IRRELEVANT) {\n      // empty 'if' just for flow-aware typing to kick in...\n    } else {\n      promises.push([\n        isHotswappableLambdaFunctionChange(logicalId, resourceHotswapEvaluation, evaluateCfnTemplate),\n        isHotswappableStateMachineChange(logicalId, resourceHotswapEvaluation, evaluateCfnTemplate),\n        isHotswappableEcsServiceChange(logicalId, resourceHotswapEvaluation, evaluateCfnTemplate),\n        isHotswappableS3BucketDeploymentChange(logicalId, resourceHotswapEvaluation, evaluateCfnTemplate),\n        isHotswappableCodeBuildProjectChange(logicalId, resourceHotswapEvaluation, evaluateCfnTemplate),\n        isHotswappableAppSyncChange(logicalId, resourceHotswapEvaluation, evaluateCfnTemplate),\n      ]);\n    }\n  }\n\n  // resolve all detector results\n  const changesDetectionResults: Array<Array<ChangeHotswapResult>> = [];\n  for (const detectorResultPromises of promises) {\n    const hotswapDetectionResults = await Promise.all(detectorResultPromises);\n    changesDetectionResults.push(hotswapDetectionResults);\n  }\n\n  for (const hotswapDetectionResults of changesDetectionResults) {\n    const perChangeHotswappableResources = new Array<HotswapOperation>();\n\n    for (const result of hotswapDetectionResults) {\n      if (typeof result !== 'string') {\n        perChangeHotswappableResources.push(result);\n      }\n    }\n\n    // if we found any hotswappable changes, return now\n    if (perChangeHotswappableResources.length > 0) {\n      hotswappableResources.push(...perChangeHotswappableResources);\n      continue;\n    }\n\n    // no hotswappable changes found, so at least one IRRELEVANT means we can ignore this change;\n    // otherwise, all answers are REQUIRES_FULL_DEPLOYMENT, so this means we can't hotswap this change,\n    // and have to do a full deployment instead\n    if (!hotswapDetectionResults.some(hdr => hdr === ChangeHotswapImpact.IRRELEVANT)) {\n      foundNonHotswappableChange = true;\n    }\n  }\n\n  return foundNonHotswappableChange ? undefined : hotswappableResources;\n}\n\n/**\n * Returns all changes to resources in the given Stack.\n *\n * @param stackChanges the collection of all changes to a given Stack\n */\nfunction getStackResourceDifferences(stackChanges: cfn_diff.TemplateDiff): { [logicalId: string]: cfn_diff.ResourceDifference } {\n  // we need to collapse logical ID rename changes into one change,\n  // as they are represented in stackChanges as a pair of two changes: one addition and one removal\n  const allResourceChanges: { [logId: string]: cfn_diff.ResourceDifference } = stackChanges.resources.changes;\n  const allRemovalChanges = filterDict(allResourceChanges, resChange => resChange.isRemoval);\n  const allNonRemovalChanges = filterDict(allResourceChanges, resChange => !resChange.isRemoval);\n  for (const [logId, nonRemovalChange] of Object.entries(allNonRemovalChanges)) {\n    if (nonRemovalChange.isAddition) {\n      const addChange = nonRemovalChange;\n      // search for an identical removal change\n      const identicalRemovalChange = Object.entries(allRemovalChanges).find(([_, remChange]) => {\n        return changesAreForSameResource(remChange, addChange);\n      });\n      // if we found one, then this means this is a rename change\n      if (identicalRemovalChange) {\n        const [removedLogId, removedResourceChange] = identicalRemovalChange;\n        allNonRemovalChanges[logId] = makeRenameDifference(removedResourceChange, addChange);\n        // delete the removal change that forms the rename pair\n        delete allRemovalChanges[removedLogId];\n      }\n    }\n  }\n  // the final result are all of the remaining removal changes,\n  // plus all of the non-removal changes\n  // (we saved the rename changes in that object already)\n  return {\n    ...allRemovalChanges,\n    ...allNonRemovalChanges,\n  };\n}\n\n/** Filters an object with string keys based on whether the callback returns 'true' for the given value in the object. */\nfunction filterDict<T>(dict: { [key: string]: T }, func: (t: T) => boolean): { [key: string]: T } {\n  return Object.entries(dict).reduce((acc, [key, t]) => {\n    if (func(t)) {\n      acc[key] = t;\n    }\n    return acc;\n  }, {} as { [key: string]: T });\n}\n\n/** Finds any hotswappable changes in all nested stacks. */\nasync function findNestedHotswappableChanges(\n  logicalId: string,\n  change: cfn_diff.ResourceDifference,\n  nestedStackNames: { [nestedStackName: string]: NestedStackNames },\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  sdk: ISDK,\n): Promise<HotswapOperation[] | undefined> {\n  const nestedStackName = nestedStackNames[logicalId].nestedStackPhysicalName;\n  // the stack name could not be found in CFN, so this is a newly created nested stack\n  if (!nestedStackName) {\n    return undefined;\n  }\n\n  const nestedStackParameters = await evaluateCfnTemplate.evaluateCfnExpression(change.newValue?.Properties?.Parameters);\n  const evaluateNestedCfnTemplate = evaluateCfnTemplate.createNestedEvaluateCloudFormationTemplate(\n    new LazyListStackResources(sdk, nestedStackName), change.newValue?.Properties?.NestedTemplate, nestedStackParameters,\n  );\n\n  const nestedDiff = cfn_diff.diffTemplate(\n    change.oldValue?.Properties?.NestedTemplate, change.newValue?.Properties?.NestedTemplate,\n  );\n\n  return findAllHotswappableChanges(nestedDiff, evaluateNestedCfnTemplate, sdk, nestedStackNames[logicalId].nestedChildStackNames);\n}\n\n/** Returns 'true' if a pair of changes is for the same resource. */\nfunction changesAreForSameResource(oldChange: cfn_diff.ResourceDifference, newChange: cfn_diff.ResourceDifference): boolean {\n  return oldChange.oldResourceType === newChange.newResourceType &&\n      // this isn't great, but I don't want to bring in something like underscore just for this comparison\n      JSON.stringify(oldChange.oldProperties) === JSON.stringify(newChange.newProperties);\n}\n\nfunction makeRenameDifference(\n  remChange: cfn_diff.ResourceDifference,\n  addChange: cfn_diff.ResourceDifference,\n): cfn_diff.ResourceDifference {\n  return new cfn_diff.ResourceDifference(\n    // we have to fill in the old value, because otherwise this will be classified as a non-hotswappable change\n    remChange.oldValue,\n    addChange.newValue,\n    {\n      resourceType: {\n        oldType: remChange.oldResourceType,\n        newType: addChange.newResourceType,\n      },\n      propertyDiffs: (addChange as any).propertyDiffs,\n      otherDiffs: (addChange as any).otherDiffs,\n    },\n  );\n}\n\n/**\n * returns `ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT` if a resource was deleted, or a change that we cannot short-circuit occured.\n * Returns `ChangeHotswapImpact.IRRELEVANT` if a change that does not impact shortcircuiting occured, such as a metadata change.\n */\nfunction isCandidateForHotswapping(change: cfn_diff.ResourceDifference): HotswappableChangeCandidate | ChangeHotswapImpact {\n  // a resource has been removed OR a resource has been added; we can't short-circuit that change\n  if (!change.newValue || !change.oldValue) {\n    return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n  }\n\n  // a resource has had its type changed\n  if (change.newValue.Type !== change.oldValue.Type) {\n    return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n  }\n\n  // Ignore Metadata changes\n  if (change.newValue.Type === 'AWS::CDK::Metadata') {\n    return ChangeHotswapImpact.IRRELEVANT;\n  }\n\n  return {\n    newValue: change.newValue,\n    propertyUpdates: change.propertyUpdates,\n  };\n}\n\nasync function applyAllHotswappableChanges(sdk: ISDK, hotswappableChanges: HotswapOperation[]): Promise<void[]> {\n  print(`\\n${ICON} hotswapping resources:`);\n  return Promise.all(hotswappableChanges.map(hotswapOperation => {\n    return applyHotswappableChange(sdk, hotswapOperation);\n  }));\n}\n\nasync function applyHotswappableChange(sdk: ISDK, hotswapOperation: HotswapOperation): Promise<any> {\n  // note the type of service that was successfully hotswapped in the User-Agent\n  const customUserAgent = `cdk-hotswap/success-${hotswapOperation.service}`;\n  sdk.appendCustomUserAgent(customUserAgent);\n\n  try {\n    for (const name of hotswapOperation.resourceNames) {\n      print(`   ${ICON} %s`, chalk.bold(name));\n    }\n    return await hotswapOperation.apply(sdk);\n  } finally {\n    for (const name of hotswapOperation.resourceNames) {\n      print(`${ICON} %s %s`, chalk.bold(name), chalk.green('hotswapped!'));\n    }\n    sdk.removeCustomUserAgent(customUserAgent);\n  }\n}\n"]}
@@ -27,7 +27,7 @@ async function findCloudWatchLogGroups(sdkProvider, stackArtifact) {
27
27
  }
28
28
  const listStackResources = new evaluate_cloudformation_template_1.LazyListStackResources(sdk, stackArtifact.stackName);
29
29
  const evaluateCfnTemplate = new evaluate_cloudformation_template_1.EvaluateCloudFormationTemplate({
30
- stackArtifact,
30
+ template: stackArtifact.template,
31
31
  parameters: {},
32
32
  account: resolvedEnv.account,
33
33
  region: resolvedEnv.region,
@@ -81,4 +81,4 @@ function findAllLogGroupNames(stackResources, evaluateCfnTemplate) {
81
81
  return logGroupNames;
82
82
  }, []);
83
83
  }
84
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"find-cloudwatch-logs.js","sourceRoot":"","sources":["find-cloudwatch-logs.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxapi from '@aws-cdk/cx-api';\nimport { CloudFormation } from 'aws-sdk';\nimport { Mode, SdkProvider, ISDK } from '../aws-auth';\nimport { prepareSdkWithLookupRoleFor } from '../cloudformation-deployments';\nimport { EvaluateCloudFormationTemplate, LazyListStackResources } from '../evaluate-cloudformation-template';\n\n// resource types that have associated CloudWatch Log Groups that should _not_ be monitored\nconst IGNORE_LOGS_RESOURCE_TYPES = ['AWS::EC2::FlowLog', 'AWS::CloudTrail::Trail', 'AWS::CodeBuild::Project'];\n\n// Resource types that will create a CloudWatch log group with a specific name if one is not provided.\n// The keys are CFN resource types, and the values are the name of the physical name property of that resource\n// and the service name that is used in the automatically created CloudWatch log group.\nconst RESOURCE_TYPES_WITH_IMPLICIT_LOGS: { [cfnResourceType: string]: { [key: string]: string } } = {\n  'AWS::Lambda::Function': {\n    PhysicalNamePropertyName: 'FunctionName',\n    LogGroupServiceName: 'lambda',\n  },\n};\n\n/**\n * Configuration needed to monitor CloudWatch Log Groups\n * found in a given CloudFormation Stack\n */\nexport interface FoundLogGroupsResult {\n  /**\n   * The resolved environment (account/region) that the log\n   * groups are deployed in\n   */\n  readonly env: cxapi.Environment;\n\n  /**\n   * The SDK that can be used to read events from the CloudWatch\n   * Log Groups in the given environment\n   */\n  readonly sdk: ISDK;\n\n  /**\n   * The names of the relevant CloudWatch Log Groups\n   * in the given CloudFormation template\n   */\n  readonly logGroupNames: string[]\n}\n\nexport async function findCloudWatchLogGroups(\n  sdkProvider: SdkProvider,\n  stackArtifact: cxapi.CloudFormationStackArtifact,\n): Promise<FoundLogGroupsResult> {\n  let sdk: ISDK;\n  const resolvedEnv = await sdkProvider.resolveEnvironment(stackArtifact.environment);\n  // try to assume the lookup role and fallback to the default credentials\n  try {\n    sdk = (await prepareSdkWithLookupRoleFor(sdkProvider, stackArtifact)).sdk;\n  } catch (e) {\n    sdk = (await sdkProvider.forEnvironment(resolvedEnv, Mode.ForReading)).sdk;\n  }\n\n  const listStackResources = new LazyListStackResources(sdk, stackArtifact.stackName);\n  const evaluateCfnTemplate = new EvaluateCloudFormationTemplate({\n    stackArtifact,\n    parameters: {},\n    account: resolvedEnv.account,\n    region: resolvedEnv.region,\n    partition: (await sdk.currentAccount()).partition,\n    urlSuffix: (region) => sdk.getEndpointSuffix(region),\n    listStackResources,\n  });\n\n  const stackResources = await listStackResources.listStackResources();\n  const logGroupNames = findAllLogGroupNames(stackResources, evaluateCfnTemplate);\n\n  return {\n    env: resolvedEnv,\n    sdk,\n    logGroupNames,\n  };\n}\n\n/**\n * Determine if a CloudWatch Log Group is associated\n * with an ignored resource\n */\nfunction isReferencedFromIgnoredResource(\n  logGroupResource: CloudFormation.StackResourceSummary,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): boolean {\n  let foundReference = false;\n  const resourcesReferencingLogGroup = evaluateCfnTemplate.findReferencesTo(logGroupResource.LogicalResourceId);\n  for (const reference of resourcesReferencingLogGroup) {\n    if (IGNORE_LOGS_RESOURCE_TYPES.includes(reference.Type)) {\n      foundReference = true;\n    }\n  }\n  return foundReference;\n}\n\n/**\n * Find all CloudWatch Log Groups in the deployed template.\n * This will find both explicitely created Log Groups (excluding those associated with ignored resources)\n * as well as Log Groups created implicitely (i.e. Lambda Functions)\n */\nfunction findAllLogGroupNames(\n  stackResources: CloudFormation.StackResourceSummary[],\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): string[] {\n  return stackResources.reduce((logGroupNames: string[], resource) => {\n    let logGroupName;\n    if (resource.ResourceType === 'AWS::Logs::LogGroup') {\n      if (!isReferencedFromIgnoredResource(resource, evaluateCfnTemplate)) {\n        logGroupName = resource.PhysicalResourceId;\n      }\n    } else if (RESOURCE_TYPES_WITH_IMPLICIT_LOGS[resource.ResourceType]) {\n      const servicePart = RESOURCE_TYPES_WITH_IMPLICIT_LOGS[resource.ResourceType].LogGroupServiceName;\n      logGroupName = `/aws/${servicePart}/${resource.PhysicalResourceId}`;\n    }\n    if (logGroupName) {\n      logGroupNames.push(logGroupName);\n    }\n    return logGroupNames;\n  }, []);\n}\n"]}
84
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"find-cloudwatch-logs.js","sourceRoot":"","sources":["find-cloudwatch-logs.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxapi from '@aws-cdk/cx-api';\nimport { CloudFormation } from 'aws-sdk';\nimport { Mode, SdkProvider, ISDK } from '../aws-auth';\nimport { prepareSdkWithLookupRoleFor } from '../cloudformation-deployments';\nimport { EvaluateCloudFormationTemplate, LazyListStackResources } from '../evaluate-cloudformation-template';\n\n// resource types that have associated CloudWatch Log Groups that should _not_ be monitored\nconst IGNORE_LOGS_RESOURCE_TYPES = ['AWS::EC2::FlowLog', 'AWS::CloudTrail::Trail', 'AWS::CodeBuild::Project'];\n\n// Resource types that will create a CloudWatch log group with a specific name if one is not provided.\n// The keys are CFN resource types, and the values are the name of the physical name property of that resource\n// and the service name that is used in the automatically created CloudWatch log group.\nconst RESOURCE_TYPES_WITH_IMPLICIT_LOGS: { [cfnResourceType: string]: { [key: string]: string } } = {\n  'AWS::Lambda::Function': {\n    PhysicalNamePropertyName: 'FunctionName',\n    LogGroupServiceName: 'lambda',\n  },\n};\n\n/**\n * Configuration needed to monitor CloudWatch Log Groups\n * found in a given CloudFormation Stack\n */\nexport interface FoundLogGroupsResult {\n  /**\n   * The resolved environment (account/region) that the log\n   * groups are deployed in\n   */\n  readonly env: cxapi.Environment;\n\n  /**\n   * The SDK that can be used to read events from the CloudWatch\n   * Log Groups in the given environment\n   */\n  readonly sdk: ISDK;\n\n  /**\n   * The names of the relevant CloudWatch Log Groups\n   * in the given CloudFormation template\n   */\n  readonly logGroupNames: string[]\n}\n\nexport async function findCloudWatchLogGroups(\n  sdkProvider: SdkProvider,\n  stackArtifact: cxapi.CloudFormationStackArtifact,\n): Promise<FoundLogGroupsResult> {\n  let sdk: ISDK;\n  const resolvedEnv = await sdkProvider.resolveEnvironment(stackArtifact.environment);\n  // try to assume the lookup role and fallback to the default credentials\n  try {\n    sdk = (await prepareSdkWithLookupRoleFor(sdkProvider, stackArtifact)).sdk;\n  } catch (e) {\n    sdk = (await sdkProvider.forEnvironment(resolvedEnv, Mode.ForReading)).sdk;\n  }\n\n  const listStackResources = new LazyListStackResources(sdk, stackArtifact.stackName);\n  const evaluateCfnTemplate = new EvaluateCloudFormationTemplate({\n    template: stackArtifact.template,\n    parameters: {},\n    account: resolvedEnv.account,\n    region: resolvedEnv.region,\n    partition: (await sdk.currentAccount()).partition,\n    urlSuffix: (region) => sdk.getEndpointSuffix(region),\n    listStackResources,\n  });\n\n  const stackResources = await listStackResources.listStackResources();\n  const logGroupNames = findAllLogGroupNames(stackResources, evaluateCfnTemplate);\n\n  return {\n    env: resolvedEnv,\n    sdk,\n    logGroupNames,\n  };\n}\n\n/**\n * Determine if a CloudWatch Log Group is associated\n * with an ignored resource\n */\nfunction isReferencedFromIgnoredResource(\n  logGroupResource: CloudFormation.StackResourceSummary,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): boolean {\n  let foundReference = false;\n  const resourcesReferencingLogGroup = evaluateCfnTemplate.findReferencesTo(logGroupResource.LogicalResourceId);\n  for (const reference of resourcesReferencingLogGroup) {\n    if (IGNORE_LOGS_RESOURCE_TYPES.includes(reference.Type)) {\n      foundReference = true;\n    }\n  }\n  return foundReference;\n}\n\n/**\n * Find all CloudWatch Log Groups in the deployed template.\n * This will find both explicitely created Log Groups (excluding those associated with ignored resources)\n * as well as Log Groups created implicitely (i.e. Lambda Functions)\n */\nfunction findAllLogGroupNames(\n  stackResources: CloudFormation.StackResourceSummary[],\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): string[] {\n  return stackResources.reduce((logGroupNames: string[], resource) => {\n    let logGroupName;\n    if (resource.ResourceType === 'AWS::Logs::LogGroup') {\n      if (!isReferencedFromIgnoredResource(resource, evaluateCfnTemplate)) {\n        logGroupName = resource.PhysicalResourceId;\n      }\n    } else if (RESOURCE_TYPES_WITH_IMPLICIT_LOGS[resource.ResourceType]) {\n      const servicePart = RESOURCE_TYPES_WITH_IMPLICIT_LOGS[resource.ResourceType].LogGroupServiceName;\n      logGroupName = `/aws/${servicePart}/${resource.PhysicalResourceId}`;\n    }\n    if (logGroupName) {\n      logGroupNames.push(logGroupName);\n    }\n    return logGroupNames;\n  }, []);\n}\n"]}
@@ -0,0 +1,28 @@
1
+ import * as cxapi from '@aws-cdk/cx-api';
2
+ import { ISDK } from './aws-auth';
3
+ import { Template } from './util/cloudformation';
4
+ /**
5
+ * Reads the currently deployed template from CloudFormation and adds a
6
+ * property, `NestedTemplate`, to any nested stacks that appear in either
7
+ * the deployed template or the newly synthesized template. `NestedTemplate`
8
+ * is populated with contents of the nested template by mutating the
9
+ * `template` property of `rootStackArtifact`. This is done for all
10
+ * nested stack resources to arbitrary depths.
11
+ */
12
+ export declare function loadCurrentTemplateWithNestedStacks(rootStackArtifact: cxapi.CloudFormationStackArtifact, sdk: ISDK): Promise<TemplateWithNestedStackNames>;
13
+ /**
14
+ * Returns the currently deployed template from CloudFormation that corresponds to `stackArtifact`.
15
+ */
16
+ export declare function loadCurrentTemplate(stackArtifact: cxapi.CloudFormationStackArtifact, sdk: ISDK): Promise<Template>;
17
+ export interface TemplateWithNestedStackNames {
18
+ readonly deployedTemplate: Template;
19
+ readonly nestedStackNames: {
20
+ [nestedStackLogicalId: string]: NestedStackNames;
21
+ };
22
+ }
23
+ export interface NestedStackNames {
24
+ readonly nestedStackPhysicalName: string | undefined;
25
+ readonly nestedChildStackNames: {
26
+ [logicalId: string]: NestedStackNames;
27
+ };
28
+ }
@@ -0,0 +1,97 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.loadCurrentTemplate = exports.loadCurrentTemplateWithNestedStacks = void 0;
4
+ const path = require("path");
5
+ const fs = require("fs-extra");
6
+ const evaluate_cloudformation_template_1 = require("./evaluate-cloudformation-template");
7
+ const cloudformation_1 = require("./util/cloudformation");
8
+ /**
9
+ * Reads the currently deployed template from CloudFormation and adds a
10
+ * property, `NestedTemplate`, to any nested stacks that appear in either
11
+ * the deployed template or the newly synthesized template. `NestedTemplate`
12
+ * is populated with contents of the nested template by mutating the
13
+ * `template` property of `rootStackArtifact`. This is done for all
14
+ * nested stack resources to arbitrary depths.
15
+ */
16
+ async function loadCurrentTemplateWithNestedStacks(rootStackArtifact, sdk) {
17
+ const deployedTemplate = await loadCurrentTemplate(rootStackArtifact, sdk);
18
+ const nestedStackNames = await addNestedTemplatesToGeneratedAndDeployedStacks(rootStackArtifact, sdk, {
19
+ generatedTemplate: rootStackArtifact.template,
20
+ deployedTemplate: deployedTemplate,
21
+ deployedStackName: rootStackArtifact.stackName,
22
+ });
23
+ return {
24
+ deployedTemplate,
25
+ nestedStackNames,
26
+ };
27
+ }
28
+ exports.loadCurrentTemplateWithNestedStacks = loadCurrentTemplateWithNestedStacks;
29
+ /**
30
+ * Returns the currently deployed template from CloudFormation that corresponds to `stackArtifact`.
31
+ */
32
+ async function loadCurrentTemplate(stackArtifact, sdk) {
33
+ return loadCurrentStackTemplate(stackArtifact.stackName, sdk);
34
+ }
35
+ exports.loadCurrentTemplate = loadCurrentTemplate;
36
+ async function loadCurrentStackTemplate(stackName, sdk) {
37
+ const cfn = sdk.cloudFormation();
38
+ const stack = await cloudformation_1.CloudFormationStack.lookup(cfn, stackName);
39
+ return stack.template();
40
+ }
41
+ async function addNestedTemplatesToGeneratedAndDeployedStacks(rootStackArtifact, sdk, parentTemplates) {
42
+ var _a, _b, _c, _d, _e;
43
+ const listStackResources = parentTemplates.deployedStackName ? new evaluate_cloudformation_template_1.LazyListStackResources(sdk, parentTemplates.deployedStackName) : undefined;
44
+ const nestedStackNames = {};
45
+ for (const [nestedStackLogicalId, generatedNestedStackResource] of Object.entries((_a = parentTemplates.generatedTemplate.Resources) !== null && _a !== void 0 ? _a : {})) {
46
+ if (!isCdkManagedNestedStack(generatedNestedStackResource)) {
47
+ continue;
48
+ }
49
+ const assetPath = generatedNestedStackResource.Metadata['aws:asset:path'];
50
+ const nestedStackTemplates = await getNestedStackTemplates(rootStackArtifact, assetPath, nestedStackLogicalId, listStackResources, sdk);
51
+ generatedNestedStackResource.Properties.NestedTemplate = nestedStackTemplates.generatedTemplate;
52
+ const deployedParentTemplate = parentTemplates.deployedTemplate;
53
+ deployedParentTemplate.Resources = (_b = deployedParentTemplate.Resources) !== null && _b !== void 0 ? _b : {};
54
+ const deployedNestedStackResource = (_c = deployedParentTemplate.Resources[nestedStackLogicalId]) !== null && _c !== void 0 ? _c : {};
55
+ deployedParentTemplate.Resources[nestedStackLogicalId] = deployedNestedStackResource;
56
+ deployedNestedStackResource.Type = (_d = deployedNestedStackResource.Type) !== null && _d !== void 0 ? _d : 'AWS::CloudFormation::Stack';
57
+ deployedNestedStackResource.Properties = (_e = deployedNestedStackResource.Properties) !== null && _e !== void 0 ? _e : {};
58
+ deployedNestedStackResource.Properties.NestedTemplate = nestedStackTemplates.deployedTemplate;
59
+ nestedStackNames[nestedStackLogicalId] = {
60
+ nestedStackPhysicalName: nestedStackTemplates.deployedStackName,
61
+ nestedChildStackNames: await addNestedTemplatesToGeneratedAndDeployedStacks(rootStackArtifact, sdk, nestedStackTemplates),
62
+ };
63
+ }
64
+ return nestedStackNames;
65
+ }
66
+ async function getNestedStackTemplates(rootStackArtifact, nestedTemplateAssetPath, nestedStackLogicalId, listStackResources, sdk) {
67
+ const nestedTemplatePath = path.join(rootStackArtifact.assembly.directory, nestedTemplateAssetPath);
68
+ // CFN generates the nested stack name in the form `ParentStackName-NestedStackLogicalID-SomeHashWeCan'tCompute,
69
+ // the arn is of the form: arn:aws:cloudformation:region:123456789012:stack/NestedStackName/AnotherHashWeDon'tNeed
70
+ // so we get the ARN and manually extract the name.
71
+ const nestedStackArn = await getNestedStackArn(nestedStackLogicalId, listStackResources);
72
+ const deployedStackName = nestedStackArn === null || nestedStackArn === void 0 ? void 0 : nestedStackArn.slice(nestedStackArn.indexOf('/') + 1, nestedStackArn.lastIndexOf('/'));
73
+ return {
74
+ generatedTemplate: JSON.parse(fs.readFileSync(nestedTemplatePath, 'utf-8')),
75
+ deployedTemplate: deployedStackName
76
+ ? await loadCurrentStackTemplate(deployedStackName, sdk)
77
+ : {},
78
+ deployedStackName,
79
+ };
80
+ }
81
+ async function getNestedStackArn(nestedStackLogicalId, listStackResources) {
82
+ var _a;
83
+ try {
84
+ const stackResources = await (listStackResources === null || listStackResources === void 0 ? void 0 : listStackResources.listStackResources());
85
+ return (_a = stackResources === null || stackResources === void 0 ? void 0 : stackResources.find(sr => sr.LogicalResourceId === nestedStackLogicalId)) === null || _a === void 0 ? void 0 : _a.PhysicalResourceId;
86
+ }
87
+ catch (e) {
88
+ if (e.message.startsWith('Stack with id ') && e.message.endsWith(' does not exist')) {
89
+ return;
90
+ }
91
+ throw e;
92
+ }
93
+ }
94
+ function isCdkManagedNestedStack(stackResource) {
95
+ return stackResource.Type === 'AWS::CloudFormation::Stack' && stackResource.Metadata && stackResource.Metadata['aws:asset:path'];
96
+ }
97
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nested-stack-helpers.js","sourceRoot":"","sources":["nested-stack-helpers.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as fs from 'fs-extra';\nimport { ISDK } from './aws-auth';\nimport { LazyListStackResources, ListStackResources } from './evaluate-cloudformation-template';\nimport { CloudFormationStack, Template } from './util/cloudformation';\n\n/**\n * Reads the currently deployed template from CloudFormation and adds a\n * property, `NestedTemplate`, to any nested stacks that appear in either\n * the deployed template or the newly synthesized template. `NestedTemplate`\n * is populated with contents of the nested template by mutating the\n * `template` property of `rootStackArtifact`. This is done for all\n * nested stack resources to arbitrary depths.\n */\nexport async function loadCurrentTemplateWithNestedStacks(\n  rootStackArtifact: cxapi.CloudFormationStackArtifact, sdk: ISDK,\n): Promise<TemplateWithNestedStackNames> {\n  const deployedTemplate = await loadCurrentTemplate(rootStackArtifact, sdk);\n  const nestedStackNames = await addNestedTemplatesToGeneratedAndDeployedStacks(rootStackArtifact, sdk, {\n    generatedTemplate: rootStackArtifact.template,\n    deployedTemplate: deployedTemplate,\n    deployedStackName: rootStackArtifact.stackName,\n  });\n\n  return {\n    deployedTemplate,\n    nestedStackNames,\n  };\n}\n\n/**\n * Returns the currently deployed template from CloudFormation that corresponds to `stackArtifact`.\n */\nexport async function loadCurrentTemplate(stackArtifact: cxapi.CloudFormationStackArtifact, sdk: ISDK): Promise<Template> {\n  return loadCurrentStackTemplate(stackArtifact.stackName, sdk);\n}\n\nasync function loadCurrentStackTemplate(stackName: string, sdk: ISDK) : Promise<Template> {\n  const cfn = sdk.cloudFormation();\n  const stack = await CloudFormationStack.lookup(cfn, stackName);\n  return stack.template();\n}\n\nasync function addNestedTemplatesToGeneratedAndDeployedStacks(\n  rootStackArtifact: cxapi.CloudFormationStackArtifact,\n  sdk: ISDK,\n  parentTemplates: StackTemplates,\n): Promise<{ [nestedStackLogicalId: string]: NestedStackNames }> {\n  const listStackResources = parentTemplates.deployedStackName ? new LazyListStackResources(sdk, parentTemplates.deployedStackName) : undefined;\n  const nestedStackNames: { [nestedStackLogicalId: string]: NestedStackNames } = {};\n  for (const [nestedStackLogicalId, generatedNestedStackResource] of Object.entries(parentTemplates.generatedTemplate.Resources ?? {})) {\n    if (!isCdkManagedNestedStack(generatedNestedStackResource)) {\n      continue;\n    }\n\n    const assetPath = generatedNestedStackResource.Metadata['aws:asset:path'];\n    const nestedStackTemplates = await getNestedStackTemplates(rootStackArtifact, assetPath, nestedStackLogicalId, listStackResources, sdk);\n\n    generatedNestedStackResource.Properties.NestedTemplate = nestedStackTemplates.generatedTemplate;\n\n    const deployedParentTemplate = parentTemplates.deployedTemplate;\n    deployedParentTemplate.Resources = deployedParentTemplate.Resources ?? {};\n    const deployedNestedStackResource = deployedParentTemplate.Resources[nestedStackLogicalId] ?? {};\n    deployedParentTemplate.Resources[nestedStackLogicalId] = deployedNestedStackResource;\n    deployedNestedStackResource.Type = deployedNestedStackResource.Type ?? 'AWS::CloudFormation::Stack';\n    deployedNestedStackResource.Properties = deployedNestedStackResource.Properties ?? {};\n    deployedNestedStackResource.Properties.NestedTemplate = nestedStackTemplates.deployedTemplate;\n\n    nestedStackNames[nestedStackLogicalId] = {\n      nestedStackPhysicalName: nestedStackTemplates.deployedStackName,\n      nestedChildStackNames: await addNestedTemplatesToGeneratedAndDeployedStacks(\n        rootStackArtifact,\n        sdk,\n        nestedStackTemplates,\n      ),\n    };\n  }\n\n  return nestedStackNames;\n}\n\nasync function getNestedStackTemplates(\n  rootStackArtifact: cxapi.CloudFormationStackArtifact, nestedTemplateAssetPath: string, nestedStackLogicalId: string,\n  listStackResources: ListStackResources | undefined, sdk: ISDK,\n): Promise<StackTemplates> {\n  const nestedTemplatePath = path.join(rootStackArtifact.assembly.directory, nestedTemplateAssetPath);\n\n  // CFN generates the nested stack name in the form `ParentStackName-NestedStackLogicalID-SomeHashWeCan'tCompute,\n  // the arn is of the form: arn:aws:cloudformation:region:123456789012:stack/NestedStackName/AnotherHashWeDon'tNeed\n  // so we get the ARN and manually extract the name.\n  const nestedStackArn = await getNestedStackArn(nestedStackLogicalId, listStackResources);\n  const deployedStackName = nestedStackArn?.slice(nestedStackArn.indexOf('/') + 1, nestedStackArn.lastIndexOf('/'));\n\n  return {\n    generatedTemplate: JSON.parse(fs.readFileSync(nestedTemplatePath, 'utf-8')),\n    deployedTemplate: deployedStackName\n      ? await loadCurrentStackTemplate(deployedStackName, sdk)\n      : {},\n    deployedStackName,\n  };\n}\n\nasync function getNestedStackArn(\n  nestedStackLogicalId: string, listStackResources?: ListStackResources,\n): Promise<string | undefined> {\n  try {\n    const stackResources = await listStackResources?.listStackResources();\n    return stackResources?.find(sr => sr.LogicalResourceId === nestedStackLogicalId)?.PhysicalResourceId;\n  } catch (e) {\n    if (e.message.startsWith('Stack with id ') && e.message.endsWith(' does not exist')) {\n      return;\n    }\n    throw e;\n  }\n}\n\nfunction isCdkManagedNestedStack(stackResource: any): stackResource is NestedStackResource {\n  return stackResource.Type === 'AWS::CloudFormation::Stack' && stackResource.Metadata && stackResource.Metadata['aws:asset:path'];\n}\n\nexport interface TemplateWithNestedStackNames {\n  readonly deployedTemplate: Template;\n  readonly nestedStackNames: { [nestedStackLogicalId: string]: NestedStackNames };\n}\n\nexport interface NestedStackNames {\n  readonly nestedStackPhysicalName: string | undefined;\n  readonly nestedChildStackNames: { [logicalId: string]: NestedStackNames };\n}\n\ninterface StackTemplates {\n  readonly generatedTemplate: any;\n  readonly deployedTemplate: any;\n  readonly deployedStackName: string | undefined;\n}\n\ninterface NestedStackResource {\n  readonly Metadata: { 'aws:asset:path': string };\n  readonly Properties: any;\n}\n"]}