aws-cdk 2.1005.0 → 2.1007.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (178) hide show
  1. package/README.md +1 -1
  2. package/THIRD_PARTY_LICENSES +86 -86
  3. package/build-info.json +2 -2
  4. package/db.json.gz +0 -0
  5. package/lib/api/aws-auth/awscli-compatible.js +9 -10
  6. package/lib/api/aws-auth/credential-plugins.js +6 -7
  7. package/lib/api/aws-auth/sdk-logger.js +3 -4
  8. package/lib/api/aws-auth/sdk-provider.js +11 -13
  9. package/lib/api/aws-auth/sdk.js +8 -9
  10. package/lib/api/aws-auth/tracing.js +3 -4
  11. package/lib/api/aws-auth/user-agent.js +4 -5
  12. package/lib/api/bootstrap/bootstrap-environment.d.ts +1 -1
  13. package/lib/api/bootstrap/bootstrap-environment.js +42 -46
  14. package/lib/api/bootstrap/bootstrap-props.d.ts +1 -1
  15. package/lib/api/bootstrap/bootstrap-props.js +1 -1
  16. package/lib/api/bootstrap/deploy-bootstrap.d.ts +1 -1
  17. package/lib/api/bootstrap/deploy-bootstrap.js +11 -14
  18. package/lib/api/{evaluate-cloudformation-template.d.ts → cloudformation/evaluate-cloudformation-template.d.ts} +4 -8
  19. package/lib/api/{evaluate-cloudformation-template.js → cloudformation/evaluate-cloudformation-template.js} +16 -25
  20. package/lib/api/cloudformation/index.d.ts +4 -0
  21. package/lib/api/cloudformation/index.js +21 -0
  22. package/lib/api/{deployments → cloudformation}/nested-stack-helpers.d.ts +3 -9
  23. package/lib/api/cloudformation/nested-stack-helpers.js +86 -0
  24. package/lib/api/cloudformation/stack-helpers.d.ts +88 -0
  25. package/lib/api/cloudformation/stack-helpers.js +158 -0
  26. package/lib/api/{util → cloudformation}/template-body-parameter.d.ts +3 -2
  27. package/lib/api/cloudformation/template-body-parameter.js +104 -0
  28. package/lib/api/context.js +3 -3
  29. package/lib/api/cxapp/cloud-assembly.d.ts +6 -4
  30. package/lib/api/cxapp/cloud-assembly.js +25 -26
  31. package/lib/api/cxapp/cloud-executable.d.ts +5 -0
  32. package/lib/api/cxapp/cloud-executable.js +9 -10
  33. package/lib/api/cxapp/environments.js +4 -4
  34. package/lib/api/cxapp/exec.d.ts +5 -4
  35. package/lib/api/cxapp/exec.js +76 -72
  36. package/lib/api/deployments/asset-publishing.d.ts +0 -2
  37. package/lib/api/deployments/asset-publishing.js +24 -31
  38. package/lib/api/deployments/assets.d.ts +1 -1
  39. package/lib/api/deployments/assets.js +12 -13
  40. package/lib/api/deployments/{cloudformation.d.ts → cfn-api.d.ts} +5 -102
  41. package/lib/api/deployments/cfn-api.js +438 -0
  42. package/lib/api/deployments/checks.d.ts +1 -1
  43. package/lib/api/deployments/checks.js +12 -13
  44. package/lib/api/deployments/deploy-stack.d.ts +2 -3
  45. package/lib/api/deployments/deploy-stack.js +34 -45
  46. package/lib/api/deployments/deployment-result.js +3 -3
  47. package/lib/api/deployments/deployments.d.ts +3 -3
  48. package/lib/api/deployments/deployments.js +35 -42
  49. package/lib/api/deployments/hotswap-deployments.d.ts +2 -2
  50. package/lib/api/deployments/hotswap-deployments.js +122 -69
  51. package/lib/api/deployments/index.d.ts +1 -2
  52. package/lib/api/deployments/index.js +2 -3
  53. package/lib/api/environment/environment-access.d.ts +2 -2
  54. package/lib/api/environment/environment-access.js +18 -20
  55. package/lib/api/environment/environment-resources.d.ts +1 -1
  56. package/lib/api/environment/environment-resources.js +17 -19
  57. package/lib/api/environment/index.d.ts +1 -0
  58. package/lib/api/environment/index.js +2 -1
  59. package/lib/api/environment/placeholders.js +23 -0
  60. package/lib/api/garbage-collection/garbage-collector.d.ts +1 -1
  61. package/lib/api/garbage-collection/garbage-collector.js +56 -66
  62. package/lib/api/garbage-collection/progress-printer.d.ts +1 -1
  63. package/lib/api/garbage-collection/progress-printer.js +7 -7
  64. package/lib/api/garbage-collection/stack-refresh.d.ts +1 -1
  65. package/lib/api/garbage-collection/stack-refresh.js +12 -15
  66. package/lib/api/hotswap/appsync-mapping-templates.d.ts +3 -3
  67. package/lib/api/hotswap/appsync-mapping-templates.js +25 -22
  68. package/lib/api/hotswap/code-build-projects.d.ts +3 -3
  69. package/lib/api/hotswap/code-build-projects.js +12 -7
  70. package/lib/api/hotswap/common.d.ts +13 -61
  71. package/lib/api/hotswap/common.js +40 -70
  72. package/lib/api/hotswap/ecs-services.d.ts +4 -4
  73. package/lib/api/hotswap/ecs-services.js +38 -21
  74. package/lib/api/hotswap/lambda-functions.d.ts +3 -3
  75. package/lib/api/hotswap/lambda-functions.js +23 -19
  76. package/lib/api/hotswap/s3-bucket-deployments.d.ts +3 -3
  77. package/lib/api/hotswap/s3-bucket-deployments.js +11 -7
  78. package/lib/api/hotswap/stepfunctions-state-machines.d.ts +3 -3
  79. package/lib/api/hotswap/stepfunctions-state-machines.js +8 -4
  80. package/lib/api/logs/find-cloudwatch-logs.js +6 -7
  81. package/lib/api/logs/logs-monitor.js +5 -8
  82. package/lib/api/plugin/plugin.js +6 -10
  83. package/lib/api/resource-import/importer.d.ts +8 -8
  84. package/lib/api/resource-import/importer.js +27 -42
  85. package/lib/api/resource-import/migrator.d.ts +3 -3
  86. package/lib/api/resource-import/migrator.js +6 -6
  87. package/lib/api/settings.d.ts +0 -3
  88. package/lib/api/settings.js +4 -40
  89. package/lib/api/stack-events/stack-activity-monitor.d.ts +1 -1
  90. package/lib/api/stack-events/stack-activity-monitor.js +12 -15
  91. package/lib/api/stack-events/stack-event-poller.js +9 -10
  92. package/lib/api/toolkit-info.d.ts +2 -2
  93. package/lib/api/toolkit-info.js +20 -24
  94. package/lib/{tree.d.ts → api/tree.d.ts} +2 -2
  95. package/lib/api/tree.js +37 -0
  96. package/lib/api/util/rwlock.js +4 -4
  97. package/lib/api/work-graph/work-graph-builder.js +4 -4
  98. package/lib/api/work-graph/work-graph.d.ts +1 -1
  99. package/lib/api/work-graph/work-graph.js +13 -15
  100. package/lib/cli/activity-printer/base.d.ts +2 -2
  101. package/lib/cli/activity-printer/base.js +6 -8
  102. package/lib/cli/activity-printer/current.js +7 -11
  103. package/lib/cli/activity-printer/history.js +2 -3
  104. package/lib/cli/cdk-toolkit.d.ts +16 -19
  105. package/lib/cli/cdk-toolkit.js +118 -74
  106. package/lib/cli/ci-systems.js +2 -3
  107. package/lib/cli/cli-config.js +4 -4
  108. package/lib/cli/cli.js +49 -50
  109. package/lib/cli/convert-to-user-input.js +110 -111
  110. package/lib/{toolkit → cli/io-host}/cli-io-host.d.ts +6 -2
  111. package/lib/cli/io-host/cli-io-host.js +356 -0
  112. package/lib/cli/io-host/index.d.ts +1 -0
  113. package/lib/{toolkit/error.js → cli/io-host/index.js} +2 -2
  114. package/lib/cli/messages.d.ts +1 -1
  115. package/lib/cli/messages.js +2 -3
  116. package/lib/cli/pretty-print-error.d.ts +1 -0
  117. package/lib/cli/pretty-print-error.js +35 -0
  118. package/lib/cli/root-dir.js +4 -4
  119. package/lib/cli/user-configuration.js +57 -14
  120. package/lib/cli/util/npm.js +3 -3
  121. package/lib/cli/util/yargs-helpers.d.ts +1 -1
  122. package/lib/cli/util/yargs-helpers.js +3 -3
  123. package/lib/cli/version.js +4 -4
  124. package/lib/commands/context.js +7 -8
  125. package/lib/commands/diff.d.ts +1 -0
  126. package/lib/commands/diff.js +7 -0
  127. package/lib/commands/init/index.d.ts +1 -0
  128. package/lib/commands/init/index.js +18 -0
  129. package/lib/commands/init/init-hooks.js +63 -0
  130. package/lib/commands/init/init.js +435 -0
  131. package/lib/{os.js → commands/init/os.js} +4 -4
  132. package/lib/{list-stacks.d.ts → commands/list-stacks.d.ts} +1 -1
  133. package/lib/{list-stacks.js → commands/list-stacks.js} +2 -2
  134. package/lib/commands/migrate.js +29 -32
  135. package/lib/context-providers/ami.d.ts +3 -1
  136. package/lib/context-providers/ami.js +8 -8
  137. package/lib/context-providers/availability-zones.d.ts +3 -1
  138. package/lib/context-providers/availability-zones.js +4 -4
  139. package/lib/context-providers/cc-api-provider.d.ts +8 -12
  140. package/lib/context-providers/cc-api-provider.js +94 -66
  141. package/lib/context-providers/endpoint-service-availability-zones.d.ts +3 -1
  142. package/lib/context-providers/endpoint-service-availability-zones.js +6 -6
  143. package/lib/context-providers/hosted-zones.d.ts +3 -1
  144. package/lib/context-providers/hosted-zones.js +11 -11
  145. package/lib/context-providers/index.d.ts +19 -5
  146. package/lib/context-providers/index.js +35 -17
  147. package/lib/context-providers/keys.d.ts +3 -1
  148. package/lib/context-providers/keys.js +8 -8
  149. package/lib/context-providers/load-balancers.js +15 -18
  150. package/lib/context-providers/security-groups.js +10 -12
  151. package/lib/context-providers/ssm-parameters.d.ts +3 -1
  152. package/lib/context-providers/ssm-parameters.js +7 -7
  153. package/lib/context-providers/vpcs.d.ts +3 -1
  154. package/lib/context-providers/vpcs.js +14 -15
  155. package/lib/index.js +124098 -123198
  156. package/lib/init-templates/.init-version.json +1 -1
  157. package/lib/init-templates/.recommended-feature-flags.json +3 -1
  158. package/lib/legacy-exports-source.d.ts +4 -5
  159. package/lib/legacy-exports-source.js +6 -7
  160. package/lib/logging.js +2 -2
  161. package/lib/notices.d.ts +1 -1
  162. package/lib/notices.js +26 -32
  163. package/package.json +29 -29
  164. package/lib/api/deployments/cloudformation.js +0 -597
  165. package/lib/api/deployments/nested-stack-helpers.js +0 -88
  166. package/lib/api/util/placeholders.js +0 -24
  167. package/lib/api/util/template-body-parameter.js +0 -103
  168. package/lib/diff.d.ts +0 -28
  169. package/lib/diff.js +0 -165
  170. package/lib/init-hooks.js +0 -63
  171. package/lib/init.js +0 -437
  172. package/lib/toolkit/cli-io-host.js +0 -353
  173. package/lib/toolkit/error.d.ts +0 -1
  174. package/lib/tree.js +0 -40
  175. /package/lib/api/{util → environment}/placeholders.d.ts +0 -0
  176. /package/lib/{init-hooks.d.ts → commands/init/init-hooks.d.ts} +0 -0
  177. /package/lib/{init.d.ts → commands/init/init.d.ts} +0 -0
  178. /package/lib/{os.d.ts → commands/init/os.d.ts} +0 -0
