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.
- package/README.md +1 -1
- package/THIRD_PARTY_LICENSES +86 -86
- package/build-info.json +2 -2
- package/db.json.gz +0 -0
- package/lib/api/aws-auth/awscli-compatible.js +9 -10
- package/lib/api/aws-auth/credential-plugins.js +6 -7
- package/lib/api/aws-auth/sdk-logger.js +3 -4
- package/lib/api/aws-auth/sdk-provider.js +11 -13
- package/lib/api/aws-auth/sdk.js +8 -9
- package/lib/api/aws-auth/tracing.js +3 -4
- package/lib/api/aws-auth/user-agent.js +4 -5
- package/lib/api/bootstrap/bootstrap-environment.d.ts +1 -1
- package/lib/api/bootstrap/bootstrap-environment.js +42 -46
- package/lib/api/bootstrap/bootstrap-props.d.ts +1 -1
- package/lib/api/bootstrap/bootstrap-props.js +1 -1
- package/lib/api/bootstrap/deploy-bootstrap.d.ts +1 -1
- package/lib/api/bootstrap/deploy-bootstrap.js +11 -14
- package/lib/api/{evaluate-cloudformation-template.d.ts → cloudformation/evaluate-cloudformation-template.d.ts} +4 -8
- package/lib/api/{evaluate-cloudformation-template.js → cloudformation/evaluate-cloudformation-template.js} +16 -25
- package/lib/api/cloudformation/index.d.ts +4 -0
- package/lib/api/cloudformation/index.js +21 -0
- package/lib/api/{deployments → cloudformation}/nested-stack-helpers.d.ts +3 -9
- package/lib/api/cloudformation/nested-stack-helpers.js +86 -0
- package/lib/api/cloudformation/stack-helpers.d.ts +88 -0
- package/lib/api/cloudformation/stack-helpers.js +158 -0
- package/lib/api/{util → cloudformation}/template-body-parameter.d.ts +3 -2
- package/lib/api/cloudformation/template-body-parameter.js +104 -0
- package/lib/api/context.js +3 -3
- package/lib/api/cxapp/cloud-assembly.d.ts +6 -4
- package/lib/api/cxapp/cloud-assembly.js +25 -26
- package/lib/api/cxapp/cloud-executable.d.ts +5 -0
- package/lib/api/cxapp/cloud-executable.js +9 -10
- package/lib/api/cxapp/environments.js +4 -4
- package/lib/api/cxapp/exec.d.ts +5 -4
- package/lib/api/cxapp/exec.js +76 -72
- package/lib/api/deployments/asset-publishing.d.ts +0 -2
- package/lib/api/deployments/asset-publishing.js +24 -31
- package/lib/api/deployments/assets.d.ts +1 -1
- package/lib/api/deployments/assets.js +12 -13
- package/lib/api/deployments/{cloudformation.d.ts → cfn-api.d.ts} +5 -102
- package/lib/api/deployments/cfn-api.js +438 -0
- package/lib/api/deployments/checks.d.ts +1 -1
- package/lib/api/deployments/checks.js +12 -13
- package/lib/api/deployments/deploy-stack.d.ts +2 -3
- package/lib/api/deployments/deploy-stack.js +34 -45
- package/lib/api/deployments/deployment-result.js +3 -3
- package/lib/api/deployments/deployments.d.ts +3 -3
- package/lib/api/deployments/deployments.js +35 -42
- package/lib/api/deployments/hotswap-deployments.d.ts +2 -2
- package/lib/api/deployments/hotswap-deployments.js +122 -69
- package/lib/api/deployments/index.d.ts +1 -2
- package/lib/api/deployments/index.js +2 -3
- package/lib/api/environment/environment-access.d.ts +2 -2
- package/lib/api/environment/environment-access.js +18 -20
- package/lib/api/environment/environment-resources.d.ts +1 -1
- package/lib/api/environment/environment-resources.js +17 -19
- package/lib/api/environment/index.d.ts +1 -0
- package/lib/api/environment/index.js +2 -1
- package/lib/api/environment/placeholders.js +23 -0
- package/lib/api/garbage-collection/garbage-collector.d.ts +1 -1
- package/lib/api/garbage-collection/garbage-collector.js +56 -66
- package/lib/api/garbage-collection/progress-printer.d.ts +1 -1
- package/lib/api/garbage-collection/progress-printer.js +7 -7
- package/lib/api/garbage-collection/stack-refresh.d.ts +1 -1
- package/lib/api/garbage-collection/stack-refresh.js +12 -15
- package/lib/api/hotswap/appsync-mapping-templates.d.ts +3 -3
- package/lib/api/hotswap/appsync-mapping-templates.js +25 -22
- package/lib/api/hotswap/code-build-projects.d.ts +3 -3
- package/lib/api/hotswap/code-build-projects.js +12 -7
- package/lib/api/hotswap/common.d.ts +13 -61
- package/lib/api/hotswap/common.js +40 -70
- package/lib/api/hotswap/ecs-services.d.ts +4 -4
- package/lib/api/hotswap/ecs-services.js +38 -21
- package/lib/api/hotswap/lambda-functions.d.ts +3 -3
- package/lib/api/hotswap/lambda-functions.js +23 -19
- package/lib/api/hotswap/s3-bucket-deployments.d.ts +3 -3
- package/lib/api/hotswap/s3-bucket-deployments.js +11 -7
- package/lib/api/hotswap/stepfunctions-state-machines.d.ts +3 -3
- package/lib/api/hotswap/stepfunctions-state-machines.js +8 -4
- package/lib/api/logs/find-cloudwatch-logs.js +6 -7
- package/lib/api/logs/logs-monitor.js +5 -8
- package/lib/api/plugin/plugin.js +6 -10
- package/lib/api/resource-import/importer.d.ts +8 -8
- package/lib/api/resource-import/importer.js +27 -42
- package/lib/api/resource-import/migrator.d.ts +3 -3
- package/lib/api/resource-import/migrator.js +6 -6
- package/lib/api/settings.d.ts +0 -3
- package/lib/api/settings.js +4 -40
- package/lib/api/stack-events/stack-activity-monitor.d.ts +1 -1
- package/lib/api/stack-events/stack-activity-monitor.js +12 -15
- package/lib/api/stack-events/stack-event-poller.js +9 -10
- package/lib/api/toolkit-info.d.ts +2 -2
- package/lib/api/toolkit-info.js +20 -24
- package/lib/{tree.d.ts → api/tree.d.ts} +2 -2
- package/lib/api/tree.js +37 -0
- package/lib/api/util/rwlock.js +4 -4
- package/lib/api/work-graph/work-graph-builder.js +4 -4
- package/lib/api/work-graph/work-graph.d.ts +1 -1
- package/lib/api/work-graph/work-graph.js +13 -15
- package/lib/cli/activity-printer/base.d.ts +2 -2
- package/lib/cli/activity-printer/base.js +6 -8
- package/lib/cli/activity-printer/current.js +7 -11
- package/lib/cli/activity-printer/history.js +2 -3
- package/lib/cli/cdk-toolkit.d.ts +16 -19
- package/lib/cli/cdk-toolkit.js +118 -74
- package/lib/cli/ci-systems.js +2 -3
- package/lib/cli/cli-config.js +4 -4
- package/lib/cli/cli.js +49 -50
- package/lib/cli/convert-to-user-input.js +110 -111
- package/lib/{toolkit → cli/io-host}/cli-io-host.d.ts +6 -2
- package/lib/cli/io-host/cli-io-host.js +356 -0
- package/lib/cli/io-host/index.d.ts +1 -0
- package/lib/{toolkit/error.js → cli/io-host/index.js} +2 -2
- package/lib/cli/messages.d.ts +1 -1
- package/lib/cli/messages.js +2 -3
- package/lib/cli/pretty-print-error.d.ts +1 -0
- package/lib/cli/pretty-print-error.js +35 -0
- package/lib/cli/root-dir.js +4 -4
- package/lib/cli/user-configuration.js +57 -14
- package/lib/cli/util/npm.js +3 -3
- package/lib/cli/util/yargs-helpers.d.ts +1 -1
- package/lib/cli/util/yargs-helpers.js +3 -3
- package/lib/cli/version.js +4 -4
- package/lib/commands/context.js +7 -8
- package/lib/commands/diff.d.ts +1 -0
- package/lib/commands/diff.js +7 -0
- package/lib/commands/init/index.d.ts +1 -0
- package/lib/commands/init/index.js +18 -0
- package/lib/commands/init/init-hooks.js +63 -0
- package/lib/commands/init/init.js +435 -0
- package/lib/{os.js → commands/init/os.js} +4 -4
- package/lib/{list-stacks.d.ts → commands/list-stacks.d.ts} +1 -1
- package/lib/{list-stacks.js → commands/list-stacks.js} +2 -2
- package/lib/commands/migrate.js +29 -32
- package/lib/context-providers/ami.d.ts +3 -1
- package/lib/context-providers/ami.js +8 -8
- package/lib/context-providers/availability-zones.d.ts +3 -1
- package/lib/context-providers/availability-zones.js +4 -4
- package/lib/context-providers/cc-api-provider.d.ts +8 -12
- package/lib/context-providers/cc-api-provider.js +94 -66
- package/lib/context-providers/endpoint-service-availability-zones.d.ts +3 -1
- package/lib/context-providers/endpoint-service-availability-zones.js +6 -6
- package/lib/context-providers/hosted-zones.d.ts +3 -1
- package/lib/context-providers/hosted-zones.js +11 -11
- package/lib/context-providers/index.d.ts +19 -5
- package/lib/context-providers/index.js +35 -17
- package/lib/context-providers/keys.d.ts +3 -1
- package/lib/context-providers/keys.js +8 -8
- package/lib/context-providers/load-balancers.js +15 -18
- package/lib/context-providers/security-groups.js +10 -12
- package/lib/context-providers/ssm-parameters.d.ts +3 -1
- package/lib/context-providers/ssm-parameters.js +7 -7
- package/lib/context-providers/vpcs.d.ts +3 -1
- package/lib/context-providers/vpcs.js +14 -15
- package/lib/index.js +124098 -123198
- package/lib/init-templates/.init-version.json +1 -1
- package/lib/init-templates/.recommended-feature-flags.json +3 -1
- package/lib/legacy-exports-source.d.ts +4 -5
- package/lib/legacy-exports-source.js +6 -7
- package/lib/logging.js +2 -2
- package/lib/notices.d.ts +1 -1
- package/lib/notices.js +26 -32
- package/package.json +29 -29
- package/lib/api/deployments/cloudformation.js +0 -597
- package/lib/api/deployments/nested-stack-helpers.js +0 -88
- package/lib/api/util/placeholders.js +0 -24
- package/lib/api/util/template-body-parameter.js +0 -103
- package/lib/diff.d.ts +0 -28
- package/lib/diff.js +0 -165
- package/lib/init-hooks.js +0 -63
- package/lib/init.js +0 -437
- package/lib/toolkit/cli-io-host.js +0 -353
- package/lib/toolkit/error.d.ts +0 -1
- package/lib/tree.js +0 -40
- /package/lib/api/{util → environment}/placeholders.d.ts +0 -0
- /package/lib/{init-hooks.d.ts → commands/init/init-hooks.d.ts} +0 -0
- /package/lib/{init.d.ts → commands/init/init.d.ts} +0 -0
- /package/lib/{os.d.ts → commands/init/os.d.ts} +0 -0
|
@@ -4,11 +4,11 @@ exports.tryHotswapDeployment = tryHotswapDeployment;
|
|
|
4
4
|
const util_1 = require("util");
|
|
5
5
|
const cfn_diff = require("@aws-cdk/cloudformation-diff");
|
|
6
6
|
const chalk = require("chalk");
|
|
7
|
+
const api_1 = require("../../../../@aws-cdk/tmp-toolkit-helpers/src/api");
|
|
8
|
+
const payloads_1 = require("../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads");
|
|
7
9
|
const private_1 = require("../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private");
|
|
8
|
-
const nested_stack_helpers_1 = require("./nested-stack-helpers");
|
|
9
|
-
const error_1 = require("../../toolkit/error");
|
|
10
10
|
const util_2 = require("../../util");
|
|
11
|
-
const
|
|
11
|
+
const cloudformation_1 = require("../cloudformation");
|
|
12
12
|
const appsync_mapping_templates_1 = require("../hotswap/appsync-mapping-templates");
|
|
13
13
|
const code_build_projects_1 = require("../hotswap/code-build-projects");
|
|
14
14
|
const common_1 = require("../hotswap/common");
|
|
@@ -39,7 +39,7 @@ const RESOURCE_DETECTORS = {
|
|
|
39
39
|
if (await (0, s3_bucket_deployments_1.skipChangeForS3DeployCustomResourcePolicy)(logicalId, change, evaluateCfnTemplate)) {
|
|
40
40
|
return [];
|
|
41
41
|
}
|
|
42
|
-
return (0, common_1.
|
|
42
|
+
return [(0, common_1.nonHotswappableResource)(change)];
|
|
43
43
|
},
|
|
44
44
|
'AWS::CDK::Metadata': async () => [],
|
|
45
45
|
};
|
|
@@ -55,8 +55,8 @@ async function tryHotswapDeployment(sdkProvider, ioHelper, assetParams, cloudFor
|
|
|
55
55
|
mode: hotswapMode,
|
|
56
56
|
});
|
|
57
57
|
const result = await hotswapDeployment(sdkProvider, hotswapSpan, assetParams, stackArtifact, hotswapMode, hotswapPropertyOverrides);
|
|
58
|
-
await hotswapSpan.end();
|
|
59
|
-
if (
|
|
58
|
+
await hotswapSpan.end(result);
|
|
59
|
+
if (result?.hotswapped === true) {
|
|
60
60
|
return {
|
|
61
61
|
type: 'did-deploy-stack',
|
|
62
62
|
noOp: result.hotswappableChanges.length === 0,
|
|
@@ -76,8 +76,8 @@ async function hotswapDeployment(sdkProvider, ioSpan, assetParams, stack, hotswa
|
|
|
76
76
|
// create a new SDK using the CLI credentials, because the default one will not work for new-style synthesis -
|
|
77
77
|
// it assumes the bootstrap deploy Role, which doesn't have permissions to update Lambda functions
|
|
78
78
|
const sdk = (await sdkProvider.forEnvironment(resolvedEnv, plugin_1.Mode.ForWriting)).sdk;
|
|
79
|
-
const currentTemplate = await (0,
|
|
80
|
-
const evaluateCfnTemplate = new
|
|
79
|
+
const currentTemplate = await (0, cloudformation_1.loadCurrentTemplateWithNestedStacks)(stack, sdk);
|
|
80
|
+
const evaluateCfnTemplate = new cloudformation_1.EvaluateCloudFormationTemplate({
|
|
81
81
|
stackArtifact: stack,
|
|
82
82
|
parameters: assetParams,
|
|
83
83
|
account: resolvedEnv.account,
|
|
@@ -87,13 +87,22 @@ async function hotswapDeployment(sdkProvider, ioSpan, assetParams, stack, hotswa
|
|
|
87
87
|
nestedStacks: currentTemplate.nestedStacks,
|
|
88
88
|
});
|
|
89
89
|
const stackChanges = cfn_diff.fullDiff(currentTemplate.deployedRootTemplate, stack.template);
|
|
90
|
-
const {
|
|
91
|
-
await
|
|
90
|
+
const { hotswappable, nonHotswappable } = await classifyResourceChanges(stackChanges, evaluateCfnTemplate, sdk, currentTemplate.nestedStacks, hotswapPropertyOverrides);
|
|
91
|
+
await logRejectedChanges(ioSpan, nonHotswappable, hotswapMode);
|
|
92
|
+
const hotswappableChanges = hotswappable.map(o => o.change);
|
|
93
|
+
const nonHotswappableChanges = nonHotswappable.map(n => n.change);
|
|
94
|
+
await ioSpan.notify(private_1.IO.CDK_TOOLKIT_I5401.msg('Hotswap plan created', {
|
|
95
|
+
stack,
|
|
96
|
+
mode: hotswapMode,
|
|
97
|
+
hotswappableChanges,
|
|
98
|
+
nonHotswappableChanges,
|
|
99
|
+
}));
|
|
92
100
|
// preserve classic hotswap behavior
|
|
93
101
|
if (hotswapMode === 'fall-back') {
|
|
94
102
|
if (nonHotswappableChanges.length > 0) {
|
|
95
103
|
return {
|
|
96
104
|
stack,
|
|
105
|
+
mode: hotswapMode,
|
|
97
106
|
hotswapped: false,
|
|
98
107
|
hotswappableChanges,
|
|
99
108
|
nonHotswappableChanges,
|
|
@@ -101,9 +110,10 @@ async function hotswapDeployment(sdkProvider, ioSpan, assetParams, stack, hotswa
|
|
|
101
110
|
}
|
|
102
111
|
}
|
|
103
112
|
// apply the short-circuitable changes
|
|
104
|
-
await
|
|
113
|
+
await applyAllHotswapOperations(sdk, ioSpan, hotswappable);
|
|
105
114
|
return {
|
|
106
115
|
stack,
|
|
116
|
+
mode: hotswapMode,
|
|
107
117
|
hotswapped: true,
|
|
108
118
|
hotswappableChanges,
|
|
109
119
|
nonHotswappableChanges,
|
|
@@ -114,7 +124,6 @@ async function hotswapDeployment(sdkProvider, ioSpan, assetParams, stack, hotswa
|
|
|
114
124
|
* Metadata changes are excluded from the list of (non)hotswappable resources.
|
|
115
125
|
*/
|
|
116
126
|
async function classifyResourceChanges(stackChanges, evaluateCfnTemplate, sdk, nestedStackNames, hotswapPropertyOverrides) {
|
|
117
|
-
var _a, _b;
|
|
118
127
|
const resourceDifferences = getStackResourceDifferences(stackChanges);
|
|
119
128
|
const promises = [];
|
|
120
129
|
const hotswappableResources = new Array();
|
|
@@ -122,21 +131,26 @@ async function classifyResourceChanges(stackChanges, evaluateCfnTemplate, sdk, n
|
|
|
122
131
|
for (const logicalId of Object.keys(stackChanges.outputs.changes)) {
|
|
123
132
|
nonHotswappableResources.push({
|
|
124
133
|
hotswappable: false,
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
134
|
+
change: {
|
|
135
|
+
reason: payloads_1.NonHotswappableReason.OUTPUT,
|
|
136
|
+
description: 'output was changed',
|
|
137
|
+
subject: {
|
|
138
|
+
type: 'Output',
|
|
139
|
+
logicalId,
|
|
140
|
+
metadata: evaluateCfnTemplate.metadataFor(logicalId),
|
|
141
|
+
},
|
|
142
|
+
},
|
|
129
143
|
});
|
|
130
144
|
}
|
|
131
145
|
// gather the results of the detector functions
|
|
132
146
|
for (const [logicalId, change] of Object.entries(resourceDifferences)) {
|
|
133
|
-
if (
|
|
147
|
+
if (change.newValue?.Type === 'AWS::CloudFormation::Stack' && change.oldValue?.Type === 'AWS::CloudFormation::Stack') {
|
|
134
148
|
const nestedHotswappableResources = await findNestedHotswappableChanges(logicalId, change, nestedStackNames, evaluateCfnTemplate, sdk, hotswapPropertyOverrides);
|
|
135
|
-
hotswappableResources.push(...nestedHotswappableResources.
|
|
136
|
-
nonHotswappableResources.push(...nestedHotswappableResources.
|
|
149
|
+
hotswappableResources.push(...nestedHotswappableResources.hotswappable);
|
|
150
|
+
nonHotswappableResources.push(...nestedHotswappableResources.nonHotswappable);
|
|
137
151
|
continue;
|
|
138
152
|
}
|
|
139
|
-
const hotswappableChangeCandidate = isCandidateForHotswapping(change,
|
|
153
|
+
const hotswappableChangeCandidate = isCandidateForHotswapping(logicalId, change, evaluateCfnTemplate);
|
|
140
154
|
// we don't need to run this through the detector functions, we can already judge this
|
|
141
155
|
if ('hotswappable' in hotswappableChangeCandidate) {
|
|
142
156
|
if (!hotswappableChangeCandidate.hotswappable) {
|
|
@@ -150,7 +164,7 @@ async function classifyResourceChanges(stackChanges, evaluateCfnTemplate, sdk, n
|
|
|
150
164
|
promises.push(() => RESOURCE_DETECTORS[resourceType](logicalId, hotswappableChangeCandidate, evaluateCfnTemplate, hotswapPropertyOverrides));
|
|
151
165
|
}
|
|
152
166
|
else {
|
|
153
|
-
(0, common_1.
|
|
167
|
+
nonHotswappableResources.push((0, common_1.nonHotswappableResource)(hotswappableChangeCandidate));
|
|
154
168
|
}
|
|
155
169
|
}
|
|
156
170
|
// resolve all detector results
|
|
@@ -169,8 +183,8 @@ async function classifyResourceChanges(stackChanges, evaluateCfnTemplate, sdk, n
|
|
|
169
183
|
}
|
|
170
184
|
}
|
|
171
185
|
return {
|
|
172
|
-
|
|
173
|
-
|
|
186
|
+
hotswappable: hotswappableResources,
|
|
187
|
+
nonHotswappable: nonHotswappableResources,
|
|
174
188
|
};
|
|
175
189
|
}
|
|
176
190
|
/**
|
|
@@ -219,23 +233,28 @@ function filterDict(dict, func) {
|
|
|
219
233
|
}
|
|
220
234
|
/** Finds any hotswappable changes in all nested stacks. */
|
|
221
235
|
async function findNestedHotswappableChanges(logicalId, change, nestedStackTemplates, evaluateCfnTemplate, sdk, hotswapPropertyOverrides) {
|
|
222
|
-
var _a, _b;
|
|
223
236
|
const nestedStack = nestedStackTemplates[logicalId];
|
|
224
237
|
if (!nestedStack.physicalName) {
|
|
225
238
|
return {
|
|
226
|
-
|
|
227
|
-
|
|
239
|
+
hotswappable: [],
|
|
240
|
+
nonHotswappable: [
|
|
228
241
|
{
|
|
229
242
|
hotswappable: false,
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
243
|
+
change: {
|
|
244
|
+
reason: payloads_1.NonHotswappableReason.NESTED_STACK_CREATION,
|
|
245
|
+
description: 'newly created nested stacks cannot be hotswapped',
|
|
246
|
+
subject: {
|
|
247
|
+
type: 'Resource',
|
|
248
|
+
logicalId,
|
|
249
|
+
resourceType: 'AWS::CloudFormation::Stack',
|
|
250
|
+
metadata: evaluateCfnTemplate.metadataFor(logicalId),
|
|
251
|
+
},
|
|
252
|
+
},
|
|
234
253
|
},
|
|
235
254
|
],
|
|
236
255
|
};
|
|
237
256
|
}
|
|
238
|
-
const evaluateNestedCfnTemplate = await evaluateCfnTemplate.createNestedEvaluateCloudFormationTemplate(nestedStack.physicalName, nestedStack.generatedTemplate,
|
|
257
|
+
const evaluateNestedCfnTemplate = await evaluateCfnTemplate.createNestedEvaluateCloudFormationTemplate(nestedStack.physicalName, nestedStack.generatedTemplate, change.newValue?.Properties?.Parameters);
|
|
239
258
|
const nestedDiff = cfn_diff.fullDiff(nestedStackTemplates[logicalId].deployedTemplate, nestedStackTemplates[logicalId].generatedTemplate);
|
|
240
259
|
return classifyResourceChanges(nestedDiff, evaluateNestedCfnTemplate, sdk, nestedStackTemplates[logicalId].nestedStackTemplates, hotswapPropertyOverrides);
|
|
241
260
|
}
|
|
@@ -262,35 +281,53 @@ function makeRenameDifference(remChange, addChange) {
|
|
|
262
281
|
* Returns an empty `HotswappableChange` if the change is to CDK::Metadata
|
|
263
282
|
* Returns a `NonHotswappableChange` if the change is not hotswappable
|
|
264
283
|
*/
|
|
265
|
-
function isCandidateForHotswapping(change,
|
|
266
|
-
var _a, _b, _c, _d, _e;
|
|
284
|
+
function isCandidateForHotswapping(logicalId, change, evaluateCfnTemplate) {
|
|
267
285
|
// a resource has been removed OR a resource has been added; we can't short-circuit that change
|
|
268
286
|
if (!change.oldValue) {
|
|
269
287
|
return {
|
|
270
288
|
hotswappable: false,
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
|
|
289
|
+
change: {
|
|
290
|
+
reason: payloads_1.NonHotswappableReason.RESOURCE_CREATION,
|
|
291
|
+
description: `resource '${logicalId}' was created by this deployment`,
|
|
292
|
+
subject: {
|
|
293
|
+
type: 'Resource',
|
|
294
|
+
logicalId,
|
|
295
|
+
resourceType: change.newValue.Type,
|
|
296
|
+
metadata: evaluateCfnTemplate.metadataFor(logicalId),
|
|
297
|
+
},
|
|
298
|
+
},
|
|
275
299
|
};
|
|
276
300
|
}
|
|
277
301
|
else if (!change.newValue) {
|
|
278
302
|
return {
|
|
279
303
|
hotswappable: false,
|
|
280
|
-
resourceType: change.oldValue.Type,
|
|
281
304
|
logicalId,
|
|
282
|
-
|
|
283
|
-
|
|
305
|
+
change: {
|
|
306
|
+
reason: payloads_1.NonHotswappableReason.RESOURCE_DELETION,
|
|
307
|
+
description: `resource '${logicalId}' was destroyed by this deployment`,
|
|
308
|
+
subject: {
|
|
309
|
+
type: 'Resource',
|
|
310
|
+
logicalId,
|
|
311
|
+
resourceType: change.oldValue.Type,
|
|
312
|
+
metadata: evaluateCfnTemplate.metadataFor(logicalId),
|
|
313
|
+
},
|
|
314
|
+
},
|
|
284
315
|
};
|
|
285
316
|
}
|
|
286
317
|
// a resource has had its type changed
|
|
287
|
-
if (
|
|
318
|
+
if (change.newValue.Type !== change.oldValue.Type) {
|
|
288
319
|
return {
|
|
289
320
|
hotswappable: false,
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
321
|
+
change: {
|
|
322
|
+
reason: payloads_1.NonHotswappableReason.RESOURCE_TYPE_CHANGED,
|
|
323
|
+
description: `resource '${logicalId}' had its type changed from '${change.oldValue?.Type}' to '${change.newValue?.Type}'`,
|
|
324
|
+
subject: {
|
|
325
|
+
type: 'Resource',
|
|
326
|
+
logicalId,
|
|
327
|
+
resourceType: change.newValue.Type,
|
|
328
|
+
metadata: evaluateCfnTemplate.metadataFor(logicalId),
|
|
329
|
+
},
|
|
330
|
+
},
|
|
294
331
|
};
|
|
295
332
|
}
|
|
296
333
|
return {
|
|
@@ -298,25 +335,26 @@ function isCandidateForHotswapping(change, logicalId) {
|
|
|
298
335
|
oldValue: change.oldValue,
|
|
299
336
|
newValue: change.newValue,
|
|
300
337
|
propertyUpdates: change.propertyUpdates,
|
|
338
|
+
metadata: evaluateCfnTemplate.metadataFor(logicalId),
|
|
301
339
|
};
|
|
302
340
|
}
|
|
303
|
-
async function
|
|
304
|
-
if (hotswappableChanges.length
|
|
305
|
-
|
|
341
|
+
async function applyAllHotswapOperations(sdk, ioSpan, hotswappableChanges) {
|
|
342
|
+
if (hotswappableChanges.length === 0) {
|
|
343
|
+
return Promise.resolve([]);
|
|
306
344
|
}
|
|
345
|
+
await ioSpan.notify(private_1.IO.DEFAULT_TOOLKIT_INFO.msg(`\n${common_1.ICON} hotswapping resources:`));
|
|
307
346
|
const limit = pLimit(10);
|
|
308
347
|
// eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism
|
|
309
348
|
return Promise.all(hotswappableChanges.map(hotswapOperation => limit(() => {
|
|
310
|
-
return
|
|
349
|
+
return applyHotswapOperation(sdk, ioSpan, hotswapOperation);
|
|
311
350
|
})));
|
|
312
351
|
}
|
|
313
|
-
async function
|
|
352
|
+
async function applyHotswapOperation(sdk, ioSpan, hotswapOperation) {
|
|
314
353
|
// note the type of service that was successfully hotswapped in the User-Agent
|
|
315
354
|
const customUserAgent = `cdk-hotswap/success-${hotswapOperation.service}`;
|
|
316
355
|
sdk.appendCustomUserAgent(customUserAgent);
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
}
|
|
356
|
+
const resourceText = (r) => r.description ?? `${r.resourceType} '${r.physicalName ?? r.logicalId}'`;
|
|
357
|
+
await ioSpan.notify(private_1.IO.CDK_TOOLKIT_I5402.msg(hotswapOperation.change.resources.map(r => (0, util_1.format)(` ${common_1.ICON} %s`, chalk.bold(resourceText(r)))).join('\n'), hotswapOperation.change));
|
|
320
358
|
// if the SDK call fails, an error will be thrown by the SDK
|
|
321
359
|
// and will prevent the green 'hotswapped!' text from being displayed
|
|
322
360
|
try {
|
|
@@ -325,15 +363,13 @@ async function applyHotswappableChange(sdk, ioSpan, hotswapOperation) {
|
|
|
325
363
|
catch (e) {
|
|
326
364
|
if (e.name === 'TimeoutError' || e.name === 'AbortError') {
|
|
327
365
|
const result = JSON.parse((0, util_2.formatErrorMessage)(e));
|
|
328
|
-
const error = new
|
|
366
|
+
const error = new api_1.ToolkitError(formatWaiterErrorResult(result));
|
|
329
367
|
error.name = e.name;
|
|
330
368
|
throw error;
|
|
331
369
|
}
|
|
332
370
|
throw e;
|
|
333
371
|
}
|
|
334
|
-
|
|
335
|
-
await ioSpan.notify(private_1.IO.DEFAULT_TOOLKIT_INFO.msg((0, util_1.format)(`${common_1.ICON} %s %s`, chalk.bold(name), chalk.green('hotswapped!'))));
|
|
336
|
-
}
|
|
372
|
+
await ioSpan.notify(private_1.IO.CDK_TOOLKIT_I5403.msg(hotswapOperation.change.resources.map(r => (0, util_1.format)(` ${common_1.ICON} %s %s`, chalk.bold(resourceText(r)), chalk.green('hotswapped!'))).join('\n'), hotswapOperation.change));
|
|
337
373
|
sdk.removeCustomUserAgent(customUserAgent);
|
|
338
374
|
}
|
|
339
375
|
function formatWaiterErrorResult(result) {
|
|
@@ -350,8 +386,8 @@ function formatWaiterErrorResult(result) {
|
|
|
350
386
|
}
|
|
351
387
|
return main;
|
|
352
388
|
}
|
|
353
|
-
async function
|
|
354
|
-
if (
|
|
389
|
+
async function logRejectedChanges(ioSpan, rejectedChanges, hotswapMode) {
|
|
390
|
+
if (rejectedChanges.length === 0) {
|
|
355
391
|
return;
|
|
356
392
|
}
|
|
357
393
|
/**
|
|
@@ -362,8 +398,8 @@ async function logNonHotswappableChanges(ioSpan, nonHotswappableChanges, hotswap
|
|
|
362
398
|
* This logic prevents us from logging that change as non-hotswappable when we hotswap it.
|
|
363
399
|
*/
|
|
364
400
|
if (hotswapMode === 'hotswap-only') {
|
|
365
|
-
|
|
366
|
-
if (
|
|
401
|
+
rejectedChanges = rejectedChanges.filter((change) => change.hotswapOnlyVisible === true);
|
|
402
|
+
if (rejectedChanges.length === 0) {
|
|
367
403
|
return;
|
|
368
404
|
}
|
|
369
405
|
}
|
|
@@ -374,15 +410,32 @@ async function logNonHotswappableChanges(ioSpan, nonHotswappableChanges, hotswap
|
|
|
374
410
|
else {
|
|
375
411
|
messages.push((0, util_1.format)('%s %s', chalk.red('⚠️'), chalk.red('The following non-hotswappable changes were found:')));
|
|
376
412
|
}
|
|
377
|
-
for (const change of
|
|
378
|
-
|
|
379
|
-
messages.push((0, util_1.format)(' logicalID: %s, type: %s, rejected changes: %s, reason: %s', chalk.bold(change.logicalId), chalk.bold(change.resourceType), chalk.bold(change.rejectedChanges), chalk.red(change.reason)));
|
|
380
|
-
}
|
|
381
|
-
else {
|
|
382
|
-
messages.push((0, util_1.format)(' logicalID: %s, type: %s, reason: %s', chalk.bold(change.logicalId), chalk.bold(change.resourceType), chalk.red(change.reason)));
|
|
383
|
-
}
|
|
413
|
+
for (const { change } of rejectedChanges) {
|
|
414
|
+
messages.push(' ' + nonHotswappableChangeMessage(change));
|
|
384
415
|
}
|
|
385
416
|
messages.push(''); // newline
|
|
386
417
|
await ioSpan.notify(private_1.IO.DEFAULT_TOOLKIT_INFO.msg(messages.join('\n')));
|
|
387
418
|
}
|
|
388
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hotswap-deployments.js","sourceRoot":"","sources":["hotswap-deployments.ts"],"names":[],"mappings":"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","sourcesContent":["import { format } from 'util';\nimport * as cfn_diff from '@aws-cdk/cloudformation-diff';\nimport type * as cxapi from '@aws-cdk/cx-api';\nimport type { WaiterResult } from '@smithy/util-waiter';\nimport * as chalk from 'chalk';\nimport type { ResourceChange } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads';\nimport type { IMessageSpan, IoHelper } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';\nimport { IO, SPAN } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';\nimport type { SDK, SdkProvider } from '../aws-auth';\nimport type { CloudFormationStack } from './cloudformation';\nimport type { NestedStackTemplates } from './nested-stack-helpers';\nimport { loadCurrentTemplateWithNestedStacks } from './nested-stack-helpers';\nimport { ToolkitError } from '../../toolkit/error';\nimport { formatErrorMessage } from '../../util';\nimport { EvaluateCloudFormationTemplate } from '../evaluate-cloudformation-template';\nimport { isHotswappableAppSyncChange } from '../hotswap/appsync-mapping-templates';\nimport { isHotswappableCodeBuildProjectChange } from '../hotswap/code-build-projects';\nimport type {\n  ChangeHotswapResult,\n  HotswapOperation,\n  NonHotswappableChange,\n  HotswapPropertyOverrides,\n  ClassifiedResourceChanges,\n  HotswapResult,\n} from '../hotswap/common';\nimport {\n  ICON,\n  reportNonHotswappableChange,\n  reportNonHotswappableResource,\n} from '../hotswap/common';\nimport { isHotswappableEcsServiceChange } from '../hotswap/ecs-services';\nimport { isHotswappableLambdaFunctionChange } from '../hotswap/lambda-functions';\nimport {\n  skipChangeForS3DeployCustomResourcePolicy,\n  isHotswappableS3BucketDeploymentChange,\n} from '../hotswap/s3-bucket-deployments';\nimport { isHotswappableStateMachineChange } from '../hotswap/stepfunctions-state-machines';\nimport { Mode } from '../plugin';\nimport type { SuccessfulDeployStackResult } from './deployment-result';\n\n// Must use a require() otherwise esbuild complains about calling a namespace\n// eslint-disable-next-line @typescript-eslint/no-require-imports,@typescript-eslint/consistent-type-imports\nconst pLimit: typeof import('p-limit') = require('p-limit');\n\ntype HotswapDetector = (\n  logicalId: string,\n  change: ResourceChange,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n) => Promise<ChangeHotswapResult>;\n\ntype HotswapMode = 'hotswap-only' | 'fall-back';\n\nconst RESOURCE_DETECTORS: { [key: string]: HotswapDetector } = {\n  // Lambda\n  'AWS::Lambda::Function': isHotswappableLambdaFunctionChange,\n  'AWS::Lambda::Version': isHotswappableLambdaFunctionChange,\n  'AWS::Lambda::Alias': isHotswappableLambdaFunctionChange,\n\n  // AppSync\n  'AWS::AppSync::Resolver': isHotswappableAppSyncChange,\n  'AWS::AppSync::FunctionConfiguration': isHotswappableAppSyncChange,\n  'AWS::AppSync::GraphQLSchema': isHotswappableAppSyncChange,\n  'AWS::AppSync::ApiKey': isHotswappableAppSyncChange,\n\n  'AWS::ECS::TaskDefinition': isHotswappableEcsServiceChange,\n  'AWS::CodeBuild::Project': isHotswappableCodeBuildProjectChange,\n  'AWS::StepFunctions::StateMachine': isHotswappableStateMachineChange,\n  'Custom::CDKBucketDeployment': isHotswappableS3BucketDeploymentChange,\n  'AWS::IAM::Policy': async (\n    logicalId: string,\n    change: ResourceChange,\n    evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  ): Promise<ChangeHotswapResult> => {\n    // If the policy is for a S3BucketDeploymentChange, we can ignore the change\n    if (await skipChangeForS3DeployCustomResourcePolicy(logicalId, change, evaluateCfnTemplate)) {\n      return [];\n    }\n\n    return reportNonHotswappableResource(change, 'This resource type is not supported for hotswap deployments');\n  },\n\n  'AWS::CDK::Metadata': async () => [],\n};\n\n/**\n * Perform a hotswap deployment, 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,\n  ioHelper: IoHelper,\n  assetParams: { [key: string]: string },\n  cloudFormationStack: CloudFormationStack,\n  stackArtifact: cxapi.CloudFormationStackArtifact,\n  hotswapMode: HotswapMode,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<SuccessfulDeployStackResult | undefined> {\n  const hotswapSpan = await ioHelper.span(SPAN.HOTSWAP).begin({\n    stack: stackArtifact,\n    mode: hotswapMode,\n  });\n\n  const result = await hotswapDeployment(\n    sdkProvider,\n    hotswapSpan,\n    assetParams,\n    stackArtifact,\n    hotswapMode,\n    hotswapPropertyOverrides,\n  );\n\n  await hotswapSpan.end();\n\n  if (result?.hotswapped === true) {\n    return {\n      type: 'did-deploy-stack',\n      noOp: result.hotswappableChanges.length === 0,\n      stackArn: cloudFormationStack.stackId,\n      outputs: cloudFormationStack.outputs,\n    };\n  }\n\n  return undefined;\n}\n\n/**\n * Perform a hotswap deployment, short-circuiting CloudFormation if possible.\n * Returns information about the attempted hotswap deployment\n */\nasync function hotswapDeployment(\n  sdkProvider: SdkProvider,\n  ioSpan: IMessageSpan<any>,\n  assetParams: { [key: string]: string },\n  stack: cxapi.CloudFormationStackArtifact,\n  hotswapMode: HotswapMode,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<HotswapResult> {\n  // resolve the environment, so we can substitute things like AWS::Region in CFN expressions\n  const resolvedEnv = await sdkProvider.resolveEnvironment(stack.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\n  const currentTemplate = await loadCurrentTemplateWithNestedStacks(stack, sdk);\n\n  const evaluateCfnTemplate = new EvaluateCloudFormationTemplate({\n    stackArtifact: stack,\n    parameters: assetParams,\n    account: resolvedEnv.account,\n    region: resolvedEnv.region,\n    partition: (await sdk.currentAccount()).partition,\n    sdk,\n    nestedStacks: currentTemplate.nestedStacks,\n  });\n\n  const stackChanges = cfn_diff.fullDiff(currentTemplate.deployedRootTemplate, stack.template);\n  const { hotswappableChanges, nonHotswappableChanges } = await classifyResourceChanges(\n    stackChanges,\n    evaluateCfnTemplate,\n    sdk,\n    currentTemplate.nestedStacks, hotswapPropertyOverrides,\n  );\n\n  await logNonHotswappableChanges(ioSpan, nonHotswappableChanges, hotswapMode);\n\n  // preserve classic hotswap behavior\n  if (hotswapMode === 'fall-back') {\n    if (nonHotswappableChanges.length > 0) {\n      return {\n        stack,\n        hotswapped: false,\n        hotswappableChanges,\n        nonHotswappableChanges,\n      };\n    }\n  }\n\n  // apply the short-circuitable changes\n  await applyAllHotswappableChanges(sdk, ioSpan, hotswappableChanges);\n\n  return {\n    stack,\n    hotswapped: true,\n    hotswappableChanges,\n    nonHotswappableChanges,\n  };\n}\n\n/**\n * Classifies all changes to all resources as either hotswappable or not.\n * Metadata changes are excluded from the list of (non)hotswappable resources.\n */\nasync function classifyResourceChanges(\n  stackChanges: cfn_diff.TemplateDiff,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  sdk: SDK,\n  nestedStackNames: { [nestedStackName: string]: NestedStackTemplates },\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<ClassifiedResourceChanges> {\n  const resourceDifferences = getStackResourceDifferences(stackChanges);\n\n  const promises: Array<() => Promise<ChangeHotswapResult>> = [];\n  const hotswappableResources = new Array<HotswapOperation>();\n  const nonHotswappableResources = new Array<NonHotswappableChange>();\n  for (const logicalId of Object.keys(stackChanges.outputs.changes)) {\n    nonHotswappableResources.push({\n      hotswappable: false,\n      reason: 'output was changed',\n      logicalId,\n      rejectedChanges: [],\n      resourceType: 'Stack Output',\n    });\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(\n        logicalId,\n        change,\n        nestedStackNames,\n        evaluateCfnTemplate,\n        sdk,\n        hotswapPropertyOverrides,\n      );\n      hotswappableResources.push(...nestedHotswappableResources.hotswappableChanges);\n      nonHotswappableResources.push(...nestedHotswappableResources.nonHotswappableChanges);\n\n      continue;\n    }\n\n    const hotswappableChangeCandidate = isCandidateForHotswapping(change, logicalId);\n    // we don't need to run this through the detector functions, we can already judge this\n    if ('hotswappable' in hotswappableChangeCandidate) {\n      if (!hotswappableChangeCandidate.hotswappable) {\n        nonHotswappableResources.push(hotswappableChangeCandidate);\n      }\n\n      continue;\n    }\n\n    const resourceType: string = hotswappableChangeCandidate.newValue.Type;\n    if (resourceType in RESOURCE_DETECTORS) {\n      // run detector functions lazily to prevent unhandled promise rejections\n      promises.push(() =>\n        RESOURCE_DETECTORS[resourceType](logicalId, hotswappableChangeCandidate, evaluateCfnTemplate, hotswapPropertyOverrides),\n      );\n    } else {\n      reportNonHotswappableChange(\n        nonHotswappableResources,\n        hotswappableChangeCandidate,\n        undefined,\n        'This resource type is not supported for hotswap deployments',\n      );\n    }\n  }\n\n  // resolve all detector results\n  const changesDetectionResults: Array<ChangeHotswapResult> = [];\n  for (const detectorResultPromises of promises) {\n    // Constant set of promises per resource\n    // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n    const hotswapDetectionResults = await Promise.all(await detectorResultPromises());\n    changesDetectionResults.push(hotswapDetectionResults);\n  }\n\n  for (const resourceDetectionResults of changesDetectionResults) {\n    for (const propertyResult of resourceDetectionResults) {\n      propertyResult.hotswappable\n        ? hotswappableResources.push(propertyResult)\n        : nonHotswappableResources.push(propertyResult);\n    }\n  }\n\n  return {\n    hotswappableChanges: hotswappableResources,\n    nonHotswappableChanges: nonHotswappableResources,\n  };\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): {\n  [logicalId: string]: cfn_diff.ResourceDifference;\n} {\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(\n    (acc, [key, t]) => {\n      if (func(t)) {\n        acc[key] = t;\n      }\n      return acc;\n    },\n    {} as { [key: string]: T },\n  );\n}\n\n/** Finds any hotswappable changes in all nested stacks. */\nasync function findNestedHotswappableChanges(\n  logicalId: string,\n  change: cfn_diff.ResourceDifference,\n  nestedStackTemplates: { [nestedStackName: string]: NestedStackTemplates },\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  sdk: SDK,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<ClassifiedResourceChanges> {\n  const nestedStack = nestedStackTemplates[logicalId];\n  if (!nestedStack.physicalName) {\n    return {\n      hotswappableChanges: [],\n      nonHotswappableChanges: [\n        {\n          hotswappable: false,\n          logicalId,\n          reason: `physical name for AWS::CloudFormation::Stack '${logicalId}' could not be found in CloudFormation, so this is a newly created nested stack and cannot be hotswapped`,\n          rejectedChanges: [],\n          resourceType: 'AWS::CloudFormation::Stack',\n        },\n      ],\n    };\n  }\n\n  const evaluateNestedCfnTemplate = await evaluateCfnTemplate.createNestedEvaluateCloudFormationTemplate(\n    nestedStack.physicalName,\n    nestedStack.generatedTemplate,\n    change.newValue?.Properties?.Parameters,\n  );\n\n  const nestedDiff = cfn_diff.fullDiff(\n    nestedStackTemplates[logicalId].deployedTemplate,\n    nestedStackTemplates[logicalId].generatedTemplate,\n  );\n\n  return classifyResourceChanges(\n    nestedDiff,\n    evaluateNestedCfnTemplate,\n    sdk,\n    nestedStackTemplates[logicalId].nestedStackTemplates,\n    hotswapPropertyOverrides,\n  );\n}\n\n/** Returns 'true' if a pair of changes is for the same resource. */\nfunction changesAreForSameResource(\n  oldChange: cfn_diff.ResourceDifference,\n  newChange: cfn_diff.ResourceDifference,\n): boolean {\n  return (\n    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}\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 a `HotswappableChangeCandidate` if the change is hotswappable\n * Returns an empty `HotswappableChange` if the change is to CDK::Metadata\n * Returns a `NonHotswappableChange` if the change is not hotswappable\n */\nfunction isCandidateForHotswapping(\n  change: cfn_diff.ResourceDifference,\n  logicalId: string,\n): HotswapOperation | NonHotswappableChange | ResourceChange {\n  // a resource has been removed OR a resource has been added; we can't short-circuit that change\n  if (!change.oldValue) {\n    return {\n      hotswappable: false,\n      resourceType: change.newValue!.Type,\n      logicalId,\n      rejectedChanges: [],\n      reason: `resource '${logicalId}' was created by this deployment`,\n    };\n  } else if (!change.newValue) {\n    return {\n      hotswappable: false,\n      resourceType: change.oldValue!.Type,\n      logicalId,\n      rejectedChanges: [],\n      reason: `resource '${logicalId}' was destroyed by this deployment`,\n    };\n  }\n\n  // a resource has had its type changed\n  if (change.newValue?.Type !== change.oldValue?.Type) {\n    return {\n      hotswappable: false,\n      resourceType: change.newValue?.Type,\n      logicalId,\n      rejectedChanges: [],\n      reason: `resource '${logicalId}' had its type changed from '${change.oldValue?.Type}' to '${change.newValue?.Type}'`,\n    };\n  }\n\n  return {\n    logicalId,\n    oldValue: change.oldValue,\n    newValue: change.newValue,\n    propertyUpdates: change.propertyUpdates,\n  };\n}\n\nasync function applyAllHotswappableChanges(sdk: SDK, ioSpan: IMessageSpan<any>, hotswappableChanges: HotswapOperation[]): Promise<void[]> {\n  if (hotswappableChanges.length > 0) {\n    await ioSpan.notify(IO.DEFAULT_TOOLKIT_INFO.msg(`\\n${ICON} hotswapping resources:`));\n  }\n  const limit = pLimit(10);\n  // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n  return Promise.all(hotswappableChanges.map(hotswapOperation => limit(() => {\n    return applyHotswappableChange(sdk, ioSpan, hotswapOperation);\n  })));\n}\n\nasync function applyHotswappableChange(sdk: SDK, ioSpan: IMessageSpan<any>, hotswapOperation: HotswapOperation): Promise<void> {\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  for (const name of hotswapOperation.resourceNames) {\n    await ioSpan.notify(IO.DEFAULT_TOOLKIT_INFO.msg(format(`   ${ICON} %s`, chalk.bold(name))));\n  }\n\n  // if the SDK call fails, an error will be thrown by the SDK\n  // and will prevent the green 'hotswapped!' text from being displayed\n  try {\n    await hotswapOperation.apply(sdk);\n  } catch (e: any) {\n    if (e.name === 'TimeoutError' || e.name === 'AbortError') {\n      const result: WaiterResult = JSON.parse(formatErrorMessage(e));\n      const error = new ToolkitError(formatWaiterErrorResult(result));\n      error.name = e.name;\n      throw error;\n    }\n    throw e;\n  }\n\n  for (const name of hotswapOperation.resourceNames) {\n    await ioSpan.notify(IO.DEFAULT_TOOLKIT_INFO.msg(format(`${ICON} %s %s`, chalk.bold(name), chalk.green('hotswapped!'))));\n  }\n\n  sdk.removeCustomUserAgent(customUserAgent);\n}\n\nfunction formatWaiterErrorResult(result: WaiterResult) {\n  const main = [\n    `Resource is not in the expected state due to waiter status: ${result.state}`,\n    result.reason ? `${result.reason}.` : '',\n  ].join('. ');\n\n  if (result.observedResponses != null) {\n    const observedResponses = Object\n      .entries(result.observedResponses)\n      .map(([msg, count]) => `  - ${msg} (${count})`)\n      .join('\\n');\n\n    return `${main} Observed responses:\\n${observedResponses}`;\n  }\n\n  return main;\n}\n\nasync function logNonHotswappableChanges(\n  ioSpan: IMessageSpan<any>,\n  nonHotswappableChanges: NonHotswappableChange[],\n  hotswapMode: HotswapMode,\n): Promise<void> {\n  if (nonHotswappableChanges.length === 0) {\n    return;\n  }\n  /**\n   * EKS 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  if (hotswapMode === 'hotswap-only') {\n    nonHotswappableChanges = nonHotswappableChanges.filter((change) => change.hotswapOnlyVisible === true);\n\n    if (nonHotswappableChanges.length === 0) {\n      return;\n    }\n  }\n\n  const messages = ['']; // start with empty line\n\n  if (hotswapMode === 'hotswap-only') {\n    messages.push(format('%s %s', chalk.red('⚠️'), chalk.red('The following non-hotswappable changes were found. To reconcile these using CloudFormation, specify --hotswap-fallback')));\n  } else {\n    messages.push(format('%s %s', chalk.red('⚠️'), chalk.red('The following non-hotswappable changes were found:')));\n  }\n\n  for (const change of nonHotswappableChanges) {\n    if (change.rejectedChanges.length > 0) {\n      messages.push(format(\n        '    logicalID: %s, type: %s, rejected changes: %s, reason: %s',\n        chalk.bold(change.logicalId),\n        chalk.bold(change.resourceType),\n        chalk.bold(change.rejectedChanges),\n        chalk.red(change.reason),\n      ));\n    } else {\n      messages.push(format(\n        '    logicalID: %s, type: %s, reason: %s',\n        chalk.bold(change.logicalId),\n        chalk.bold(change.resourceType),\n        chalk.red(change.reason),\n      ));\n    }\n  }\n  messages.push(''); // newline\n\n  await ioSpan.notify(IO.DEFAULT_TOOLKIT_INFO.msg(messages.join('\\n')));\n}\n"]}
|
|
419
|
+
/**
|
|
420
|
+
* Formats a NonHotswappableChange
|
|
421
|
+
*/
|
|
422
|
+
function nonHotswappableChangeMessage(change) {
|
|
423
|
+
const subject = change.subject;
|
|
424
|
+
const reason = change.description ?? change.reason;
|
|
425
|
+
switch (subject.type) {
|
|
426
|
+
case 'Output':
|
|
427
|
+
return (0, util_1.format)('output: %s, reason: %s', chalk.bold(subject.logicalId), chalk.red(reason));
|
|
428
|
+
case 'Resource':
|
|
429
|
+
return nonHotswappableResourceMessage(subject, reason);
|
|
430
|
+
}
|
|
431
|
+
}
|
|
432
|
+
/**
|
|
433
|
+
* Formats a non-hotswappable resource subject
|
|
434
|
+
*/
|
|
435
|
+
function nonHotswappableResourceMessage(subject, reason) {
|
|
436
|
+
if (subject.rejectedProperties?.length) {
|
|
437
|
+
return (0, util_1.format)('resource: %s, type: %s, rejected changes: %s, reason: %s', chalk.bold(subject.logicalId), chalk.bold(subject.resourceType), chalk.bold(subject.rejectedProperties), chalk.red(reason));
|
|
438
|
+
}
|
|
439
|
+
return (0, util_1.format)('resource: %s, type: %s, reason: %s', chalk.bold(subject.logicalId), chalk.bold(subject.resourceType), chalk.red(reason));
|
|
440
|
+
}
|
|
441
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hotswap-deployments.js","sourceRoot":"","sources":["hotswap-deployments.ts"],"names":[],"mappings":";;AAuFA,oDAmCC;AA1HD,+BAA8B;AAC9B,yDAAyD;AAGzD,+BAA+B;AAE/B,0EAAgF;AAEhF,2FAAqG;AAErG,yFAAuF;AACvF,qCAAgD;AAGhD,sDAAwG;AACxG,oFAAmF;AACnF,wEAAsF;AAOtF,8CAG2B;AAC3B,0DAAyE;AACzE,kEAAiF;AACjF,4EAG0C;AAC1C,0FAA2F;AAC3F,sCAAiC;AAEjC,6EAA6E;AAC7E,4GAA4G;AAC5G,MAAM,MAAM,GAA6B,OAAO,CAAC,SAAS,CAAC,CAAC;AAW5D,MAAM,kBAAkB,GAAuC;IAC7D,SAAS;IACT,uBAAuB,EAAE,qDAAkC;IAC3D,sBAAsB,EAAE,qDAAkC;IAC1D,oBAAoB,EAAE,qDAAkC;IAExD,UAAU;IACV,wBAAwB,EAAE,uDAA2B;IACrD,qCAAqC,EAAE,uDAA2B;IAClE,6BAA6B,EAAE,uDAA2B;IAC1D,sBAAsB,EAAE,uDAA2B;IAEnD,0BAA0B,EAAE,6CAA8B;IAC1D,yBAAyB,EAAE,0DAAoC;IAC/D,kCAAkC,EAAE,+DAAgC;IACpE,6BAA6B,EAAE,8DAAsC;IACrE,kBAAkB,EAAE,KAAK,EACvB,SAAiB,EACjB,MAAsB,EACtB,mBAAmD,EACzB,EAAE;QAC5B,4EAA4E;QAC5E,IAAI,MAAM,IAAA,iEAAyC,EAAC,SAAS,EAAE,MAAM,EAAE,mBAAmB,CAAC,EAAE,CAAC;YAC5F,OAAO,EAAE,CAAC;QACZ,CAAC;QAED,OAAO,CAAC,IAAA,gCAAuB,EAAC,MAAM,CAAC,CAAC,CAAC;IAC3C,CAAC;IAED,oBAAoB,EAAE,KAAK,IAAI,EAAE,CAAC,EAAE;CACrC,CAAC;AAEF;;;;;GAKG;AACI,KAAK,UAAU,oBAAoB,CACxC,WAAwB,EACxB,QAAkB,EAClB,WAAsC,EACtC,mBAAwC,EACxC,aAAgD,EAChD,WAAwB,EACxB,wBAAkD;IAElD,MAAM,WAAW,GAAG,MAAM,QAAQ,CAAC,IAAI,CAAC,cAAI,CAAC,OAAO,CAAC,CAAC,KAAK,CAAC;QAC1D,KAAK,EAAE,aAAa;QACpB,IAAI,EAAE,WAAW;KAClB,CAAC,CAAC;IAEH,MAAM,MAAM,GAAG,MAAM,iBAAiB,CACpC,WAAW,EACX,WAAW,EACX,WAAW,EACX,aAAa,EACb,WAAW,EACX,wBAAwB,CACzB,CAAC;IAEF,MAAM,WAAW,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;IAE9B,IAAI,MAAM,EAAE,UAAU,KAAK,IAAI,EAAE,CAAC;QAChC,OAAO;YACL,IAAI,EAAE,kBAAkB;YACxB,IAAI,EAAE,MAAM,CAAC,mBAAmB,CAAC,MAAM,KAAK,CAAC;YAC7C,QAAQ,EAAE,mBAAmB,CAAC,OAAO;YACrC,OAAO,EAAE,mBAAmB,CAAC,OAAO;SACrC,CAAC;IACJ,CAAC;IAED,OAAO,SAAS,CAAC;AACnB,CAAC;AAED;;;GAGG;AACH,KAAK,UAAU,iBAAiB,CAC9B,WAAwB,EACxB,MAAyB,EACzB,WAAsC,EACtC,KAAwC,EACxC,WAAwB,EACxB,wBAAkD;IAElD,2FAA2F;IAC3F,MAAM,WAAW,GAAG,MAAM,WAAW,CAAC,kBAAkB,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;IAC5E,8GAA8G;IAC9G,kGAAkG;IAClG,MAAM,GAAG,GAAG,CAAC,MAAM,WAAW,CAAC,cAAc,CAAC,WAAW,EAAE,aAAI,CAAC,UAAU,CAAC,CAAC,CAAC,GAAG,CAAC;IAEjF,MAAM,eAAe,GAAG,MAAM,IAAA,oDAAmC,EAAC,KAAK,EAAE,GAAG,CAAC,CAAC;IAE9E,MAAM,mBAAmB,GAAG,IAAI,+CAA8B,CAAC;QAC7D,aAAa,EAAE,KAAK;QACpB,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,GAAG;QACH,YAAY,EAAE,eAAe,CAAC,YAAY;KAC3C,CAAC,CAAC;IAEH,MAAM,YAAY,GAAG,QAAQ,CAAC,QAAQ,CAAC,eAAe,CAAC,oBAAoB,EAAE,KAAK,CAAC,QAAQ,CAAC,CAAC;IAC7F,MAAM,EAAE,YAAY,EAAE,eAAe,EAAE,GAAG,MAAM,uBAAuB,CACrE,YAAY,EACZ,mBAAmB,EACnB,GAAG,EACH,eAAe,CAAC,YAAY,EAAE,wBAAwB,CACvD,CAAC;IAEF,MAAM,kBAAkB,CAAC,MAAM,EAAE,eAAe,EAAE,WAAW,CAAC,CAAC;IAE/D,MAAM,mBAAmB,GAAG,YAAY,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC;IAC5D,MAAM,sBAAsB,GAAG,eAAe,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC;IAElE,MAAM,MAAM,CAAC,MAAM,CAAC,YAAE,CAAC,iBAAiB,CAAC,GAAG,CAAC,sBAAsB,EAAE;QACnE,KAAK;QACL,IAAI,EAAE,WAAW;QACjB,mBAAmB;QACnB,sBAAsB;KACvB,CAAC,CAAC,CAAC;IAEJ,oCAAoC;IACpC,IAAI,WAAW,KAAK,WAAW,EAAE,CAAC;QAChC,IAAI,sBAAsB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACtC,OAAO;gBACL,KAAK;gBACL,IAAI,EAAE,WAAW;gBACjB,UAAU,EAAE,KAAK;gBACjB,mBAAmB;gBACnB,sBAAsB;aACvB,CAAC;QACJ,CAAC;IACH,CAAC;IAED,sCAAsC;IACtC,MAAM,yBAAyB,CAAC,GAAG,EAAE,MAAM,EAAE,YAAY,CAAC,CAAC;IAE3D,OAAO;QACL,KAAK;QACL,IAAI,EAAE,WAAW;QACjB,UAAU,EAAE,IAAI;QAChB,mBAAmB;QACnB,sBAAsB;KACvB,CAAC;AACJ,CAAC;AAOD;;;GAGG;AACH,KAAK,UAAU,uBAAuB,CACpC,YAAmC,EACnC,mBAAmD,EACnD,GAAQ,EACR,gBAAqE,EACrE,wBAAkD;IAElD,MAAM,mBAAmB,GAAG,2BAA2B,CAAC,YAAY,CAAC,CAAC;IAEtE,MAAM,QAAQ,GAA0C,EAAE,CAAC;IAC3D,MAAM,qBAAqB,GAAG,IAAI,KAAK,EAAoB,CAAC;IAC5D,MAAM,wBAAwB,GAAG,IAAI,KAAK,EAAkB,CAAC;IAC7D,KAAK,MAAM,SAAS,IAAI,MAAM,CAAC,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;QAClE,wBAAwB,CAAC,IAAI,CAAC;YAC5B,YAAY,EAAE,KAAK;YACnB,MAAM,EAAE;gBACN,MAAM,EAAE,gCAAqB,CAAC,MAAM;gBACpC,WAAW,EAAE,oBAAoB;gBACjC,OAAO,EAAE;oBACP,IAAI,EAAE,QAAQ;oBACd,SAAS;oBACT,QAAQ,EAAE,mBAAmB,CAAC,WAAW,CAAC,SAAS,CAAC;iBACrD;aACF;SACF,CAAC,CAAC;IACL,CAAC;IACD,+CAA+C;IAC/C,KAAK,MAAM,CAAC,SAAS,EAAE,MAAM,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,mBAAmB,CAAC,EAAE,CAAC;QACtE,IAAI,MAAM,CAAC,QAAQ,EAAE,IAAI,KAAK,4BAA4B,IAAI,MAAM,CAAC,QAAQ,EAAE,IAAI,KAAK,4BAA4B,EAAE,CAAC;YACrH,MAAM,2BAA2B,GAAG,MAAM,6BAA6B,CACrE,SAAS,EACT,MAAM,EACN,gBAAgB,EAChB,mBAAmB,EACnB,GAAG,EACH,wBAAwB,CACzB,CAAC;YACF,qBAAqB,CAAC,IAAI,CAAC,GAAG,2BAA2B,CAAC,YAAY,CAAC,CAAC;YACxE,wBAAwB,CAAC,IAAI,CAAC,GAAG,2BAA2B,CAAC,eAAe,CAAC,CAAC;YAE9E,SAAS;QACX,CAAC;QAED,MAAM,2BAA2B,GAAG,yBAAyB,CAAC,SAAS,EAAE,MAAM,EAAE,mBAAmB,CAAC,CAAC;QACtG,sFAAsF;QACtF,IAAI,cAAc,IAAI,2BAA2B,EAAE,CAAC;YAClD,IAAI,CAAC,2BAA2B,CAAC,YAAY,EAAE,CAAC;gBAC9C,wBAAwB,CAAC,IAAI,CAAC,2BAA2B,CAAC,CAAC;YAC7D,CAAC;YAED,SAAS;QACX,CAAC;QAED,MAAM,YAAY,GAAW,2BAA2B,CAAC,QAAQ,CAAC,IAAI,CAAC;QACvE,IAAI,YAAY,IAAI,kBAAkB,EAAE,CAAC;YACvC,wEAAwE;YACxE,QAAQ,CAAC,IAAI,CAAC,GAAG,EAAE,CACjB,kBAAkB,CAAC,YAAY,CAAC,CAAC,SAAS,EAAE,2BAA2B,EAAE,mBAAmB,EAAE,wBAAwB,CAAC,CACxH,CAAC;QACJ,CAAC;aAAM,CAAC;YACN,wBAAwB,CAAC,IAAI,CAAC,IAAA,gCAAuB,EAAC,2BAA2B,CAAC,CAAC,CAAC;QACtF,CAAC;IACH,CAAC;IAED,+BAA+B;IAC/B,MAAM,uBAAuB,GAA2B,EAAE,CAAC;IAC3D,KAAK,MAAM,sBAAsB,IAAI,QAAQ,EAAE,CAAC;QAC9C,wCAAwC;QACxC,wEAAwE;QACxE,MAAM,uBAAuB,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,MAAM,sBAAsB,EAAE,CAAC,CAAC;QAClF,uBAAuB,CAAC,IAAI,CAAC,uBAAuB,CAAC,CAAC;IACxD,CAAC;IAED,KAAK,MAAM,wBAAwB,IAAI,uBAAuB,EAAE,CAAC;QAC/D,KAAK,MAAM,cAAc,IAAI,wBAAwB,EAAE,CAAC;YACtD,cAAc,CAAC,YAAY;gBACzB,CAAC,CAAC,qBAAqB,CAAC,IAAI,CAAC,cAAc,CAAC;gBAC5C,CAAC,CAAC,wBAAwB,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;QACpD,CAAC;IACH,CAAC;IAED,OAAO;QACL,YAAY,EAAE,qBAAqB;QACnC,eAAe,EAAE,wBAAwB;KAC1C,CAAC;AACJ,CAAC;AAED;;;;GAIG;AACH,SAAS,2BAA2B,CAAC,YAAmC;IAGtE,iEAAiE;IACjE,iGAAiG;IACjG,MAAM,kBAAkB,GAAqD,YAAY,CAAC,SAAS,CAAC,OAAO,CAAC;IAC5G,MAAM,iBAAiB,GAAG,UAAU,CAAC,kBAAkB,EAAE,CAAC,SAAS,EAAE,EAAE,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;IAC7F,MAAM,oBAAoB,GAAG,UAAU,CAAC,kBAAkB,EAAE,CAAC,SAAS,EAAE,EAAE,CAAC,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;IACjG,KAAK,MAAM,CAAC,KAAK,EAAE,gBAAgB,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,oBAAoB,CAAC,EAAE,CAAC;QAC7E,IAAI,gBAAgB,CAAC,UAAU,EAAE,CAAC;YAChC,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,CAAC;gBAC3B,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;YACzC,CAAC;QACH,CAAC;IACH,CAAC;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,CAChC,CAAC,GAAG,EAAE,CAAC,GAAG,EAAE,CAAC,CAAC,EAAE,EAAE;QAChB,IAAI,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC;YACZ,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QACf,CAAC;QACD,OAAO,GAAG,CAAC;IACb,CAAC,EACD,EAA0B,CAC3B,CAAC;AACJ,CAAC;AAED,2DAA2D;AAC3D,KAAK,UAAU,6BAA6B,CAC1C,SAAiB,EACjB,MAAmC,EACnC,oBAAyE,EACzE,mBAAmD,EACnD,GAAQ,EACR,wBAAkD;IAElD,MAAM,WAAW,GAAG,oBAAoB,CAAC,SAAS,CAAC,CAAC;IACpD,IAAI,CAAC,WAAW,CAAC,YAAY,EAAE,CAAC;QAC9B,OAAO;YACL,YAAY,EAAE,EAAE;YAChB,eAAe,EAAE;gBACf;oBACE,YAAY,EAAE,KAAK;oBACnB,MAAM,EAAE;wBACN,MAAM,EAAE,gCAAqB,CAAC,qBAAqB;wBACnD,WAAW,EAAE,kDAAkD;wBAC/D,OAAO,EAAE;4BACP,IAAI,EAAE,UAAU;4BAChB,SAAS;4BACT,YAAY,EAAE,4BAA4B;4BAC1C,QAAQ,EAAE,mBAAmB,CAAC,WAAW,CAAC,SAAS,CAAC;yBACrD;qBACF;iBACF;aACF;SACF,CAAC;IACJ,CAAC;IAED,MAAM,yBAAyB,GAAG,MAAM,mBAAmB,CAAC,0CAA0C,CACpG,WAAW,CAAC,YAAY,EACxB,WAAW,CAAC,iBAAiB,EAC7B,MAAM,CAAC,QAAQ,EAAE,UAAU,EAAE,UAAU,CACxC,CAAC;IAEF,MAAM,UAAU,GAAG,QAAQ,CAAC,QAAQ,CAClC,oBAAoB,CAAC,SAAS,CAAC,CAAC,gBAAgB,EAChD,oBAAoB,CAAC,SAAS,CAAC,CAAC,iBAAiB,CAClD,CAAC;IAEF,OAAO,uBAAuB,CAC5B,UAAU,EACV,yBAAyB,EACzB,GAAG,EACH,oBAAoB,CAAC,SAAS,CAAC,CAAC,oBAAoB,EACpD,wBAAwB,CACzB,CAAC;AACJ,CAAC;AAED,oEAAoE;AACpE,SAAS,yBAAyB,CAChC,SAAsC,EACtC,SAAsC;IAEtC,OAAO,CACL,SAAS,CAAC,eAAe,KAAK,SAAS,CAAC,eAAe;QACvD,oGAAoG;QACpG,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,aAAa,CAAC,KAAK,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,aAAa,CAAC,CACpF,CAAC;AACJ,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;;;;GAIG;AACH,SAAS,yBAAyB,CAChC,SAAiB,EACjB,MAAmC,EACnC,mBAAmD;IAEnD,+FAA+F;IAC/F,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC;QACrB,OAAO;YACL,YAAY,EAAE,KAAK;YACnB,MAAM,EAAE;gBACN,MAAM,EAAE,gCAAqB,CAAC,iBAAiB;gBAC/C,WAAW,EAAE,aAAa,SAAS,kCAAkC;gBACrE,OAAO,EAAE;oBACP,IAAI,EAAE,UAAU;oBAChB,SAAS;oBACT,YAAY,EAAE,MAAM,CAAC,QAAS,CAAC,IAAI;oBACnC,QAAQ,EAAE,mBAAmB,CAAC,WAAW,CAAC,SAAS,CAAC;iBACrD;aACF;SACF,CAAC;IACJ,CAAC;SAAM,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC;QAC5B,OAAO;YACL,YAAY,EAAE,KAAK;YACnB,SAAS;YACT,MAAM,EAAE;gBACN,MAAM,EAAE,gCAAqB,CAAC,iBAAiB;gBAC/C,WAAW,EAAE,aAAa,SAAS,oCAAoC;gBACvE,OAAO,EAAE;oBACP,IAAI,EAAE,UAAU;oBAChB,SAAS;oBACT,YAAY,EAAE,MAAM,CAAC,QAAQ,CAAC,IAAI;oBAClC,QAAQ,EAAE,mBAAmB,CAAC,WAAW,CAAC,SAAS,CAAC;iBACrD;aACF;SACF,CAAC;IACJ,CAAC;IAED,sCAAsC;IACtC,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,KAAK,MAAM,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QAClD,OAAO;YACL,YAAY,EAAE,KAAK;YACnB,MAAM,EAAE;gBACN,MAAM,EAAE,gCAAqB,CAAC,qBAAqB;gBACnD,WAAW,EAAE,aAAa,SAAS,gCAAgC,MAAM,CAAC,QAAQ,EAAE,IAAI,SAAS,MAAM,CAAC,QAAQ,EAAE,IAAI,GAAG;gBACzH,OAAO,EAAE;oBACP,IAAI,EAAE,UAAU;oBAChB,SAAS;oBACT,YAAY,EAAE,MAAM,CAAC,QAAQ,CAAC,IAAI;oBAClC,QAAQ,EAAE,mBAAmB,CAAC,WAAW,CAAC,SAAS,CAAC;iBACrD;aACF;SACF,CAAC;IACJ,CAAC;IAED,OAAO;QACL,SAAS;QACT,QAAQ,EAAE,MAAM,CAAC,QAAQ;QACzB,QAAQ,EAAE,MAAM,CAAC,QAAQ;QACzB,eAAe,EAAE,MAAM,CAAC,eAAe;QACvC,QAAQ,EAAE,mBAAmB,CAAC,WAAW,CAAC,SAAS,CAAC;KACrD,CAAC;AACJ,CAAC;AAED,KAAK,UAAU,yBAAyB,CAAC,GAAQ,EAAE,MAAyB,EAAE,mBAAuC;IACnH,IAAI,mBAAmB,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACrC,OAAO,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;IAC7B,CAAC;IAED,MAAM,MAAM,CAAC,MAAM,CAAC,YAAE,CAAC,oBAAoB,CAAC,GAAG,CAAC,KAAK,aAAI,yBAAyB,CAAC,CAAC,CAAC;IACrF,MAAM,KAAK,GAAG,MAAM,CAAC,EAAE,CAAC,CAAC;IACzB,wEAAwE;IACxE,OAAO,OAAO,CAAC,GAAG,CAAC,mBAAmB,CAAC,GAAG,CAAC,gBAAgB,CAAC,EAAE,CAAC,KAAK,CAAC,GAAG,EAAE;QACxE,OAAO,qBAAqB,CAAC,GAAG,EAAE,MAAM,EAAE,gBAAgB,CAAC,CAAC;IAC9D,CAAC,CAAC,CAAC,CAAC,CAAC;AACP,CAAC;AAED,KAAK,UAAU,qBAAqB,CAAC,GAAQ,EAAE,MAAyB,EAAE,gBAAkC;IAC1G,8EAA8E;IAC9E,MAAM,eAAe,GAAG,uBAAuB,gBAAgB,CAAC,OAAO,EAAE,CAAC;IAC1E,GAAG,CAAC,qBAAqB,CAAC,eAAe,CAAC,CAAC;IAC3C,MAAM,YAAY,GAAG,CAAC,CAAmB,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,IAAI,GAAG,CAAC,CAAC,YAAY,KAAK,CAAC,CAAC,YAAY,IAAI,CAAC,CAAC,SAAS,GAAG,CAAC;IAEtH,MAAM,MAAM,CAAC,MAAM,CAAC,YAAE,CAAC,iBAAiB,CAAC,GAAG,CAC1C,gBAAgB,CAAC,MAAM,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAA,aAAM,EAAC,MAAM,aAAI,KAAK,EAAE,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,EAC3G,gBAAgB,CAAC,MAAM,CACxB,CAAC,CAAC;IAEH,4DAA4D;IAC5D,qEAAqE;IACrE,IAAI,CAAC;QACH,MAAM,gBAAgB,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IACpC,CAAC;IAAC,OAAO,CAAM,EAAE,CAAC;QAChB,IAAI,CAAC,CAAC,IAAI,KAAK,cAAc,IAAI,CAAC,CAAC,IAAI,KAAK,YAAY,EAAE,CAAC;YACzD,MAAM,MAAM,GAAiB,IAAI,CAAC,KAAK,CAAC,IAAA,yBAAkB,EAAC,CAAC,CAAC,CAAC,CAAC;YAC/D,MAAM,KAAK,GAAG,IAAI,kBAAY,CAAC,uBAAuB,CAAC,MAAM,CAAC,CAAC,CAAC;YAChE,KAAK,CAAC,IAAI,GAAG,CAAC,CAAC,IAAI,CAAC;YACpB,MAAM,KAAK,CAAC;QACd,CAAC;QACD,MAAM,CAAC,CAAC;IACV,CAAC;IAED,MAAM,MAAM,CAAC,MAAM,CAAC,YAAE,CAAC,iBAAiB,CAAC,GAAG,CAC1C,gBAAgB,CAAC,MAAM,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAA,aAAM,EAAC,MAAM,aAAI,QAAQ,EAAE,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,EAAE,KAAK,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,EAC1I,gBAAgB,CAAC,MAAM,CACxB,CAAC,CAAC;IAEH,GAAG,CAAC,qBAAqB,CAAC,eAAe,CAAC,CAAC;AAC7C,CAAC;AAED,SAAS,uBAAuB,CAAC,MAAoB;IACnD,MAAM,IAAI,GAAG;QACX,+DAA+D,MAAM,CAAC,KAAK,EAAE;QAC7E,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,GAAG,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,EAAE;KACzC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAEb,IAAI,MAAM,CAAC,iBAAiB,IAAI,IAAI,EAAE,CAAC;QACrC,MAAM,iBAAiB,GAAG,MAAM;aAC7B,OAAO,CAAC,MAAM,CAAC,iBAAiB,CAAC;aACjC,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,KAAK,CAAC,EAAE,EAAE,CAAC,OAAO,GAAG,KAAK,KAAK,GAAG,CAAC;aAC9C,IAAI,CAAC,IAAI,CAAC,CAAC;QAEd,OAAO,GAAG,IAAI,yBAAyB,iBAAiB,EAAE,CAAC;IAC7D,CAAC;IAED,OAAO,IAAI,CAAC;AACd,CAAC;AAED,KAAK,UAAU,kBAAkB,CAC/B,MAAyB,EACzB,eAAiC,EACjC,WAAwB;IAExB,IAAI,eAAe,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACjC,OAAO;IACT,CAAC;IACD;;;;;;OAMG;IACH,IAAI,WAAW,KAAK,cAAc,EAAE,CAAC;QACnC,eAAe,GAAG,eAAe,CAAC,MAAM,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,kBAAkB,KAAK,IAAI,CAAC,CAAC;QAEzF,IAAI,eAAe,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACjC,OAAO;QACT,CAAC;IACH,CAAC;IAED,MAAM,QAAQ,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,wBAAwB;IAE/C,IAAI,WAAW,KAAK,cAAc,EAAE,CAAC;QACnC,QAAQ,CAAC,IAAI,CAAC,IAAA,aAAM,EAAC,OAAO,EAAE,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,KAAK,CAAC,GAAG,CAAC,wHAAwH,CAAC,CAAC,CAAC,CAAC;IACvL,CAAC;SAAM,CAAC;QACN,QAAQ,CAAC,IAAI,CAAC,IAAA,aAAM,EAAC,OAAO,EAAE,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,KAAK,CAAC,GAAG,CAAC,oDAAoD,CAAC,CAAC,CAAC,CAAC;IACnH,CAAC;IAED,KAAK,MAAM,EAAE,MAAM,EAAE,IAAI,eAAe,EAAE,CAAC;QACzC,QAAQ,CAAC,IAAI,CAAC,MAAM,GAAG,4BAA4B,CAAC,MAAM,CAAC,CAAC,CAAC;IAC/D,CAAC;IACD,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC,UAAU;IAE7B,MAAM,MAAM,CAAC,MAAM,CAAC,YAAE,CAAC,oBAAoB,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;AACxE,CAAC;AAED;;GAEG;AACH,SAAS,4BAA4B,CAAC,MAA6B;IACjE,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC;IAC/B,MAAM,MAAM,GAAG,MAAM,CAAC,WAAW,IAAI,MAAM,CAAC,MAAM,CAAC;IAEnD,QAAQ,OAAO,CAAC,IAAI,EAAE,CAAC;QACrB,KAAK,QAAQ;YACX,OAAO,IAAA,aAAM,EACX,wBAAwB,EACxB,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,EAC7B,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,CAClB,CAAC;QACJ,KAAK,UAAU;YACb,OAAO,8BAA8B,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;IAC3D,CAAC;AACH,CAAC;AAED;;GAEG;AACH,SAAS,8BAA8B,CAAC,OAAwB,EAAE,MAAc;IAC9E,IAAI,OAAO,CAAC,kBAAkB,EAAE,MAAM,EAAE,CAAC;QACvC,OAAO,IAAA,aAAM,EACX,0DAA0D,EAC1D,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,EAC7B,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,EAChC,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,kBAAkB,CAAC,EACtC,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,CAClB,CAAC;IACJ,CAAC;IAED,OAAO,IAAA,aAAM,EACX,oCAAoC,EACpC,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,EAC7B,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,EAChC,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,CAClB,CAAC;AACJ,CAAC","sourcesContent":["import { format } from 'util';\nimport * as cfn_diff from '@aws-cdk/cloudformation-diff';\nimport type * as cxapi from '@aws-cdk/cx-api';\nimport type { WaiterResult } from '@smithy/util-waiter';\nimport * as chalk from 'chalk';\nimport type { SuccessfulDeployStackResult } from './deployment-result';\nimport { ToolkitError } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api';\nimport type { AffectedResource, HotswapResult, ResourceSubject, ResourceChange, NonHotswappableChange } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads';\nimport { NonHotswappableReason } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/payloads';\nimport type { IMessageSpan, IoHelper } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';\nimport { IO, SPAN } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';\nimport { formatErrorMessage } from '../../util';\nimport type { SDK, SdkProvider } from '../aws-auth';\nimport type { CloudFormationStack, NestedStackTemplates } from '../cloudformation';\nimport { loadCurrentTemplateWithNestedStacks, EvaluateCloudFormationTemplate } from '../cloudformation';\nimport { isHotswappableAppSyncChange } from '../hotswap/appsync-mapping-templates';\nimport { isHotswappableCodeBuildProjectChange } from '../hotswap/code-build-projects';\nimport type {\n  HotswapChange,\n  HotswapOperation,\n  RejectedChange,\n  HotswapPropertyOverrides,\n} from '../hotswap/common';\nimport {\n  ICON,\n  nonHotswappableResource,\n} from '../hotswap/common';\nimport { isHotswappableEcsServiceChange } from '../hotswap/ecs-services';\nimport { isHotswappableLambdaFunctionChange } from '../hotswap/lambda-functions';\nimport {\n  skipChangeForS3DeployCustomResourcePolicy,\n  isHotswappableS3BucketDeploymentChange,\n} from '../hotswap/s3-bucket-deployments';\nimport { isHotswappableStateMachineChange } from '../hotswap/stepfunctions-state-machines';\nimport { Mode } from '../plugin';\n\n// Must use a require() otherwise esbuild complains about calling a namespace\n// eslint-disable-next-line @typescript-eslint/no-require-imports,@typescript-eslint/consistent-type-imports\nconst pLimit: typeof import('p-limit') = require('p-limit');\n\ntype HotswapDetector = (\n  logicalId: string,\n  change: ResourceChange,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n) => Promise<HotswapChange[]>;\n\ntype HotswapMode = 'hotswap-only' | 'fall-back';\n\nconst RESOURCE_DETECTORS: { [key: string]: HotswapDetector } = {\n  // Lambda\n  'AWS::Lambda::Function': isHotswappableLambdaFunctionChange,\n  'AWS::Lambda::Version': isHotswappableLambdaFunctionChange,\n  'AWS::Lambda::Alias': isHotswappableLambdaFunctionChange,\n\n  // AppSync\n  'AWS::AppSync::Resolver': isHotswappableAppSyncChange,\n  'AWS::AppSync::FunctionConfiguration': isHotswappableAppSyncChange,\n  'AWS::AppSync::GraphQLSchema': isHotswappableAppSyncChange,\n  'AWS::AppSync::ApiKey': isHotswappableAppSyncChange,\n\n  'AWS::ECS::TaskDefinition': isHotswappableEcsServiceChange,\n  'AWS::CodeBuild::Project': isHotswappableCodeBuildProjectChange,\n  'AWS::StepFunctions::StateMachine': isHotswappableStateMachineChange,\n  'Custom::CDKBucketDeployment': isHotswappableS3BucketDeploymentChange,\n  'AWS::IAM::Policy': async (\n    logicalId: string,\n    change: ResourceChange,\n    evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  ): Promise<HotswapChange[]> => {\n    // If the policy is for a S3BucketDeploymentChange, we can ignore the change\n    if (await skipChangeForS3DeployCustomResourcePolicy(logicalId, change, evaluateCfnTemplate)) {\n      return [];\n    }\n\n    return [nonHotswappableResource(change)];\n  },\n\n  'AWS::CDK::Metadata': async () => [],\n};\n\n/**\n * Perform a hotswap deployment, 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,\n  ioHelper: IoHelper,\n  assetParams: { [key: string]: string },\n  cloudFormationStack: CloudFormationStack,\n  stackArtifact: cxapi.CloudFormationStackArtifact,\n  hotswapMode: HotswapMode,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<SuccessfulDeployStackResult | undefined> {\n  const hotswapSpan = await ioHelper.span(SPAN.HOTSWAP).begin({\n    stack: stackArtifact,\n    mode: hotswapMode,\n  });\n\n  const result = await hotswapDeployment(\n    sdkProvider,\n    hotswapSpan,\n    assetParams,\n    stackArtifact,\n    hotswapMode,\n    hotswapPropertyOverrides,\n  );\n\n  await hotswapSpan.end(result);\n\n  if (result?.hotswapped === true) {\n    return {\n      type: 'did-deploy-stack',\n      noOp: result.hotswappableChanges.length === 0,\n      stackArn: cloudFormationStack.stackId,\n      outputs: cloudFormationStack.outputs,\n    };\n  }\n\n  return undefined;\n}\n\n/**\n * Perform a hotswap deployment, short-circuiting CloudFormation if possible.\n * Returns information about the attempted hotswap deployment\n */\nasync function hotswapDeployment(\n  sdkProvider: SdkProvider,\n  ioSpan: IMessageSpan<any>,\n  assetParams: { [key: string]: string },\n  stack: cxapi.CloudFormationStackArtifact,\n  hotswapMode: HotswapMode,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<Omit<HotswapResult, 'duration'>> {\n  // resolve the environment, so we can substitute things like AWS::Region in CFN expressions\n  const resolvedEnv = await sdkProvider.resolveEnvironment(stack.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\n  const currentTemplate = await loadCurrentTemplateWithNestedStacks(stack, sdk);\n\n  const evaluateCfnTemplate = new EvaluateCloudFormationTemplate({\n    stackArtifact: stack,\n    parameters: assetParams,\n    account: resolvedEnv.account,\n    region: resolvedEnv.region,\n    partition: (await sdk.currentAccount()).partition,\n    sdk,\n    nestedStacks: currentTemplate.nestedStacks,\n  });\n\n  const stackChanges = cfn_diff.fullDiff(currentTemplate.deployedRootTemplate, stack.template);\n  const { hotswappable, nonHotswappable } = await classifyResourceChanges(\n    stackChanges,\n    evaluateCfnTemplate,\n    sdk,\n    currentTemplate.nestedStacks, hotswapPropertyOverrides,\n  );\n\n  await logRejectedChanges(ioSpan, nonHotswappable, hotswapMode);\n\n  const hotswappableChanges = hotswappable.map(o => o.change);\n  const nonHotswappableChanges = nonHotswappable.map(n => n.change);\n\n  await ioSpan.notify(IO.CDK_TOOLKIT_I5401.msg('Hotswap plan created', {\n    stack,\n    mode: hotswapMode,\n    hotswappableChanges,\n    nonHotswappableChanges,\n  }));\n\n  // preserve classic hotswap behavior\n  if (hotswapMode === 'fall-back') {\n    if (nonHotswappableChanges.length > 0) {\n      return {\n        stack,\n        mode: hotswapMode,\n        hotswapped: false,\n        hotswappableChanges,\n        nonHotswappableChanges,\n      };\n    }\n  }\n\n  // apply the short-circuitable changes\n  await applyAllHotswapOperations(sdk, ioSpan, hotswappable);\n\n  return {\n    stack,\n    mode: hotswapMode,\n    hotswapped: true,\n    hotswappableChanges,\n    nonHotswappableChanges,\n  };\n}\n\ninterface ClassifiedChanges {\n  hotswappable: HotswapOperation[];\n  nonHotswappable: RejectedChange[];\n}\n\n/**\n * Classifies all changes to all resources as either hotswappable or not.\n * Metadata changes are excluded from the list of (non)hotswappable resources.\n */\nasync function classifyResourceChanges(\n  stackChanges: cfn_diff.TemplateDiff,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  sdk: SDK,\n  nestedStackNames: { [nestedStackName: string]: NestedStackTemplates },\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<ClassifiedChanges> {\n  const resourceDifferences = getStackResourceDifferences(stackChanges);\n\n  const promises: Array<() => Promise<HotswapChange[]>> = [];\n  const hotswappableResources = new Array<HotswapOperation>();\n  const nonHotswappableResources = new Array<RejectedChange>();\n  for (const logicalId of Object.keys(stackChanges.outputs.changes)) {\n    nonHotswappableResources.push({\n      hotswappable: false,\n      change: {\n        reason: NonHotswappableReason.OUTPUT,\n        description: 'output was changed',\n        subject: {\n          type: 'Output',\n          logicalId,\n          metadata: evaluateCfnTemplate.metadataFor(logicalId),\n        },\n      },\n    });\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(\n        logicalId,\n        change,\n        nestedStackNames,\n        evaluateCfnTemplate,\n        sdk,\n        hotswapPropertyOverrides,\n      );\n      hotswappableResources.push(...nestedHotswappableResources.hotswappable);\n      nonHotswappableResources.push(...nestedHotswappableResources.nonHotswappable);\n\n      continue;\n    }\n\n    const hotswappableChangeCandidate = isCandidateForHotswapping(logicalId, change, evaluateCfnTemplate);\n    // we don't need to run this through the detector functions, we can already judge this\n    if ('hotswappable' in hotswappableChangeCandidate) {\n      if (!hotswappableChangeCandidate.hotswappable) {\n        nonHotswappableResources.push(hotswappableChangeCandidate);\n      }\n\n      continue;\n    }\n\n    const resourceType: string = hotswappableChangeCandidate.newValue.Type;\n    if (resourceType in RESOURCE_DETECTORS) {\n      // run detector functions lazily to prevent unhandled promise rejections\n      promises.push(() =>\n        RESOURCE_DETECTORS[resourceType](logicalId, hotswappableChangeCandidate, evaluateCfnTemplate, hotswapPropertyOverrides),\n      );\n    } else {\n      nonHotswappableResources.push(nonHotswappableResource(hotswappableChangeCandidate));\n    }\n  }\n\n  // resolve all detector results\n  const changesDetectionResults: Array<HotswapChange[]> = [];\n  for (const detectorResultPromises of promises) {\n    // Constant set of promises per resource\n    // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n    const hotswapDetectionResults = await Promise.all(await detectorResultPromises());\n    changesDetectionResults.push(hotswapDetectionResults);\n  }\n\n  for (const resourceDetectionResults of changesDetectionResults) {\n    for (const propertyResult of resourceDetectionResults) {\n      propertyResult.hotswappable\n        ? hotswappableResources.push(propertyResult)\n        : nonHotswappableResources.push(propertyResult);\n    }\n  }\n\n  return {\n    hotswappable: hotswappableResources,\n    nonHotswappable: nonHotswappableResources,\n  };\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): {\n  [logicalId: string]: cfn_diff.ResourceDifference;\n} {\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(\n    (acc, [key, t]) => {\n      if (func(t)) {\n        acc[key] = t;\n      }\n      return acc;\n    },\n    {} as { [key: string]: T },\n  );\n}\n\n/** Finds any hotswappable changes in all nested stacks. */\nasync function findNestedHotswappableChanges(\n  logicalId: string,\n  change: cfn_diff.ResourceDifference,\n  nestedStackTemplates: { [nestedStackName: string]: NestedStackTemplates },\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  sdk: SDK,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<ClassifiedChanges> {\n  const nestedStack = nestedStackTemplates[logicalId];\n  if (!nestedStack.physicalName) {\n    return {\n      hotswappable: [],\n      nonHotswappable: [\n        {\n          hotswappable: false,\n          change: {\n            reason: NonHotswappableReason.NESTED_STACK_CREATION,\n            description: 'newly created nested stacks cannot be hotswapped',\n            subject: {\n              type: 'Resource',\n              logicalId,\n              resourceType: 'AWS::CloudFormation::Stack',\n              metadata: evaluateCfnTemplate.metadataFor(logicalId),\n            },\n          },\n        },\n      ],\n    };\n  }\n\n  const evaluateNestedCfnTemplate = await evaluateCfnTemplate.createNestedEvaluateCloudFormationTemplate(\n    nestedStack.physicalName,\n    nestedStack.generatedTemplate,\n    change.newValue?.Properties?.Parameters,\n  );\n\n  const nestedDiff = cfn_diff.fullDiff(\n    nestedStackTemplates[logicalId].deployedTemplate,\n    nestedStackTemplates[logicalId].generatedTemplate,\n  );\n\n  return classifyResourceChanges(\n    nestedDiff,\n    evaluateNestedCfnTemplate,\n    sdk,\n    nestedStackTemplates[logicalId].nestedStackTemplates,\n    hotswapPropertyOverrides,\n  );\n}\n\n/** Returns 'true' if a pair of changes is for the same resource. */\nfunction changesAreForSameResource(\n  oldChange: cfn_diff.ResourceDifference,\n  newChange: cfn_diff.ResourceDifference,\n): boolean {\n  return (\n    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}\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 a `HotswappableChangeCandidate` if the change is hotswappable\n * Returns an empty `HotswappableChange` if the change is to CDK::Metadata\n * Returns a `NonHotswappableChange` if the change is not hotswappable\n */\nfunction isCandidateForHotswapping(\n  logicalId: string,\n  change: cfn_diff.ResourceDifference,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): RejectedChange | ResourceChange {\n  // a resource has been removed OR a resource has been added; we can't short-circuit that change\n  if (!change.oldValue) {\n    return {\n      hotswappable: false,\n      change: {\n        reason: NonHotswappableReason.RESOURCE_CREATION,\n        description: `resource '${logicalId}' was created by this deployment`,\n        subject: {\n          type: 'Resource',\n          logicalId,\n          resourceType: change.newValue!.Type,\n          metadata: evaluateCfnTemplate.metadataFor(logicalId),\n        },\n      },\n    };\n  } else if (!change.newValue) {\n    return {\n      hotswappable: false,\n      logicalId,\n      change: {\n        reason: NonHotswappableReason.RESOURCE_DELETION,\n        description: `resource '${logicalId}' was destroyed by this deployment`,\n        subject: {\n          type: 'Resource',\n          logicalId,\n          resourceType: change.oldValue.Type,\n          metadata: evaluateCfnTemplate.metadataFor(logicalId),\n        },\n      },\n    };\n  }\n\n  // a resource has had its type changed\n  if (change.newValue.Type !== change.oldValue.Type) {\n    return {\n      hotswappable: false,\n      change: {\n        reason: NonHotswappableReason.RESOURCE_TYPE_CHANGED,\n        description: `resource '${logicalId}' had its type changed from '${change.oldValue?.Type}' to '${change.newValue?.Type}'`,\n        subject: {\n          type: 'Resource',\n          logicalId,\n          resourceType: change.newValue.Type,\n          metadata: evaluateCfnTemplate.metadataFor(logicalId),\n        },\n      },\n    };\n  }\n\n  return {\n    logicalId,\n    oldValue: change.oldValue,\n    newValue: change.newValue,\n    propertyUpdates: change.propertyUpdates,\n    metadata: evaluateCfnTemplate.metadataFor(logicalId),\n  };\n}\n\nasync function applyAllHotswapOperations(sdk: SDK, ioSpan: IMessageSpan<any>, hotswappableChanges: HotswapOperation[]): Promise<void[]> {\n  if (hotswappableChanges.length === 0) {\n    return Promise.resolve([]);\n  }\n\n  await ioSpan.notify(IO.DEFAULT_TOOLKIT_INFO.msg(`\\n${ICON} hotswapping resources:`));\n  const limit = pLimit(10);\n  // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n  return Promise.all(hotswappableChanges.map(hotswapOperation => limit(() => {\n    return applyHotswapOperation(sdk, ioSpan, hotswapOperation);\n  })));\n}\n\nasync function applyHotswapOperation(sdk: SDK, ioSpan: IMessageSpan<any>, hotswapOperation: HotswapOperation): Promise<void> {\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  const resourceText = (r: AffectedResource) => r.description ?? `${r.resourceType} '${r.physicalName ?? r.logicalId}'`;\n\n  await ioSpan.notify(IO.CDK_TOOLKIT_I5402.msg(\n    hotswapOperation.change.resources.map(r => format(`   ${ICON} %s`, chalk.bold(resourceText(r)))).join('\\n'),\n    hotswapOperation.change,\n  ));\n\n  // if the SDK call fails, an error will be thrown by the SDK\n  // and will prevent the green 'hotswapped!' text from being displayed\n  try {\n    await hotswapOperation.apply(sdk);\n  } catch (e: any) {\n    if (e.name === 'TimeoutError' || e.name === 'AbortError') {\n      const result: WaiterResult = JSON.parse(formatErrorMessage(e));\n      const error = new ToolkitError(formatWaiterErrorResult(result));\n      error.name = e.name;\n      throw error;\n    }\n    throw e;\n  }\n\n  await ioSpan.notify(IO.CDK_TOOLKIT_I5403.msg(\n    hotswapOperation.change.resources.map(r => format(`   ${ICON} %s %s`, chalk.bold(resourceText(r)), chalk.green('hotswapped!'))).join('\\n'),\n    hotswapOperation.change,\n  ));\n\n  sdk.removeCustomUserAgent(customUserAgent);\n}\n\nfunction formatWaiterErrorResult(result: WaiterResult) {\n  const main = [\n    `Resource is not in the expected state due to waiter status: ${result.state}`,\n    result.reason ? `${result.reason}.` : '',\n  ].join('. ');\n\n  if (result.observedResponses != null) {\n    const observedResponses = Object\n      .entries(result.observedResponses)\n      .map(([msg, count]) => `  - ${msg} (${count})`)\n      .join('\\n');\n\n    return `${main} Observed responses:\\n${observedResponses}`;\n  }\n\n  return main;\n}\n\nasync function logRejectedChanges(\n  ioSpan: IMessageSpan<any>,\n  rejectedChanges: RejectedChange[],\n  hotswapMode: HotswapMode,\n): Promise<void> {\n  if (rejectedChanges.length === 0) {\n    return;\n  }\n  /**\n   * EKS 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  if (hotswapMode === 'hotswap-only') {\n    rejectedChanges = rejectedChanges.filter((change) => change.hotswapOnlyVisible === true);\n\n    if (rejectedChanges.length === 0) {\n      return;\n    }\n  }\n\n  const messages = ['']; // start with empty line\n\n  if (hotswapMode === 'hotswap-only') {\n    messages.push(format('%s %s', chalk.red('⚠️'), chalk.red('The following non-hotswappable changes were found. To reconcile these using CloudFormation, specify --hotswap-fallback')));\n  } else {\n    messages.push(format('%s %s', chalk.red('⚠️'), chalk.red('The following non-hotswappable changes were found:')));\n  }\n\n  for (const { change } of rejectedChanges) {\n    messages.push('    ' + nonHotswappableChangeMessage(change));\n  }\n  messages.push(''); // newline\n\n  await ioSpan.notify(IO.DEFAULT_TOOLKIT_INFO.msg(messages.join('\\n')));\n}\n\n/**\n * Formats a NonHotswappableChange\n */\nfunction nonHotswappableChangeMessage(change: NonHotswappableChange): string {\n  const subject = change.subject;\n  const reason = change.description ?? change.reason;\n\n  switch (subject.type) {\n    case 'Output':\n      return format(\n        'output: %s, reason: %s',\n        chalk.bold(subject.logicalId),\n        chalk.red(reason),\n      );\n    case 'Resource':\n      return nonHotswappableResourceMessage(subject, reason);\n  }\n}\n\n/**\n * Formats a non-hotswappable resource subject\n */\nfunction nonHotswappableResourceMessage(subject: ResourceSubject, reason: string): string {\n  if (subject.rejectedProperties?.length) {\n    return format(\n      'resource: %s, type: %s, rejected changes: %s, reason: %s',\n      chalk.bold(subject.logicalId),\n      chalk.bold(subject.resourceType),\n      chalk.bold(subject.rejectedProperties),\n      chalk.red(reason),\n    );\n  }\n\n  return format(\n    'resource: %s, type: %s, reason: %s',\n    chalk.bold(subject.logicalId),\n    chalk.bold(subject.resourceType),\n    chalk.red(reason),\n  );\n}\n"]}
|
|
@@ -14,10 +14,9 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
|
14
14
|
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
15
|
};
|
|
16
16
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
-
__exportStar(require("./cloudformation"), exports);
|
|
18
17
|
__exportStar(require("./deployments"), exports);
|
|
19
18
|
__exportStar(require("./deployment-result"), exports);
|
|
20
19
|
__exportStar(require("./deployment-method"), exports);
|
|
21
|
-
__exportStar(require("./nested-stack-helpers"), exports);
|
|
22
20
|
__exportStar(require("./asset-manifest-builder"), exports);
|
|
23
|
-
|
|
21
|
+
__exportStar(require("./cfn-api"), exports);
|
|
22
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsZ0RBQThCO0FBQzlCLHNEQUFvQztBQUNwQyxzREFBb0M7QUFDcEMsMkRBQXlDO0FBQ3pDLDRDQUEwQiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vZGVwbG95bWVudHMnO1xuZXhwb3J0ICogZnJvbSAnLi9kZXBsb3ltZW50LXJlc3VsdCc7XG5leHBvcnQgKiBmcm9tICcuL2RlcGxveW1lbnQtbWV0aG9kJztcbmV4cG9ydCAqIGZyb20gJy4vYXNzZXQtbWFuaWZlc3QtYnVpbGRlcic7XG5leHBvcnQgKiBmcm9tICcuL2Nmbi1hcGknO1xuIl19
|
|
@@ -1,9 +1,9 @@
|
|
|
1
1
|
import type * as cxapi from '@aws-cdk/cx-api';
|
|
2
2
|
import type { SDK } from '../aws-auth';
|
|
3
3
|
import type { EnvironmentResources } from './environment-resources';
|
|
4
|
-
import type {
|
|
4
|
+
import type { StringWithoutPlaceholders } from './placeholders';
|
|
5
|
+
import { type IoHelper } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';
|
|
5
6
|
import type { SdkProvider } from '../aws-auth/sdk-provider';
|
|
6
|
-
import type { StringWithoutPlaceholders } from '../util/placeholders';
|
|
7
7
|
/**
|
|
8
8
|
* Access particular AWS resources, based on information from the CX manifest
|
|
9
9
|
*
|