@@ -1,31 +1,8 @@
1
1
  import type { PropertyDifference } from '@aws-cdk/cloudformation-diff';
2
- import type { CloudFormationStackArtifact } from '@aws-cdk/cx-api';
3
- import type { HotswappableChange, ResourceChange } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads/hotswap';
2
+ import type { HotswappableChange, NonHotswappableChange, ResourceChange } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads/hotswap';
3
+ import { NonHotswappableReason } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads/hotswap';
4
4
  import type { SDK } from '../aws-auth';
5
5
  export declare const ICON = "\u2728";
6
- /**
7
- * The result of an attempted hotswap deployment
8
- */
9
- export interface HotswapResult {
10
- /**
11
- * The stack that was hotswapped
12
- */
13
- readonly stack: CloudFormationStackArtifact;
14
- /**
15
- * Whether hotswapping happened or not.
16
- *
17
- * `false` indicates that the deployment could not be hotswapped and full deployment may be attempted as fallback.
18
- */
19
- readonly hotswapped: boolean;
20
- /**
21
- * The changes that were deemed hotswappable
22
- */
23
- readonly hotswappableChanges: any[];
24
- /**
25
- * The changes that were deemed not hotswappable
26
- */
27
- readonly nonHotswappableChanges: any[];
28
- }
29
6
  export interface HotswapOperation {
30
7
  /**
31
8
  * Marks the operation as hotswappable
@@ -40,25 +17,20 @@ export interface HotswapOperation {
40
17
  * Description of the change that is applied as part of the operation
41
18
  */
42
19
  readonly change: HotswappableChange;
43
- /**
44
- * The names of the resources being hotswapped.
45
- */
46
- readonly resourceNames: string[];
47
20
  /**
48
21
  * Applies the hotswap operation
49
22
  */
50
23
  readonly apply: (sdk: SDK) => Promise<void>;
51
24
  }
52
- export interface NonHotswappableChange {
25
+ export interface RejectedChange {
26
+ /**
27
+ * Marks the change as not hotswappable
28
+ */
53
29
  readonly hotswappable: false;
54
- readonly resourceType: string;
55
- readonly rejectedChanges: Array<string>;
56
- readonly logicalId: string;
57
30
  /**
58
- * Tells the user exactly why this change was deemed non-hotswappable and what its logical ID is.
59
- * If not specified, `reason` will be autofilled to state that the properties listed in `rejectedChanges` are not hotswappable.
31
+ * The change that got rejected
60
32
  */
61
- readonly reason?: string;
33
+ readonly change: NonHotswappableChange;
62
34
  /**
63
35
  * Whether or not to show this change when listing non-hotswappable changes in HOTSWAP_ONLY mode. Does not affect
64
36
  * listing in FALL_BACK mode.
@@ -67,11 +39,7 @@ export interface NonHotswappableChange {
67
39
  */
68
40
  readonly hotswapOnlyVisible?: boolean;
69
41
  }
70
- export type ChangeHotswapResult = Array<HotswapOperation | NonHotswappableChange>;
71
- export interface ClassifiedResourceChanges {
72
- hotswappableChanges: HotswapOperation[];
73
- nonHotswappableChanges: NonHotswappableChange[];
74
- }
42
+ export type HotswapChange = HotswapOperation | RejectedChange;
75
43
  export declare enum HotswapMode {
76
44
  /**
77
45
  * Will fall back to CloudFormation when a non-hotswappable change is detected
@@ -86,9 +54,6 @@ export declare enum HotswapMode {
86
54
  */
87
55
  FULL_DEPLOYMENT = "full-deployment"
88
56
  }
89
- type Exclude = {
90
- [key: string]: Exclude | true;
91
- };
92
57
  /**
93
58
  * Represents configuration property overrides for hotswap deployments
94
59
  */
@@ -109,29 +74,16 @@ export declare class EcsHotswapProperties {
109
74
  */
110
75
  isEmpty(): boolean;
111
76
  }
112
- /**
113
- * This function transforms all keys (recursively) in the provided `val` object.
114
- *
115
- * @param val The object whose keys need to be transformed.
116
- * @param transform The function that will be applied to each key.
117
- * @param exclude The keys that will not be transformed and copied to output directly
118
- * @returns A new object with the same values as `val`, but with all keys transformed according to `transform`.
119
- */
120
- export declare function transformObjectKeys(val: any, transform: (str: string) => string, exclude?: Exclude): any;
121
- /**
122
- * This function lower cases the first character of the string provided.
123
- */
124
- export declare function lowerCaseFirstCharacter(str: string): string;
125
77
  type PropDiffs = Record<string, PropertyDifference<any>>;
126
- export declare class ClassifiedChanges {
78
+ declare class ClassifiedChanges {
127
79
  readonly change: ResourceChange;
128
80
  readonly hotswappableProps: PropDiffs;
129
81
  readonly nonHotswappableProps: PropDiffs;
130
82
  constructor(change: ResourceChange, hotswappableProps: PropDiffs, nonHotswappableProps: PropDiffs);
131
- reportNonHotswappablePropertyChanges(ret: ChangeHotswapResult): void;
83
+ reportNonHotswappablePropertyChanges(ret: HotswapChange[]): void;
132
84
  get namesOfHotswappableProps(): string[];
133
85
  }
134
86
  export declare function classifyChanges(xs: ResourceChange, hotswappablePropNames: string[]): ClassifiedChanges;
135
- export declare function reportNonHotswappableChange(ret: ChangeHotswapResult, change: ResourceChange, nonHotswappableProps?: PropDiffs, reason?: string, hotswapOnlyVisible?: boolean): void;
136
- export declare function reportNonHotswappableResource(change: ResourceChange, reason?: string): ChangeHotswapResult;
87
+ export declare function nonHotswappableChange(change: ResourceChange, reason: NonHotswappableReason, description: string, nonHotswappableProps?: PropDiffs, hotswapOnlyVisible?: boolean): RejectedChange;
88
+ export declare function nonHotswappableResource(change: ResourceChange): RejectedChange;
137
89
  export {};
@@ -1,12 +1,11 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.ClassifiedChanges = exports.EcsHotswapProperties = exports.HotswapPropertyOverrides = exports.HotswapMode = exports.ICON = void 0;
4
- exports.transformObjectKeys = transformObjectKeys;
5
- exports.lowerCaseFirstCharacter = lowerCaseFirstCharacter;
3
+ exports.EcsHotswapProperties = exports.HotswapPropertyOverrides = exports.HotswapMode = exports.ICON = void 0;
6
4
  exports.classifyChanges = classifyChanges;
7
- exports.reportNonHotswappableChange = reportNonHotswappableChange;
8
- exports.reportNonHotswappableResource = reportNonHotswappableResource;
9
- const error_1 = require("../../toolkit/error");
5
+ exports.nonHotswappableChange = nonHotswappableChange;
6
+ exports.nonHotswappableResource = nonHotswappableResource;
7
+ const api_1 = require("../../../../@aws-cdk/tmp-toolkit-helpers/src/api");
8
+ const hotswap_1 = require("../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads/hotswap");
10
9
  exports.ICON = '✨';
11
10
  var HotswapMode;
12
11
  (function (HotswapMode) {
@@ -38,10 +37,10 @@ exports.HotswapPropertyOverrides = HotswapPropertyOverrides;
38
37
  class EcsHotswapProperties {
39
38
  constructor(minimumHealthyPercent, maximumHealthyPercent) {
40
39
  if (minimumHealthyPercent !== undefined && minimumHealthyPercent < 0) {
41
- throw new error_1.ToolkitError('hotswap-ecs-minimum-healthy-percent can\'t be a negative number');
40
+ throw new api_1.ToolkitError('hotswap-ecs-minimum-healthy-percent can\'t be a negative number');
42
41
  }
43
42
  if (maximumHealthyPercent !== undefined && maximumHealthyPercent < 0) {
44
- throw new error_1.ToolkitError('hotswap-ecs-maximum-healthy-percent can\'t be a negative number');
43
+ throw new api_1.ToolkitError('hotswap-ecs-maximum-healthy-percent can\'t be a negative number');
45
44
  }
46
45
  // In order to preserve the current behaviour, when minimumHealthyPercent is not defined, it will be set to the currently default value of 0
47
46
  if (minimumHealthyPercent == undefined) {
@@ -61,42 +60,6 @@ class EcsHotswapProperties {
61
60
  }
62
61
  }
63
62
  exports.EcsHotswapProperties = EcsHotswapProperties;
64
- /**
65
- * This function transforms all keys (recursively) in the provided `val` object.
66
- *
67
- * @param val The object whose keys need to be transformed.
68
- * @param transform The function that will be applied to each key.
69
- * @param exclude The keys that will not be transformed and copied to output directly
70
- * @returns A new object with the same values as `val`, but with all keys transformed according to `transform`.
71
- */
72
- function transformObjectKeys(val, transform, exclude = {}) {
73
- if (val == null || typeof val !== 'object') {
74
- return val;
75
- }
76
- if (Array.isArray(val)) {
77
- // For arrays we just pass parent's exclude object directly
78
- // since it makes no sense to specify different exclude options for each array element
79
- return val.map((input) => transformObjectKeys(input, transform, exclude));
80
- }
81
- const ret = {};
82
- for (const [k, v] of Object.entries(val)) {
83
- const childExclude = exclude[k];
84
- if (childExclude === true) {
85
- // we don't transform this object if the key is specified in exclude
86
- ret[transform(k)] = v;
87
- }
88
- else {
89
- ret[transform(k)] = transformObjectKeys(v, transform, childExclude);
90
- }
91
- }
92
- return ret;
93
- }
94
- /**
95
- * This function lower cases the first character of the string provided.
96
- */
97
- function lowerCaseFirstCharacter(str) {
98
- return str.length > 0 ? `${str[0].toLowerCase()}${str.slice(1)}` : str;
99
- }
100
63
  class ClassifiedChanges {
101
64
  constructor(change, hotswappableProps, nonHotswappableProps) {
102
65
  this.change = change;
@@ -107,16 +70,15 @@ class ClassifiedChanges {
107
70
  const nonHotswappablePropNames = Object.keys(this.nonHotswappableProps);
108
71
  if (nonHotswappablePropNames.length > 0) {
109
72
  const tagOnlyChange = nonHotswappablePropNames.length === 1 && nonHotswappablePropNames[0] === 'Tags';
110
- reportNonHotswappableChange(ret, this.change, this.nonHotswappableProps, tagOnlyChange
111
- ? 'Tags are not hotswappable'
112
- : `resource properties '${nonHotswappablePropNames}' are not hotswappable on this resource type`);
73
+ const reason = tagOnlyChange ? hotswap_1.NonHotswappableReason.TAGS : hotswap_1.NonHotswappableReason.PROPERTIES;
74
+ const description = tagOnlyChange ? 'Tags are not hotswappable' : `resource properties '${nonHotswappablePropNames}' are not hotswappable on this resource type`;
75
+ ret.push(nonHotswappableChange(this.change, reason, description, this.nonHotswappableProps));
113
76
  }
114
77
  }
115
78
  get namesOfHotswappableProps() {
116
79
  return Object.keys(this.hotswappableProps);
117
80
  }
118
81
  }
119
- exports.ClassifiedChanges = ClassifiedChanges;
120
82
  function classifyChanges(xs, hotswappablePropNames) {
121
83
  const hotswappableProps = {};
122
84
  const nonHotswappableProps = {};
@@ -130,29 +92,37 @@ function classifyChanges(xs, hotswappablePropNames) {
130
92
  }
131
93
  return new ClassifiedChanges(xs, hotswappableProps, nonHotswappableProps);
132
94
  }
133
- function reportNonHotswappableChange(ret, change, nonHotswappableProps, reason, hotswapOnlyVisible) {
134
- let hotswapOnlyVisibility = true;
135
- if (hotswapOnlyVisible === false) {
136
- hotswapOnlyVisibility = false;
137
- }
138
- ret.push({
95
+ function nonHotswappableChange(change, reason, description, nonHotswappableProps, hotswapOnlyVisible = true) {
96
+ return {
139
97
  hotswappable: false,
140
- rejectedChanges: Object.keys(nonHotswappableProps !== null && nonHotswappableProps !== void 0 ? nonHotswappableProps : change.propertyUpdates),
141
- logicalId: change.logicalId,
142
- resourceType: change.newValue.Type,
143
- reason,
144
- hotswapOnlyVisible: hotswapOnlyVisibility,
145
- });
146
- }
147
- function reportNonHotswappableResource(change, reason) {
148
- return [
149
- {
150
- hotswappable: false,
151
- rejectedChanges: Object.keys(change.propertyUpdates),
152
- logicalId: change.logicalId,
153
- resourceType: change.newValue.Type,
98
+ hotswapOnlyVisible,
99
+ change: {
154
100
  reason,
101
+ description,
102
+ subject: {
103
+ type: 'Resource',
104
+ logicalId: change.logicalId,
105
+ resourceType: change.newValue.Type,
106
+ rejectedProperties: Object.keys(nonHotswappableProps ?? change.propertyUpdates),
107
+ metadata: change.metadata,
108
+ },
109
+ },
110
+ };
111
+ }
112
+ function nonHotswappableResource(change) {
113
+ return {
114
+ hotswappable: false,
115
+ change: {
116
+ reason: hotswap_1.NonHotswappableReason.RESOURCE_UNSUPPORTED,
117
+ description: 'This resource type is not supported for hotswap deployments',
118
+ subject: {
119
+ type: 'Resource',
120
+ logicalId: change.logicalId,
121
+ resourceType: change.newValue.Type,
122
+ rejectedProperties: Object.keys(change.propertyUpdates),
123
+ metadata: change.metadata,
124
+ },
155
125
  },
156
- ];
126
+ };
157
127
  }
158
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"common.js","sourceRoot":"","sources":["common.ts"],"names":[],"mappings":"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","sourcesContent":["import type { PropertyDifference } from '@aws-cdk/cloudformation-diff';\nimport type { CloudFormationStackArtifact } from '@aws-cdk/cx-api';\nimport type { HotswappableChange, ResourceChange } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads/hotswap';\nimport { ToolkitError } from '../../toolkit/error';\nimport type { SDK } from '../aws-auth';\n\nexport const ICON = '✨';\n\n/**\n * The result of an attempted hotswap deployment\n */\nexport interface HotswapResult {\n  /**\n   * The stack that was hotswapped\n   */\n  readonly stack: CloudFormationStackArtifact;\n  /**\n   * Whether hotswapping happened or not.\n   *\n   * `false` indicates that the deployment could not be hotswapped and full deployment may be attempted as fallback.\n   */\n  readonly hotswapped: boolean;\n  /**\n   * The changes that were deemed hotswappable\n   */\n  readonly hotswappableChanges: any[];\n  /**\n   * The changes that were deemed not hotswappable\n   */\n  readonly nonHotswappableChanges: any[];\n}\n\nexport interface HotswapOperation {\n  /**\n   * Marks the operation as hotswappable\n   */\n  readonly hotswappable: true;\n\n  /**\n   * The name of the service being hotswapped.\n   * Used to set a custom User-Agent for SDK calls.\n   */\n  readonly service: string;\n\n  /**\n   * Description of the change that is applied as part of the operation\n   */\n  readonly change: HotswappableChange;\n\n  /**\n   * The names of the resources being hotswapped.\n   */\n  readonly resourceNames: string[];\n\n  /**\n   * Applies the hotswap operation\n   */\n  readonly apply: (sdk: SDK) => Promise<void>;\n}\n\nexport interface NonHotswappableChange {\n  readonly hotswappable: false;\n  readonly resourceType: string;\n  readonly rejectedChanges: Array<string>;\n  readonly logicalId: string;\n  /**\n   * Tells the user exactly why this change was deemed non-hotswappable and what its logical ID is.\n   * If not specified, `reason` will be autofilled to state that the properties listed in `rejectedChanges` are not hotswappable.\n   */\n  readonly reason?: string;\n  /**\n   * Whether or not to show this change when listing non-hotswappable changes in HOTSWAP_ONLY mode. Does not affect\n   * listing in FALL_BACK mode.\n   *\n   * @default true\n   */\n  readonly hotswapOnlyVisible?: boolean;\n}\n\nexport type ChangeHotswapResult = Array<HotswapOperation | NonHotswappableChange>;\n\nexport interface ClassifiedResourceChanges {\n  hotswappableChanges: HotswapOperation[];\n  nonHotswappableChanges: NonHotswappableChange[];\n}\n\nexport enum HotswapMode {\n  /**\n   * Will fall back to CloudFormation when a non-hotswappable change is detected\n   */\n  FALL_BACK = 'fall-back',\n\n  /**\n   * Will not fall back to CloudFormation when a non-hotswappable change is detected\n   */\n  HOTSWAP_ONLY = 'hotswap-only',\n\n  /**\n   * Will not attempt to hotswap anything and instead go straight to CloudFormation\n   */\n  FULL_DEPLOYMENT = 'full-deployment',\n}\n\ntype Exclude = { [key: string]: Exclude | true };\n\n/**\n * Represents configuration property overrides for hotswap deployments\n */\nexport class HotswapPropertyOverrides {\n  // Each supported resource type will have its own properties. Currently this is ECS\n  ecsHotswapProperties?: EcsHotswapProperties;\n\n  public constructor (ecsHotswapProperties?: EcsHotswapProperties) {\n    this.ecsHotswapProperties = ecsHotswapProperties;\n  }\n}\n\n/**\n * Represents configuration properties for ECS hotswap deployments\n */\nexport class EcsHotswapProperties {\n  // The lower limit on the number of your service's tasks that must remain in the RUNNING state during a deployment, as a percentage of the desiredCount\n  readonly minimumHealthyPercent?: number;\n  // The upper limit on the number of your service's tasks that are allowed in the RUNNING or PENDING state during a deployment, as a percentage of the desiredCount\n  readonly maximumHealthyPercent?: number;\n\n  public constructor (minimumHealthyPercent?: number, maximumHealthyPercent?: number) {\n    if (minimumHealthyPercent !== undefined && minimumHealthyPercent < 0 ) {\n      throw new ToolkitError('hotswap-ecs-minimum-healthy-percent can\\'t be a negative number');\n    }\n    if (maximumHealthyPercent !== undefined && maximumHealthyPercent < 0 ) {\n      throw new ToolkitError('hotswap-ecs-maximum-healthy-percent can\\'t be a negative number');\n    }\n    // In order to preserve the current behaviour, when minimumHealthyPercent is not defined, it will be set to the currently default value of 0\n    if (minimumHealthyPercent == undefined) {\n      this.minimumHealthyPercent = 0;\n    } else {\n      this.minimumHealthyPercent = minimumHealthyPercent;\n    }\n    this.maximumHealthyPercent = maximumHealthyPercent;\n  }\n\n  /**\n   * Check if any hotswap properties are defined\n   * @returns true if all properties are undefined, false otherwise\n   */\n  public isEmpty(): boolean {\n    return this.minimumHealthyPercent === 0 && this.maximumHealthyPercent === undefined;\n  }\n}\n\n/**\n * This function transforms all keys (recursively) in the provided `val` object.\n *\n * @param val The object whose keys need to be transformed.\n * @param transform The function that will be applied to each key.\n * @param exclude The keys that will not be transformed and copied to output directly\n * @returns A new object with the same values as `val`, but with all keys transformed according to `transform`.\n */\nexport function transformObjectKeys(val: any, transform: (str: string) => string, exclude: Exclude = {}): any {\n  if (val == null || typeof val !== 'object') {\n    return val;\n  }\n  if (Array.isArray(val)) {\n    // For arrays we just pass parent's exclude object directly\n    // since it makes no sense to specify different exclude options for each array element\n    return val.map((input: any) => transformObjectKeys(input, transform, exclude));\n  }\n  const ret: { [k: string]: any } = {};\n  for (const [k, v] of Object.entries(val)) {\n    const childExclude = exclude[k];\n    if (childExclude === true) {\n      // we don't transform this object if the key is specified in exclude\n      ret[transform(k)] = v;\n    } else {\n      ret[transform(k)] = transformObjectKeys(v, transform, childExclude);\n    }\n  }\n  return ret;\n}\n\n/**\n * This function lower cases the first character of the string provided.\n */\nexport function lowerCaseFirstCharacter(str: string): string {\n  return str.length > 0 ? `${str[0].toLowerCase()}${str.slice(1)}` : str;\n}\n\ntype PropDiffs = Record<string, PropertyDifference<any>>;\n\nexport class ClassifiedChanges {\n  public constructor(\n    public readonly change: ResourceChange,\n    public readonly hotswappableProps: PropDiffs,\n    public readonly nonHotswappableProps: PropDiffs,\n  ) {\n  }\n\n  public reportNonHotswappablePropertyChanges(ret: ChangeHotswapResult): void {\n    const nonHotswappablePropNames = Object.keys(this.nonHotswappableProps);\n    if (nonHotswappablePropNames.length > 0) {\n      const tagOnlyChange = nonHotswappablePropNames.length === 1 && nonHotswappablePropNames[0] === 'Tags';\n      reportNonHotswappableChange(\n        ret,\n        this.change,\n        this.nonHotswappableProps,\n        tagOnlyChange\n          ? 'Tags are not hotswappable'\n          : `resource properties '${nonHotswappablePropNames}' are not hotswappable on this resource type`,\n      );\n    }\n  }\n\n  public get namesOfHotswappableProps(): string[] {\n    return Object.keys(this.hotswappableProps);\n  }\n}\n\nexport function classifyChanges(xs: ResourceChange, hotswappablePropNames: string[]): ClassifiedChanges {\n  const hotswappableProps: PropDiffs = {};\n  const nonHotswappableProps: PropDiffs = {};\n\n  for (const [name, propDiff] of Object.entries(xs.propertyUpdates)) {\n    if (hotswappablePropNames.includes(name)) {\n      hotswappableProps[name] = propDiff;\n    } else {\n      nonHotswappableProps[name] = propDiff;\n    }\n  }\n\n  return new ClassifiedChanges(xs, hotswappableProps, nonHotswappableProps);\n}\n\nexport function reportNonHotswappableChange(\n  ret: ChangeHotswapResult,\n  change: ResourceChange,\n  nonHotswappableProps?: PropDiffs,\n  reason?: string,\n  hotswapOnlyVisible?: boolean,\n): void {\n  let hotswapOnlyVisibility = true;\n  if (hotswapOnlyVisible === false) {\n    hotswapOnlyVisibility = false;\n  }\n  ret.push({\n    hotswappable: false,\n    rejectedChanges: Object.keys(nonHotswappableProps ?? change.propertyUpdates),\n    logicalId: change.logicalId,\n    resourceType: change.newValue.Type,\n    reason,\n    hotswapOnlyVisible: hotswapOnlyVisibility,\n  });\n}\n\nexport function reportNonHotswappableResource(\n  change: ResourceChange,\n  reason?: string,\n): ChangeHotswapResult {\n  return [\n    {\n      hotswappable: false,\n      rejectedChanges: Object.keys(change.propertyUpdates),\n      logicalId: change.logicalId,\n      resourceType: change.newValue.Type,\n      reason,\n    },\n  ];\n}\n"]}
128
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"common.js","sourceRoot":"","sources":["common.ts"],"names":[],"mappings":";;;AAiJA,0CAaC;AAED,sDAsBC;AAED,0DAeC;AAtMD,0EAAgF;AAEhF,kGAA6G;AAGhG,QAAA,IAAI,GAAG,GAAG,CAAC;AA6CxB,IAAY,WAeX;AAfD,WAAY,WAAW;IACrB;;OAEG;IACH,sCAAuB,CAAA;IAEvB;;OAEG;IACH,4CAA6B,CAAA;IAE7B;;OAEG;IACH,kDAAmC,CAAA;AACrC,CAAC,EAfW,WAAW,2BAAX,WAAW,QAetB;AAED;;GAEG;AACH,MAAa,wBAAwB;IAInC,YAAoB,oBAA2C;QAC7D,IAAI,CAAC,oBAAoB,GAAG,oBAAoB,CAAC;IACnD,CAAC;CACF;AAPD,4DAOC;AAED;;GAEG;AACH,MAAa,oBAAoB;IAM/B,YAAoB,qBAA8B,EAAE,qBAA8B;QAChF,IAAI,qBAAqB,KAAK,SAAS,IAAI,qBAAqB,GAAG,CAAC,EAAG,CAAC;YACtE,MAAM,IAAI,kBAAY,CAAC,iEAAiE,CAAC,CAAC;QAC5F,CAAC;QACD,IAAI,qBAAqB,KAAK,SAAS,IAAI,qBAAqB,GAAG,CAAC,EAAG,CAAC;YACtE,MAAM,IAAI,kBAAY,CAAC,iEAAiE,CAAC,CAAC;QAC5F,CAAC;QACD,4IAA4I;QAC5I,IAAI,qBAAqB,IAAI,SAAS,EAAE,CAAC;YACvC,IAAI,CAAC,qBAAqB,GAAG,CAAC,CAAC;QACjC,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,qBAAqB,GAAG,qBAAqB,CAAC;QACrD,CAAC;QACD,IAAI,CAAC,qBAAqB,GAAG,qBAAqB,CAAC;IACrD,CAAC;IAED;;;OAGG;IACI,OAAO;QACZ,OAAO,IAAI,CAAC,qBAAqB,KAAK,CAAC,IAAI,IAAI,CAAC,qBAAqB,KAAK,SAAS,CAAC;IACtF,CAAC;CACF;AA7BD,oDA6BC;AAID,MAAM,iBAAiB;IACrB,YACkB,MAAsB,EACtB,iBAA4B,EAC5B,oBAA+B;QAF/B,WAAM,GAAN,MAAM,CAAgB;QACtB,sBAAiB,GAAjB,iBAAiB,CAAW;QAC5B,yBAAoB,GAApB,oBAAoB,CAAW;IAEjD,CAAC;IAEM,oCAAoC,CAAC,GAAoB;QAC9D,MAAM,wBAAwB,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC;QACxE,IAAI,wBAAwB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACxC,MAAM,aAAa,GAAG,wBAAwB,CAAC,MAAM,KAAK,CAAC,IAAI,wBAAwB,CAAC,CAAC,CAAC,KAAK,MAAM,CAAC;YACtG,MAAM,MAAM,GAAG,aAAa,CAAC,CAAC,CAAC,+BAAqB,CAAC,IAAI,CAAC,CAAC,CAAC,+BAAqB,CAAC,UAAU,CAAC;YAC7F,MAAM,WAAW,GAAG,aAAa,CAAC,CAAC,CAAC,2BAA2B,CAAC,CAAC,CAAC,wBAAwB,wBAAwB,8CAA8C,CAAC;YAEjK,GAAG,CAAC,IAAI,CAAC,qBAAqB,CAC5B,IAAI,CAAC,MAAM,EACX,MAAM,EACN,WAAW,EACX,IAAI,CAAC,oBAAoB,CAC1B,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,IAAW,wBAAwB;QACjC,OAAO,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;IAC7C,CAAC;CACF;AAED,SAAgB,eAAe,CAAC,EAAkB,EAAE,qBAA+B;IACjF,MAAM,iBAAiB,GAAc,EAAE,CAAC;IACxC,MAAM,oBAAoB,GAAc,EAAE,CAAC;IAE3C,KAAK,MAAM,CAAC,IAAI,EAAE,QAAQ,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,eAAe,CAAC,EAAE,CAAC;QAClE,IAAI,qBAAqB,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;YACzC,iBAAiB,CAAC,IAAI,CAAC,GAAG,QAAQ,CAAC;QACrC,CAAC;aAAM,CAAC;YACN,oBAAoB,CAAC,IAAI,CAAC,GAAG,QAAQ,CAAC;QACxC,CAAC;IACH,CAAC;IAED,OAAO,IAAI,iBAAiB,CAAC,EAAE,EAAE,iBAAiB,EAAE,oBAAoB,CAAC,CAAC;AAC5E,CAAC;AAED,SAAgB,qBAAqB,CACnC,MAAsB,EACtB,MAA6B,EAC7B,WAAmB,EACnB,oBAAgC,EAChC,qBAA8B,IAAI;IAElC,OAAO;QACL,YAAY,EAAE,KAAK;QACnB,kBAAkB;QAClB,MAAM,EAAE;YACN,MAAM;YACN,WAAW;YACX,OAAO,EAAE;gBACP,IAAI,EAAE,UAAU;gBAChB,SAAS,EAAE,MAAM,CAAC,SAAS;gBAC3B,YAAY,EAAE,MAAM,CAAC,QAAQ,CAAC,IAAI;gBAClC,kBAAkB,EAAE,MAAM,CAAC,IAAI,CAAC,oBAAoB,IAAI,MAAM,CAAC,eAAe,CAAC;gBAC/E,QAAQ,EAAE,MAAM,CAAC,QAAQ;aAC1B;SACF;KACF,CAAC;AACJ,CAAC;AAED,SAAgB,uBAAuB,CAAC,MAAsB;IAC5D,OAAO;QACL,YAAY,EAAE,KAAK;QACnB,MAAM,EAAE;YACN,MAAM,EAAE,+BAAqB,CAAC,oBAAoB;YAClD,WAAW,EAAE,6DAA6D;YAC1E,OAAO,EAAE;gBACP,IAAI,EAAE,UAAU;gBAChB,SAAS,EAAE,MAAM,CAAC,SAAS;gBAC3B,YAAY,EAAE,MAAM,CAAC,QAAQ,CAAC,IAAI;gBAClC,kBAAkB,EAAE,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,eAAe,CAAC;gBACvD,QAAQ,EAAE,MAAM,CAAC,QAAQ;aAC1B;SACF;KACF,CAAC;AACJ,CAAC","sourcesContent":["import type { PropertyDifference } from '@aws-cdk/cloudformation-diff';\nimport { ToolkitError } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api';\nimport type { HotswappableChange, NonHotswappableChange, ResourceChange } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads/hotswap';\nimport { NonHotswappableReason } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads/hotswap';\nimport type { SDK } from '../aws-auth';\n\nexport const ICON = '✨';\n\nexport interface HotswapOperation {\n  /**\n   * Marks the operation as hotswappable\n   */\n  readonly hotswappable: true;\n\n  /**\n   * The name of the service being hotswapped.\n   * Used to set a custom User-Agent for SDK calls.\n   */\n  readonly service: string;\n\n  /**\n   * Description of the change that is applied as part of the operation\n   */\n  readonly change: HotswappableChange;\n\n  /**\n   * Applies the hotswap operation\n   */\n  readonly apply: (sdk: SDK) => Promise<void>;\n}\n\nexport interface RejectedChange {\n  /**\n   * Marks the change as not hotswappable\n   */\n  readonly hotswappable: false;\n  /**\n   * The change that got rejected\n   */\n  readonly change: NonHotswappableChange;\n  /**\n   * Whether or not to show this change when listing non-hotswappable changes in HOTSWAP_ONLY mode. Does not affect\n   * listing in FALL_BACK mode.\n   *\n   * @default true\n   */\n  readonly hotswapOnlyVisible?: boolean;\n}\n\nexport type HotswapChange = HotswapOperation | RejectedChange;\n\nexport enum HotswapMode {\n  /**\n   * Will fall back to CloudFormation when a non-hotswappable change is detected\n   */\n  FALL_BACK = 'fall-back',\n\n  /**\n   * Will not fall back to CloudFormation when a non-hotswappable change is detected\n   */\n  HOTSWAP_ONLY = 'hotswap-only',\n\n  /**\n   * Will not attempt to hotswap anything and instead go straight to CloudFormation\n   */\n  FULL_DEPLOYMENT = 'full-deployment',\n}\n\n/**\n * Represents configuration property overrides for hotswap deployments\n */\nexport class HotswapPropertyOverrides {\n  // Each supported resource type will have its own properties. Currently this is ECS\n  ecsHotswapProperties?: EcsHotswapProperties;\n\n  public constructor (ecsHotswapProperties?: EcsHotswapProperties) {\n    this.ecsHotswapProperties = ecsHotswapProperties;\n  }\n}\n\n/**\n * Represents configuration properties for ECS hotswap deployments\n */\nexport class EcsHotswapProperties {\n  // The lower limit on the number of your service's tasks that must remain in the RUNNING state during a deployment, as a percentage of the desiredCount\n  readonly minimumHealthyPercent?: number;\n  // The upper limit on the number of your service's tasks that are allowed in the RUNNING or PENDING state during a deployment, as a percentage of the desiredCount\n  readonly maximumHealthyPercent?: number;\n\n  public constructor (minimumHealthyPercent?: number, maximumHealthyPercent?: number) {\n    if (minimumHealthyPercent !== undefined && minimumHealthyPercent < 0 ) {\n      throw new ToolkitError('hotswap-ecs-minimum-healthy-percent can\\'t be a negative number');\n    }\n    if (maximumHealthyPercent !== undefined && maximumHealthyPercent < 0 ) {\n      throw new ToolkitError('hotswap-ecs-maximum-healthy-percent can\\'t be a negative number');\n    }\n    // In order to preserve the current behaviour, when minimumHealthyPercent is not defined, it will be set to the currently default value of 0\n    if (minimumHealthyPercent == undefined) {\n      this.minimumHealthyPercent = 0;\n    } else {\n      this.minimumHealthyPercent = minimumHealthyPercent;\n    }\n    this.maximumHealthyPercent = maximumHealthyPercent;\n  }\n\n  /**\n   * Check if any hotswap properties are defined\n   * @returns true if all properties are undefined, false otherwise\n   */\n  public isEmpty(): boolean {\n    return this.minimumHealthyPercent === 0 && this.maximumHealthyPercent === undefined;\n  }\n}\n\ntype PropDiffs = Record<string, PropertyDifference<any>>;\n\nclass ClassifiedChanges {\n  public constructor(\n    public readonly change: ResourceChange,\n    public readonly hotswappableProps: PropDiffs,\n    public readonly nonHotswappableProps: PropDiffs,\n  ) {\n  }\n\n  public reportNonHotswappablePropertyChanges(ret: HotswapChange[]): void {\n    const nonHotswappablePropNames = Object.keys(this.nonHotswappableProps);\n    if (nonHotswappablePropNames.length > 0) {\n      const tagOnlyChange = nonHotswappablePropNames.length === 1 && nonHotswappablePropNames[0] === 'Tags';\n      const reason = tagOnlyChange ? NonHotswappableReason.TAGS : NonHotswappableReason.PROPERTIES;\n      const description = tagOnlyChange ? 'Tags are not hotswappable' : `resource properties '${nonHotswappablePropNames}' are not hotswappable on this resource type`;\n\n      ret.push(nonHotswappableChange(\n        this.change,\n        reason,\n        description,\n        this.nonHotswappableProps,\n      ));\n    }\n  }\n\n  public get namesOfHotswappableProps(): string[] {\n    return Object.keys(this.hotswappableProps);\n  }\n}\n\nexport function classifyChanges(xs: ResourceChange, hotswappablePropNames: string[]): ClassifiedChanges {\n  const hotswappableProps: PropDiffs = {};\n  const nonHotswappableProps: PropDiffs = {};\n\n  for (const [name, propDiff] of Object.entries(xs.propertyUpdates)) {\n    if (hotswappablePropNames.includes(name)) {\n      hotswappableProps[name] = propDiff;\n    } else {\n      nonHotswappableProps[name] = propDiff;\n    }\n  }\n\n  return new ClassifiedChanges(xs, hotswappableProps, nonHotswappableProps);\n}\n\nexport function nonHotswappableChange(\n  change: ResourceChange,\n  reason: NonHotswappableReason,\n  description: string,\n  nonHotswappableProps?: PropDiffs,\n  hotswapOnlyVisible: boolean = true,\n): RejectedChange {\n  return {\n    hotswappable: false,\n    hotswapOnlyVisible,\n    change: {\n      reason,\n      description,\n      subject: {\n        type: 'Resource',\n        logicalId: change.logicalId,\n        resourceType: change.newValue.Type,\n        rejectedProperties: Object.keys(nonHotswappableProps ?? change.propertyUpdates),\n        metadata: change.metadata,\n      },\n    },\n  };\n}\n\nexport function nonHotswappableResource(change: ResourceChange): RejectedChange {\n  return {\n    hotswappable: false,\n    change: {\n      reason: NonHotswappableReason.RESOURCE_UNSUPPORTED,\n      description: 'This resource type is not supported for hotswap deployments',\n      subject: {\n        type: 'Resource',\n        logicalId: change.logicalId,\n        resourceType: change.newValue.Type,\n        rejectedProperties: Object.keys(change.propertyUpdates),\n        metadata: change.metadata,\n      },\n    },\n  };\n}\n"]}
@@ -1,4 +1,4 @@
1
- import type { HotswapPropertyOverrides, ChangeHotswapResult } from './common';
2
- import type { ResourceChange } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads/hotswap';
3
- import type { EvaluateCloudFormationTemplate } from '../evaluate-cloudformation-template';
4
- export declare function isHotswappableEcsServiceChange(logicalId: string, change: ResourceChange, evaluateCfnTemplate: EvaluateCloudFormationTemplate, hotswapPropertyOverrides: HotswapPropertyOverrides): Promise<ChangeHotswapResult>;
1
+ import type { HotswapPropertyOverrides, HotswapChange } from './common';
2
+ import { type ResourceChange } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads/hotswap';
3
+ import type { EvaluateCloudFormationTemplate } from '../cloudformation';
4
+ export declare function isHotswappableEcsServiceChange(logicalId: string, change: ResourceChange, evaluateCfnTemplate: EvaluateCloudFormationTemplate, hotswapPropertyOverrides: HotswapPropertyOverrides): Promise<HotswapChange[]>;
@@ -2,6 +2,8 @@
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.isHotswappableEcsServiceChange = isHotswappableEcsServiceChange;
4
4
  const common_1 = require("./common");
5
+ const hotswap_1 = require("../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads/hotswap");
6
+ const util_1 = require("../../util");
5
7
  const ECS_SERVICE_RESOURCE_TYPE = 'AWS::ECS::Service';
6
8
  async function isHotswappableEcsServiceChange(logicalId, change, evaluateCfnTemplate, hotswapPropertyOverrides) {
7
9
  // the only resource change we can evaluate here is an ECS TaskDefinition
@@ -21,20 +23,28 @@ async function isHotswappableEcsServiceChange(logicalId, change, evaluateCfnTemp
21
23
  for (const ecsServiceResource of ecsServiceResourcesReferencingTaskDef) {
22
24
  const serviceArn = await evaluateCfnTemplate.findPhysicalNameFor(ecsServiceResource.LogicalId);
23
25
  if (serviceArn) {
24
- ecsServicesReferencingTaskDef.push({ serviceArn });
26
+ ecsServicesReferencingTaskDef.push({
27
+ logicalId: ecsServiceResource.LogicalId,
28
+ serviceArn,
29
+ });
25
30
  }
26
31
  }
27
32
  if (ecsServicesReferencingTaskDef.length === 0) {
28
- // if there are no resources referencing the TaskDefinition,
29
- // hotswap is not possible in FALL_BACK mode
30
- (0, common_1.reportNonHotswappableChange)(ret, change, undefined, 'No ECS services reference the changed task definition', false);
33
+ /**
34
+ * ECS Services can have a task definition that doesn't refer to the task definition being updated.
35
+ * We have to log this as a non-hotswappable change to the task definition, but when we do,
36
+ * we wind up hotswapping the task definition and logging it as a non-hotswappable change.
37
+ *
38
+ * This logic prevents us from logging that change as non-hotswappable when we hotswap it.
39
+ */
40
+ ret.push((0, common_1.nonHotswappableChange)(change, hotswap_1.NonHotswappableReason.DEPENDENCY_UNSUPPORTED, 'No ECS services reference the changed task definition', undefined, false));
31
41
  }
32
42
  if (resourcesReferencingTaskDef.length > ecsServicesReferencingTaskDef.length) {
33
43
  // if something besides an ECS Service is referencing the TaskDefinition,
34
44
  // hotswap is not possible in FALL_BACK mode
35
45
  const nonEcsServiceTaskDefRefs = resourcesReferencingTaskDef.filter((r) => r.Type !== ECS_SERVICE_RESOURCE_TYPE);
36
46
  for (const taskRef of nonEcsServiceTaskDefRefs) {
37
- (0, common_1.reportNonHotswappableChange)(ret, change, undefined, `A resource '${taskRef.LogicalId}' with Type '${taskRef.Type}' that is not an ECS Service was found referencing the changed TaskDefinition '${logicalId}'`);
47
+ ret.push((0, common_1.nonHotswappableChange)(change, hotswap_1.NonHotswappableReason.DEPENDENCY_UNSUPPORTED, `A resource '${taskRef.LogicalId}' with Type '${taskRef.Type}' that is not an ECS Service was found referencing the changed TaskDefinition '${logicalId}'`));
38
48
  }
39
49
  }
40
50
  const namesOfHotswappableChanges = Object.keys(classifiedChanges.hotswappableProps);
@@ -43,21 +53,30 @@ async function isHotswappableEcsServiceChange(logicalId, change, evaluateCfnTemp
43
53
  ret.push({
44
54
  change: {
45
55
  cause: change,
56
+ resources: [
57
+ {
58
+ logicalId,
59
+ resourceType: change.newValue.Type,
60
+ physicalName: await taskDefinitionResource.Family,
61
+ metadata: evaluateCfnTemplate.metadataFor(logicalId),
62
+ },
63
+ ...ecsServicesReferencingTaskDef.map((ecsService) => ({
64
+ resourceType: ECS_SERVICE_RESOURCE_TYPE,
65
+ physicalName: ecsService.serviceArn.split('/')[2],
66
+ logicalId: ecsService.logicalId,
67
+ metadata: evaluateCfnTemplate.metadataFor(ecsService.logicalId),
68
+ })),
69
+ ],
46
70
  },
47
71
  hotswappable: true,
48
72
  service: 'ecs-service',
49
- resourceNames: [
50
- `ECS Task Definition '${await taskDefinitionResource.Family}'`,
51
- ...ecsServicesReferencingTaskDef.map((ecsService) => `ECS Service '${ecsService.serviceArn.split('/')[2]}'`),
52
- ],
53
73
  apply: async (sdk) => {
54
74
  // Step 1 - update the changed TaskDefinition, creating a new TaskDefinition Revision
55
75
  // we need to lowercase the evaluated TaskDef from CloudFormation,
56
76
  // as the AWS SDK uses lowercase property names for these
57
- var _a;
58
77
  // The SDK requires more properties here than its worth doing explicit typing for
59
78
  // instead, just use all the old values in the diff to fill them in implicitly
60
- const lowercasedTaskDef = (0, common_1.transformObjectKeys)(taskDefinitionResource, common_1.lowerCaseFirstCharacter, {
79
+ const lowercasedTaskDef = (0, util_1.transformObjectKeys)(taskDefinitionResource, util_1.lowerCaseFirstCharacter, {
61
80
  // All the properties that take arbitrary string as keys i.e. { "string" : "string" }
62
81
  // https://docs.aws.amazon.com/AmazonECS/latest/APIReference/API_RegisterTaskDefinition.html#API_RegisterTaskDefinition_RequestSyntax
63
82
  ContainerDefinitions: {
@@ -77,16 +96,15 @@ async function isHotswappableEcsServiceChange(logicalId, change, evaluateCfnTemp
77
96
  },
78
97
  });
79
98
  const registerTaskDefResponse = await sdk.ecs().registerTaskDefinition(lowercasedTaskDef);
80
- const taskDefRevArn = (_a = registerTaskDefResponse.taskDefinition) === null || _a === void 0 ? void 0 : _a.taskDefinitionArn;
99
+ const taskDefRevArn = registerTaskDefResponse.taskDefinition?.taskDefinitionArn;
81
100
  let ecsHotswapProperties = hotswapPropertyOverrides.ecsHotswapProperties;
82
- let minimumHealthyPercent = ecsHotswapProperties === null || ecsHotswapProperties === void 0 ? void 0 : ecsHotswapProperties.minimumHealthyPercent;
83
- let maximumHealthyPercent = ecsHotswapProperties === null || ecsHotswapProperties === void 0 ? void 0 : ecsHotswapProperties.maximumHealthyPercent;
101
+ let minimumHealthyPercent = ecsHotswapProperties?.minimumHealthyPercent;
102
+ let maximumHealthyPercent = ecsHotswapProperties?.maximumHealthyPercent;
84
103
  // Step 2 - update the services using that TaskDefinition to point to the new TaskDefinition Revision
85
104
  // Forcing New Deployment and setting Minimum Healthy Percent to 0.
86
105
  // As CDK HotSwap is development only, this seems the most efficient way to ensure all tasks are replaced immediately, regardless of original amount
87
106
  // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism
88
107
  await Promise.all(ecsServicesReferencingTaskDef.map(async (service) => {
89
- var _a;
90
108
  const cluster = service.serviceArn.split('/')[1];
91
109
  const update = await sdk.ecs().updateService({
92
110
  service: service.serviceArn,
@@ -99,7 +117,7 @@ async function isHotswappableEcsServiceChange(logicalId, change, evaluateCfnTemp
99
117
  },
100
118
  });
101
119
  await sdk.ecs().waitUntilServicesStable({
102
- cluster: (_a = update.service) === null || _a === void 0 ? void 0 : _a.clusterArn,
120
+ cluster: update.service?.clusterArn,
103
121
  services: [service.serviceArn],
104
122
  });
105
123
  }));
@@ -109,13 +127,12 @@ async function isHotswappableEcsServiceChange(logicalId, change, evaluateCfnTemp
109
127
  return ret;
110
128
  }
111
129
  async function prepareTaskDefinitionChange(evaluateCfnTemplate, logicalId, change) {
112
- var _a;
113
130
  const taskDefinitionResource = {
114
131
  ...change.oldValue.Properties,
115
- ContainerDefinitions: (_a = change.newValue.Properties) === null || _a === void 0 ? void 0 : _a.ContainerDefinitions,
132
+ ContainerDefinitions: change.newValue.Properties?.ContainerDefinitions,
116
133
  };
117
134
  // first, let's get the name of the family
118
- const familyNameOrArn = await evaluateCfnTemplate.establishResourcePhysicalName(logicalId, taskDefinitionResource === null || taskDefinitionResource === void 0 ? void 0 : taskDefinitionResource.Family);
135
+ const familyNameOrArn = await evaluateCfnTemplate.establishResourcePhysicalName(logicalId, taskDefinitionResource?.Family);
119
136
  if (!familyNameOrArn) {
120
137
  // if the Family property has not been provided, and we can't find it in the current Stack,
121
138
  // this means hotswapping is not possible
@@ -133,10 +150,10 @@ async function prepareTaskDefinitionChange(evaluateCfnTemplate, logicalId, chang
133
150
  // then, let's evaluate the body of the remainder of the TaskDef (without the Family property)
134
151
  return {
135
152
  ...(await evaluateCfnTemplate.evaluateCfnExpression({
136
- ...(taskDefinitionResource !== null && taskDefinitionResource !== void 0 ? taskDefinitionResource : {}),
153
+ ...(taskDefinitionResource ?? {}),
137
154
  Family: undefined,
138
155
  })),
139
156
  Family: family,
140
157
  };
141
158
  }
142
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ecs-services.js","sourceRoot":"","sources":["ecs-services.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n  HotswapPropertyOverrides,\n  ChangeHotswapResult,\n} from './common';\nimport {\n  classifyChanges, lowerCaseFirstCharacter,\n  reportNonHotswappableChange,\n  transformObjectKeys,\n} from './common';\nimport type { ResourceChange } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads/hotswap';\nimport type { SDK } from '../aws-auth';\nimport type { EvaluateCloudFormationTemplate } from '../evaluate-cloudformation-template';\n\nconst ECS_SERVICE_RESOURCE_TYPE = 'AWS::ECS::Service';\n\nexport async function isHotswappableEcsServiceChange(\n  logicalId: string,\n  change: ResourceChange,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<ChangeHotswapResult> {\n  // the only resource change we can evaluate here is an ECS TaskDefinition\n  if (change.newValue.Type !== 'AWS::ECS::TaskDefinition') {\n    return [];\n  }\n\n  const ret: ChangeHotswapResult = [];\n\n  // We only allow a change in the ContainerDefinitions of the TaskDefinition for now -\n  // it contains the image and environment variables, so seems like a safe bet for now.\n  // We might revisit this decision in the future though!\n  const classifiedChanges = classifyChanges(change, ['ContainerDefinitions']);\n  classifiedChanges.reportNonHotswappablePropertyChanges(ret);\n\n  // find all ECS Services that reference the TaskDefinition that changed\n  const resourcesReferencingTaskDef = evaluateCfnTemplate.findReferencesTo(logicalId);\n  const ecsServiceResourcesReferencingTaskDef = resourcesReferencingTaskDef.filter(\n    (r) => r.Type === ECS_SERVICE_RESOURCE_TYPE,\n  );\n  const ecsServicesReferencingTaskDef = new Array<EcsService>();\n  for (const ecsServiceResource of ecsServiceResourcesReferencingTaskDef) {\n    const serviceArn = await evaluateCfnTemplate.findPhysicalNameFor(ecsServiceResource.LogicalId);\n    if (serviceArn) {\n      ecsServicesReferencingTaskDef.push({ serviceArn });\n    }\n  }\n  if (ecsServicesReferencingTaskDef.length === 0) {\n    // if there are no resources referencing the TaskDefinition,\n    // hotswap is not possible in FALL_BACK mode\n    reportNonHotswappableChange(ret, change, undefined, 'No ECS services reference the changed task definition', false);\n  }\n  if (resourcesReferencingTaskDef.length > ecsServicesReferencingTaskDef.length) {\n    // if something besides an ECS Service is referencing the TaskDefinition,\n    // hotswap is not possible in FALL_BACK mode\n    const nonEcsServiceTaskDefRefs = resourcesReferencingTaskDef.filter((r) => r.Type !== ECS_SERVICE_RESOURCE_TYPE);\n    for (const taskRef of nonEcsServiceTaskDefRefs) {\n      reportNonHotswappableChange(\n        ret,\n        change,\n        undefined,\n        `A resource '${taskRef.LogicalId}' with Type '${taskRef.Type}' that is not an ECS Service was found referencing the changed TaskDefinition '${logicalId}'`,\n      );\n    }\n  }\n\n  const namesOfHotswappableChanges = Object.keys(classifiedChanges.hotswappableProps);\n  if (namesOfHotswappableChanges.length > 0) {\n    const taskDefinitionResource = await prepareTaskDefinitionChange(evaluateCfnTemplate, logicalId, change);\n    ret.push({\n      change: {\n        cause: change,\n      },\n      hotswappable: true,\n      service: 'ecs-service',\n      resourceNames: [\n        `ECS Task Definition '${await taskDefinitionResource.Family}'`,\n        ...ecsServicesReferencingTaskDef.map((ecsService) => `ECS Service '${ecsService.serviceArn.split('/')[2]}'`),\n      ],\n      apply: async (sdk: SDK) => {\n        // Step 1 - update the changed TaskDefinition, creating a new TaskDefinition Revision\n        // we need to lowercase the evaluated TaskDef from CloudFormation,\n        // as the AWS SDK uses lowercase property names for these\n\n        // The SDK requires more properties here than its worth doing explicit typing for\n        // instead, just use all the old values in the diff to fill them in implicitly\n        const lowercasedTaskDef = transformObjectKeys(taskDefinitionResource, lowerCaseFirstCharacter, {\n          // All the properties that take arbitrary string as keys i.e. { \"string\" : \"string\" }\n          // https://docs.aws.amazon.com/AmazonECS/latest/APIReference/API_RegisterTaskDefinition.html#API_RegisterTaskDefinition_RequestSyntax\n          ContainerDefinitions: {\n            DockerLabels: true,\n            FirelensConfiguration: {\n              Options: true,\n            },\n            LogConfiguration: {\n              Options: true,\n            },\n          },\n          Volumes: {\n            DockerVolumeConfiguration: {\n              DriverOpts: true,\n              Labels: true,\n            },\n          },\n        });\n        const registerTaskDefResponse = await sdk.ecs().registerTaskDefinition(lowercasedTaskDef);\n        const taskDefRevArn = registerTaskDefResponse.taskDefinition?.taskDefinitionArn;\n\n        let ecsHotswapProperties = hotswapPropertyOverrides.ecsHotswapProperties;\n        let minimumHealthyPercent = ecsHotswapProperties?.minimumHealthyPercent;\n        let maximumHealthyPercent = ecsHotswapProperties?.maximumHealthyPercent;\n\n        // Step 2 - update the services using that TaskDefinition to point to the new TaskDefinition Revision\n        // Forcing New Deployment and setting Minimum Healthy Percent to 0.\n        // As CDK HotSwap is development only, this seems the most efficient way to ensure all tasks are replaced immediately, regardless of original amount\n        // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n        await Promise.all(\n          ecsServicesReferencingTaskDef.map(async (service) => {\n            const cluster = service.serviceArn.split('/')[1];\n            const update = await sdk.ecs().updateService({\n              service: service.serviceArn,\n              taskDefinition: taskDefRevArn,\n              cluster,\n              forceNewDeployment: true,\n              deploymentConfiguration: {\n                minimumHealthyPercent: minimumHealthyPercent !== undefined ? minimumHealthyPercent : 0,\n                maximumPercent: maximumHealthyPercent !== undefined ? maximumHealthyPercent : undefined,\n              },\n            });\n\n            await sdk.ecs().waitUntilServicesStable({\n              cluster: update.service?.clusterArn,\n              services: [service.serviceArn],\n            });\n          }),\n        );\n      },\n    });\n  }\n\n  return ret;\n}\n\ninterface EcsService {\n  readonly serviceArn: string;\n}\n\nasync function prepareTaskDefinitionChange(\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  logicalId: string,\n  change: ResourceChange,\n) {\n  const taskDefinitionResource: { [name: string]: any } = {\n    ...change.oldValue.Properties,\n    ContainerDefinitions: change.newValue.Properties?.ContainerDefinitions,\n  };\n  // first, let's get the name of the family\n  const familyNameOrArn = await evaluateCfnTemplate.establishResourcePhysicalName(\n    logicalId,\n    taskDefinitionResource?.Family,\n  );\n  if (!familyNameOrArn) {\n    // if the Family property has not been provided, and we can't find it in the current Stack,\n    // this means hotswapping is not possible\n    return;\n  }\n  // the physical name of the Task Definition in CloudFormation includes its current revision number at the end,\n  // remove it if needed\n  const familyNameOrArnParts = familyNameOrArn.split(':');\n  const family =\n    familyNameOrArnParts.length > 1\n      ? // familyNameOrArn is actually an ARN, of the format 'arn:aws:ecs:region:account:task-definition/<family-name>:<revision-nr>'\n    // so, take the 6th element, at index 5, and split it on '/'\n      familyNameOrArnParts[5].split('/')[1]\n      : // otherwise, familyNameOrArn is just the simple name evaluated from the CloudFormation template\n      familyNameOrArn;\n  // then, let's evaluate the body of the remainder of the TaskDef (without the Family property)\n  return {\n    ...(await evaluateCfnTemplate.evaluateCfnExpression({\n      ...(taskDefinitionResource ?? {}),\n      Family: undefined,\n    })),\n    Family: family,\n  };\n}\n"]}
159
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ecs-services.js","sourceRoot":"","sources":["ecs-services.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n  HotswapPropertyOverrides,\n  HotswapChange,\n} from './common';\nimport {\n  classifyChanges,\n  nonHotswappableChange,\n} from './common';\nimport { NonHotswappableReason, type ResourceChange } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads/hotswap';\nimport { lowerCaseFirstCharacter, transformObjectKeys } from '../../util';\nimport type { SDK } from '../aws-auth';\nimport type { EvaluateCloudFormationTemplate } from '../cloudformation';\n\nconst ECS_SERVICE_RESOURCE_TYPE = 'AWS::ECS::Service';\n\nexport async function isHotswappableEcsServiceChange(\n  logicalId: string,\n  change: ResourceChange,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<HotswapChange[]> {\n  // the only resource change we can evaluate here is an ECS TaskDefinition\n  if (change.newValue.Type !== 'AWS::ECS::TaskDefinition') {\n    return [];\n  }\n\n  const ret: HotswapChange[] = [];\n\n  // We only allow a change in the ContainerDefinitions of the TaskDefinition for now -\n  // it contains the image and environment variables, so seems like a safe bet for now.\n  // We might revisit this decision in the future though!\n  const classifiedChanges = classifyChanges(change, ['ContainerDefinitions']);\n  classifiedChanges.reportNonHotswappablePropertyChanges(ret);\n\n  // find all ECS Services that reference the TaskDefinition that changed\n  const resourcesReferencingTaskDef = evaluateCfnTemplate.findReferencesTo(logicalId);\n  const ecsServiceResourcesReferencingTaskDef = resourcesReferencingTaskDef.filter(\n    (r) => r.Type === ECS_SERVICE_RESOURCE_TYPE,\n  );\n  const ecsServicesReferencingTaskDef = new Array<EcsService>();\n  for (const ecsServiceResource of ecsServiceResourcesReferencingTaskDef) {\n    const serviceArn = await evaluateCfnTemplate.findPhysicalNameFor(ecsServiceResource.LogicalId);\n    if (serviceArn) {\n      ecsServicesReferencingTaskDef.push({\n        logicalId: ecsServiceResource.LogicalId,\n        serviceArn,\n      });\n    }\n  }\n  if (ecsServicesReferencingTaskDef.length === 0) {\n    /**\n     * ECS Services can have a task definition that doesn't refer to the task definition being updated.\n     * We have to log this as a non-hotswappable change to the task definition, but when we do,\n     * we wind up hotswapping the task definition and logging it as a non-hotswappable change.\n     *\n     * This logic prevents us from logging that change as non-hotswappable when we hotswap it.\n     */\n    ret.push(nonHotswappableChange(\n      change,\n      NonHotswappableReason.DEPENDENCY_UNSUPPORTED,\n      'No ECS services reference the changed task definition',\n      undefined,\n      false,\n    ));\n  }\n  if (resourcesReferencingTaskDef.length > ecsServicesReferencingTaskDef.length) {\n    // if something besides an ECS Service is referencing the TaskDefinition,\n    // hotswap is not possible in FALL_BACK mode\n    const nonEcsServiceTaskDefRefs = resourcesReferencingTaskDef.filter((r) => r.Type !== ECS_SERVICE_RESOURCE_TYPE);\n    for (const taskRef of nonEcsServiceTaskDefRefs) {\n      ret.push(nonHotswappableChange(\n        change,\n        NonHotswappableReason.DEPENDENCY_UNSUPPORTED,\n        `A resource '${taskRef.LogicalId}' with Type '${taskRef.Type}' that is not an ECS Service was found referencing the changed TaskDefinition '${logicalId}'`,\n      ));\n    }\n  }\n\n  const namesOfHotswappableChanges = Object.keys(classifiedChanges.hotswappableProps);\n  if (namesOfHotswappableChanges.length > 0) {\n    const taskDefinitionResource = await prepareTaskDefinitionChange(evaluateCfnTemplate, logicalId, change);\n    ret.push({\n      change: {\n        cause: change,\n        resources: [\n          {\n            logicalId,\n            resourceType: change.newValue.Type,\n            physicalName: await taskDefinitionResource.Family,\n            metadata: evaluateCfnTemplate.metadataFor(logicalId),\n          },\n          ...ecsServicesReferencingTaskDef.map((ecsService) => ({\n            resourceType: ECS_SERVICE_RESOURCE_TYPE,\n            physicalName: ecsService.serviceArn.split('/')[2],\n            logicalId: ecsService.logicalId,\n            metadata: evaluateCfnTemplate.metadataFor(ecsService.logicalId),\n          })),\n        ],\n      },\n      hotswappable: true,\n      service: 'ecs-service',\n      apply: async (sdk: SDK) => {\n        // Step 1 - update the changed TaskDefinition, creating a new TaskDefinition Revision\n        // we need to lowercase the evaluated TaskDef from CloudFormation,\n        // as the AWS SDK uses lowercase property names for these\n\n        // The SDK requires more properties here than its worth doing explicit typing for\n        // instead, just use all the old values in the diff to fill them in implicitly\n        const lowercasedTaskDef = transformObjectKeys(taskDefinitionResource, lowerCaseFirstCharacter, {\n          // All the properties that take arbitrary string as keys i.e. { \"string\" : \"string\" }\n          // https://docs.aws.amazon.com/AmazonECS/latest/APIReference/API_RegisterTaskDefinition.html#API_RegisterTaskDefinition_RequestSyntax\n          ContainerDefinitions: {\n            DockerLabels: true,\n            FirelensConfiguration: {\n              Options: true,\n            },\n            LogConfiguration: {\n              Options: true,\n            },\n          },\n          Volumes: {\n            DockerVolumeConfiguration: {\n              DriverOpts: true,\n              Labels: true,\n            },\n          },\n        });\n        const registerTaskDefResponse = await sdk.ecs().registerTaskDefinition(lowercasedTaskDef);\n        const taskDefRevArn = registerTaskDefResponse.taskDefinition?.taskDefinitionArn;\n\n        let ecsHotswapProperties = hotswapPropertyOverrides.ecsHotswapProperties;\n        let minimumHealthyPercent = ecsHotswapProperties?.minimumHealthyPercent;\n        let maximumHealthyPercent = ecsHotswapProperties?.maximumHealthyPercent;\n\n        // Step 2 - update the services using that TaskDefinition to point to the new TaskDefinition Revision\n        // Forcing New Deployment and setting Minimum Healthy Percent to 0.\n        // As CDK HotSwap is development only, this seems the most efficient way to ensure all tasks are replaced immediately, regardless of original amount\n        // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n        await Promise.all(\n          ecsServicesReferencingTaskDef.map(async (service) => {\n            const cluster = service.serviceArn.split('/')[1];\n            const update = await sdk.ecs().updateService({\n              service: service.serviceArn,\n              taskDefinition: taskDefRevArn,\n              cluster,\n              forceNewDeployment: true,\n              deploymentConfiguration: {\n                minimumHealthyPercent: minimumHealthyPercent !== undefined ? minimumHealthyPercent : 0,\n                maximumPercent: maximumHealthyPercent !== undefined ? maximumHealthyPercent : undefined,\n              },\n            });\n\n            await sdk.ecs().waitUntilServicesStable({\n              cluster: update.service?.clusterArn,\n              services: [service.serviceArn],\n            });\n          }),\n        );\n      },\n    });\n  }\n\n  return ret;\n}\n\ninterface EcsService {\n  readonly logicalId: string;\n  readonly serviceArn: string;\n}\n\nasync function prepareTaskDefinitionChange(\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  logicalId: string,\n  change: ResourceChange,\n) {\n  const taskDefinitionResource: { [name: string]: any } = {\n    ...change.oldValue.Properties,\n    ContainerDefinitions: change.newValue.Properties?.ContainerDefinitions,\n  };\n  // first, let's get the name of the family\n  const familyNameOrArn = await evaluateCfnTemplate.establishResourcePhysicalName(\n    logicalId,\n    taskDefinitionResource?.Family,\n  );\n  if (!familyNameOrArn) {\n    // if the Family property has not been provided, and we can't find it in the current Stack,\n    // this means hotswapping is not possible\n    return;\n  }\n  // the physical name of the Task Definition in CloudFormation includes its current revision number at the end,\n  // remove it if needed\n  const familyNameOrArnParts = familyNameOrArn.split(':');\n  const family =\n    familyNameOrArnParts.length > 1\n      ? // familyNameOrArn is actually an ARN, of the format 'arn:aws:ecs:region:account:task-definition/<family-name>:<revision-nr>'\n    // so, take the 6th element, at index 5, and split it on '/'\n      familyNameOrArnParts[5].split('/')[1]\n      : // otherwise, familyNameOrArn is just the simple name evaluated from the CloudFormation template\n      familyNameOrArn;\n  // then, let's evaluate the body of the remainder of the TaskDef (without the Family property)\n  return {\n    ...(await evaluateCfnTemplate.evaluateCfnExpression({\n      ...(taskDefinitionResource ?? {}),\n      Family: undefined,\n    })),\n    Family: family,\n  };\n}\n"]}
@@ -1,4 +1,4 @@
1
- import type { ChangeHotswapResult } from './common';
1
+ import type { HotswapChange } from './common';
2
2
  import type { ResourceChange } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads/hotswap';
3
- import { type EvaluateCloudFormationTemplate } from '../evaluate-cloudformation-template';
4
- export declare function isHotswappableLambdaFunctionChange(logicalId: string, change: ResourceChange, evaluateCfnTemplate: EvaluateCloudFormationTemplate): Promise<ChangeHotswapResult>;
3
+ import { type EvaluateCloudFormationTemplate } from '../cloudformation';
4
+ export declare function isHotswappableLambdaFunctionChange(logicalId: string, change: ResourceChange, evaluateCfnTemplate: EvaluateCloudFormationTemplate): Promise<HotswapChange[]>;