aws-cdk 2.2.0 → 2.6.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/LICENSE +1 -1
- package/NOTICE +1 -1
- package/README.md +4 -1
- package/build-info.json +2 -2
- package/lib/api/aws-auth/sdk-provider.js +11 -21
- package/lib/api/aws-auth/sdk.d.ts +2 -0
- package/lib/api/aws-auth/sdk.js +4 -1
- package/lib/api/bootstrap/bootstrap-template.yaml +3 -1
- package/lib/api/deploy-stack.js +7 -1
- package/lib/api/hotswap/code-build-projects.d.ts +3 -0
- package/lib/api/hotswap/code-build-projects.js +53 -0
- package/lib/api/hotswap/common.d.ts +17 -1
- package/lib/api/hotswap/common.js +30 -6
- package/lib/api/hotswap/ecs-services.js +4 -18
- package/lib/api/hotswap/lambda-functions.js +169 -44
- package/lib/api/hotswap/s3-bucket-deployments.js +3 -10
- package/lib/api/hotswap/stepfunctions-state-machines.js +2 -1
- package/lib/api/hotswap-deployments.d.ts +0 -7
- package/lib/api/hotswap-deployments.js +84 -12
- package/lib/cdk-toolkit.js +50 -16
- package/lib/commands/docs.js +2 -2
- package/lib/init.js +2 -1
- package/lib/util/npm.d.ts +1 -0
- package/lib/util/npm.js +21 -0
- package/lib/version.d.ts +1 -1
- package/lib/version.js +22 -19
- package/npm-shrinkwrap.json +166 -33
- package/package.json +14 -14
- package/test/api/hotswap/{lambda-hotswap-deployments.test.d.ts → code-build-projects-hotswap-deployments.test.d.ts} +0 -0
- package/test/api/hotswap/code-build-projects-hotswap-deployments.test.js +576 -0
- package/test/api/hotswap/hotswap-deployments.test.js +4 -2
- package/test/api/hotswap/hotswap-test-setup.d.ts +3 -1
- package/test/api/hotswap/hotswap-test-setup.js +7 -4
- package/test/api/hotswap/lambda-functions-hotswap-deployments.test.d.ts +1 -0
- package/test/api/hotswap/lambda-functions-hotswap-deployments.test.js +502 -0
- package/test/api/hotswap/lambda-versions-aliases-hotswap-deployments.test.d.ts +1 -0
- package/test/api/hotswap/lambda-versions-aliases-hotswap-deployments.test.js +197 -0
- package/test/cdk-toolkit.test.js +11 -4
- package/test/init.test.js +5 -4
- package/test/util/mock-sdk.d.ts +2 -0
- package/test/util/mock-sdk.js +5 -1
- package/test/version.test.js +45 -3
- package/test/api/hotswap/lambda-hotswap-deployments.test.js +0 -418
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
exports.isHotswappableLambdaFunctionChange = void 0;
|
|
4
|
+
const util_1 = require("../../util");
|
|
4
5
|
const common_1 = require("./common");
|
|
5
6
|
/**
|
|
6
7
|
* Returns `ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT` if the change cannot be short-circuited,
|
|
@@ -10,29 +11,55 @@ const common_1 = require("./common");
|
|
|
10
11
|
*/
|
|
11
12
|
async function isHotswappableLambdaFunctionChange(logicalId, change, evaluateCfnTemplate) {
|
|
12
13
|
var _a;
|
|
14
|
+
// if the change is for a Lambda Version,
|
|
15
|
+
// ignore it by returning an empty hotswap operation -
|
|
16
|
+
// we will publish a new version when we get to hotswapping the actual Function this Version points to, below
|
|
17
|
+
// (Versions can't be changed in CloudFormation anyway, they're immutable)
|
|
18
|
+
if (change.newValue.Type === 'AWS::Lambda::Version') {
|
|
19
|
+
return common_1.ChangeHotswapImpact.IRRELEVANT;
|
|
20
|
+
}
|
|
21
|
+
// we handle Aliases specially too
|
|
22
|
+
if (change.newValue.Type === 'AWS::Lambda::Alias') {
|
|
23
|
+
return checkAliasHasVersionOnlyChange(change);
|
|
24
|
+
}
|
|
13
25
|
const lambdaCodeChange = await isLambdaFunctionCodeOnlyChange(change, evaluateCfnTemplate);
|
|
14
26
|
if (typeof lambdaCodeChange === 'string') {
|
|
15
27
|
return lambdaCodeChange;
|
|
16
28
|
}
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
29
|
+
const functionName = await common_1.establishResourcePhysicalName(logicalId, (_a = change.newValue.Properties) === null || _a === void 0 ? void 0 : _a.FunctionName, evaluateCfnTemplate);
|
|
30
|
+
if (!functionName) {
|
|
31
|
+
return common_1.ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;
|
|
32
|
+
}
|
|
33
|
+
const functionArn = await evaluateCfnTemplate.evaluateCfnExpression({
|
|
34
|
+
'Fn::Sub': 'arn:${AWS::Partition}:lambda:${AWS::Region}:${AWS::AccountId}:function:' + functionName,
|
|
35
|
+
});
|
|
36
|
+
// find all Lambda Versions that reference this Function
|
|
37
|
+
const versionsReferencingFunction = evaluateCfnTemplate.findReferencesTo(logicalId)
|
|
38
|
+
.filter(r => r.Type === 'AWS::Lambda::Version');
|
|
39
|
+
// find all Lambda Aliases that reference the above Versions
|
|
40
|
+
const aliasesReferencingVersions = util_1.flatMap(versionsReferencingFunction, v => evaluateCfnTemplate.findReferencesTo(v.LogicalId));
|
|
41
|
+
const aliasesNames = await Promise.all(aliasesReferencingVersions.map(a => { var _a; return evaluateCfnTemplate.evaluateCfnExpression((_a = a.Properties) === null || _a === void 0 ? void 0 : _a.Name); }));
|
|
42
|
+
return new LambdaFunctionHotswapOperation({
|
|
43
|
+
physicalName: functionName,
|
|
44
|
+
functionArn: functionArn,
|
|
45
|
+
resource: lambdaCodeChange,
|
|
46
|
+
publishVersion: versionsReferencingFunction.length > 0,
|
|
47
|
+
aliasesNames,
|
|
48
|
+
});
|
|
49
|
+
}
|
|
50
|
+
exports.isHotswappableLambdaFunctionChange = isHotswappableLambdaFunctionChange;
|
|
51
|
+
/**
|
|
52
|
+
* Returns is a given Alias change is only in the 'FunctionVersion' property,
|
|
53
|
+
* and `ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT` is the change is for any other property.
|
|
54
|
+
*/
|
|
55
|
+
function checkAliasHasVersionOnlyChange(change) {
|
|
56
|
+
for (const updatedPropName in change.propertyUpdates) {
|
|
57
|
+
if (updatedPropName !== 'FunctionVersion') {
|
|
27
58
|
return common_1.ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;
|
|
28
59
|
}
|
|
29
|
-
return new LambdaFunctionHotswapOperation({
|
|
30
|
-
physicalName: functionName,
|
|
31
|
-
code: lambdaCodeChange,
|
|
32
|
-
});
|
|
33
60
|
}
|
|
61
|
+
return common_1.ChangeHotswapImpact.IRRELEVANT;
|
|
34
62
|
}
|
|
35
|
-
exports.isHotswappableLambdaFunctionChange = isHotswappableLambdaFunctionChange;
|
|
36
63
|
/**
|
|
37
64
|
* Returns `ChangeHotswapImpact.IRRELEVANT` if the change is not for a AWS::Lambda::Function,
|
|
38
65
|
* but doesn't prevent short-circuiting
|
|
@@ -48,7 +75,7 @@ async function isLambdaFunctionCodeOnlyChange(change, evaluateCfnTemplate) {
|
|
|
48
75
|
return common_1.ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;
|
|
49
76
|
}
|
|
50
77
|
/*
|
|
51
|
-
*
|
|
78
|
+
* At first glance, we would want to initialize these using the "previous" values (change.oldValue),
|
|
52
79
|
* in case only one of them changed, like the key, and the Bucket stayed the same.
|
|
53
80
|
* However, that actually fails for old-style synthesis, which uses CFN Parameters!
|
|
54
81
|
* Because the names of the Parameters depend on the hash of the Asset,
|
|
@@ -58,45 +85,143 @@ async function isLambdaFunctionCodeOnlyChange(change, evaluateCfnTemplate) {
|
|
|
58
85
|
* even if only one of them was actually changed,
|
|
59
86
|
* which means we don't need the "old" values at all, and we can safely initialize these with just `''`.
|
|
60
87
|
*/
|
|
61
|
-
let s3Bucket = '', s3Key = '';
|
|
62
|
-
let foundCodeDifference = false;
|
|
63
|
-
// Make sure only the code in the Lambda function changed
|
|
64
88
|
const propertyUpdates = change.propertyUpdates;
|
|
89
|
+
let code = undefined;
|
|
90
|
+
let tags = undefined;
|
|
65
91
|
for (const updatedPropName in propertyUpdates) {
|
|
66
92
|
const updatedProp = propertyUpdates[updatedPropName];
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
93
|
+
switch (updatedPropName) {
|
|
94
|
+
case 'Code':
|
|
95
|
+
let foundCodeDifference = false;
|
|
96
|
+
let s3Bucket = '', s3Key = '';
|
|
97
|
+
for (const newPropName in updatedProp.newValue) {
|
|
98
|
+
switch (newPropName) {
|
|
99
|
+
case 'S3Bucket':
|
|
100
|
+
foundCodeDifference = true;
|
|
101
|
+
s3Bucket = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue[newPropName]);
|
|
102
|
+
break;
|
|
103
|
+
case 'S3Key':
|
|
104
|
+
foundCodeDifference = true;
|
|
105
|
+
s3Key = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue[newPropName]);
|
|
106
|
+
break;
|
|
107
|
+
default:
|
|
108
|
+
return common_1.ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
if (foundCodeDifference) {
|
|
112
|
+
code = {
|
|
113
|
+
s3Bucket,
|
|
114
|
+
s3Key,
|
|
115
|
+
};
|
|
116
|
+
}
|
|
117
|
+
break;
|
|
118
|
+
case 'Tags':
|
|
119
|
+
/*
|
|
120
|
+
* Tag updates are a bit odd; they manifest as two lists, are flagged only as
|
|
121
|
+
* `isDifferent`, and we have to reconcile them.
|
|
122
|
+
*/
|
|
123
|
+
const tagUpdates = {};
|
|
124
|
+
if (updatedProp === null || updatedProp === void 0 ? void 0 : updatedProp.isDifferent) {
|
|
125
|
+
updatedProp.newValue.forEach((tag) => {
|
|
126
|
+
tagUpdates[tag.Key] = tag.Value;
|
|
127
|
+
});
|
|
128
|
+
updatedProp.oldValue.forEach((tag) => {
|
|
129
|
+
if (tagUpdates[tag.Key] === undefined) {
|
|
130
|
+
tagUpdates[tag.Key] = TagDeletion.DELETE;
|
|
131
|
+
}
|
|
132
|
+
});
|
|
133
|
+
tags = { tagUpdates };
|
|
134
|
+
}
|
|
135
|
+
break;
|
|
136
|
+
default:
|
|
137
|
+
return common_1.ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;
|
|
80
138
|
}
|
|
81
139
|
}
|
|
82
|
-
return
|
|
83
|
-
? {
|
|
84
|
-
s3Bucket,
|
|
85
|
-
s3Key,
|
|
86
|
-
}
|
|
87
|
-
: common_1.ChangeHotswapImpact.IRRELEVANT;
|
|
140
|
+
return code || tags ? { code, tags } : common_1.ChangeHotswapImpact.IRRELEVANT;
|
|
88
141
|
}
|
|
142
|
+
var TagDeletion;
|
|
143
|
+
(function (TagDeletion) {
|
|
144
|
+
TagDeletion[TagDeletion["DELETE"] = -1] = "DELETE";
|
|
145
|
+
})(TagDeletion || (TagDeletion = {}));
|
|
89
146
|
class LambdaFunctionHotswapOperation {
|
|
90
147
|
constructor(lambdaFunctionResource) {
|
|
91
148
|
this.lambdaFunctionResource = lambdaFunctionResource;
|
|
92
149
|
this.service = 'lambda-function';
|
|
150
|
+
this.resourceNames = [
|
|
151
|
+
`Lambda Function '${lambdaFunctionResource.physicalName}'`,
|
|
152
|
+
// add Version here if we're publishing a new one
|
|
153
|
+
...(lambdaFunctionResource.publishVersion ? [`Lambda Version for Function '${lambdaFunctionResource.physicalName}'`] : []),
|
|
154
|
+
// add any Aliases that we are hotswapping here
|
|
155
|
+
...lambdaFunctionResource.aliasesNames.map(alias => `Lambda Alias '${alias}' for Function '${lambdaFunctionResource.physicalName}'`),
|
|
156
|
+
];
|
|
93
157
|
}
|
|
94
158
|
async apply(sdk) {
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
159
|
+
const lambda = sdk.lambda();
|
|
160
|
+
const resource = this.lambdaFunctionResource.resource;
|
|
161
|
+
const operations = [];
|
|
162
|
+
if (resource.code !== undefined) {
|
|
163
|
+
const updateFunctionCodePromise = lambda.updateFunctionCode({
|
|
164
|
+
FunctionName: this.lambdaFunctionResource.physicalName,
|
|
165
|
+
S3Bucket: resource.code.s3Bucket,
|
|
166
|
+
S3Key: resource.code.s3Key,
|
|
167
|
+
}).promise();
|
|
168
|
+
// only if the code changed is there any point in publishing a new Version
|
|
169
|
+
if (this.lambdaFunctionResource.publishVersion) {
|
|
170
|
+
// we need to wait for the code update to be done before publishing a new Version
|
|
171
|
+
await updateFunctionCodePromise;
|
|
172
|
+
// if we don't wait for the Function to finish updating,
|
|
173
|
+
// we can get a "The operation cannot be performed at this time. An update is in progress for resource:"
|
|
174
|
+
// error when publishing a new Version
|
|
175
|
+
await lambda.waitFor('functionUpdated', {
|
|
176
|
+
FunctionName: this.lambdaFunctionResource.physicalName,
|
|
177
|
+
}).promise();
|
|
178
|
+
const publishVersionPromise = lambda.publishVersion({
|
|
179
|
+
FunctionName: this.lambdaFunctionResource.physicalName,
|
|
180
|
+
}).promise();
|
|
181
|
+
if (this.lambdaFunctionResource.aliasesNames.length > 0) {
|
|
182
|
+
// we need to wait for the Version to finish publishing
|
|
183
|
+
const versionUpdate = await publishVersionPromise;
|
|
184
|
+
for (const alias of this.lambdaFunctionResource.aliasesNames) {
|
|
185
|
+
operations.push(lambda.updateAlias({
|
|
186
|
+
FunctionName: this.lambdaFunctionResource.physicalName,
|
|
187
|
+
Name: alias,
|
|
188
|
+
FunctionVersion: versionUpdate.Version,
|
|
189
|
+
}).promise());
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
else {
|
|
193
|
+
operations.push(publishVersionPromise);
|
|
194
|
+
}
|
|
195
|
+
}
|
|
196
|
+
else {
|
|
197
|
+
operations.push(updateFunctionCodePromise);
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
if (resource.tags !== undefined) {
|
|
201
|
+
const tagsToDelete = Object.entries(resource.tags.tagUpdates)
|
|
202
|
+
.filter(([_key, val]) => val === TagDeletion.DELETE)
|
|
203
|
+
.map(([key, _val]) => key);
|
|
204
|
+
const tagsToSet = {};
|
|
205
|
+
Object.entries(resource.tags.tagUpdates)
|
|
206
|
+
.filter(([_key, val]) => val !== TagDeletion.DELETE)
|
|
207
|
+
.forEach(([tagName, tagValue]) => {
|
|
208
|
+
tagsToSet[tagName] = tagValue;
|
|
209
|
+
});
|
|
210
|
+
if (tagsToDelete.length > 0) {
|
|
211
|
+
operations.push(lambda.untagResource({
|
|
212
|
+
Resource: this.lambdaFunctionResource.functionArn,
|
|
213
|
+
TagKeys: tagsToDelete,
|
|
214
|
+
}).promise());
|
|
215
|
+
}
|
|
216
|
+
if (Object.keys(tagsToSet).length > 0) {
|
|
217
|
+
operations.push(lambda.tagResource({
|
|
218
|
+
Resource: this.lambdaFunctionResource.functionArn,
|
|
219
|
+
Tags: tagsToSet,
|
|
220
|
+
}).promise());
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
// run all of our updates in parallel
|
|
224
|
+
return Promise.all(operations);
|
|
100
225
|
}
|
|
101
226
|
}
|
|
102
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lambda-functions.js","sourceRoot":"","sources":["lambda-functions.ts"],"names":[],"mappings":";;;AACA,qCAAwK;AAGxK;;;;;GAKG;AACI,KAAK,UAAU,kCAAkC,CACtD,SAAiB,EAAE,MAAmC,EAAE,mBAAmD;;IAE3G,MAAM,gBAAgB,GAAG,MAAM,8BAA8B,CAAC,MAAM,EAAE,mBAAmB,CAAC,CAAC;IAC3F,IAAI,OAAO,gBAAgB,KAAK,QAAQ,EAAE;QACxC,OAAO,gBAAgB,CAAC;KACzB;SAAM;QACL,6CAA6C;QAC7C,oCAAoC;QACpC,6BAA6B;QAC7B,kEAAkE;QAClE,IAAI,CAAC,6BAAoB,CAAC,MAAM,CAAC,EAAE;YACjC,OAAO,4BAAmB,CAAC,wBAAwB,CAAC;SACrD;QAED,MAAM,YAAY,GAAG,MAAM,sCAA6B,CAAC,SAAS,QAAE,MAAM,CAAC,QAAQ,CAAC,UAAU,0CAAE,YAAY,EAAE,mBAAmB,CAAC,CAAC;QACnI,IAAI,CAAC,YAAY,EAAE;YACjB,OAAO,4BAAmB,CAAC,wBAAwB,CAAC;SACrD;QAED,OAAO,IAAI,8BAA8B,CAAC;YACxC,YAAY,EAAE,YAAY;YAC1B,IAAI,EAAE,gBAAgB;SACvB,CAAC,CAAC;KACJ;AACH,CAAC;AAzBD,gFAyBC;AAED;;;;;;;;GAQG;AACH,KAAK,UAAU,8BAA8B,CAC3C,MAAmC,EAAE,mBAAmD;IAExF,MAAM,eAAe,GAAG,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC;IAC7C,IAAI,eAAe,KAAK,uBAAuB,EAAE;QAC/C,OAAO,4BAAmB,CAAC,wBAAwB,CAAC;KACrD;IAED;;;;;;;;;;OAUG;IACH,IAAI,QAAQ,GAAG,EAAE,EAAE,KAAK,GAAG,EAAE,CAAC;IAC9B,IAAI,mBAAmB,GAAG,KAAK,CAAC;IAChC,yDAAyD;IACzD,MAAM,eAAe,GAAG,MAAM,CAAC,eAAe,CAAC;IAC/C,KAAK,MAAM,eAAe,IAAI,eAAe,EAAE;QAC7C,MAAM,WAAW,GAAG,eAAe,CAAC,eAAe,CAAC,CAAC;QACrD,KAAK,MAAM,WAAW,IAAI,WAAW,CAAC,QAAQ,EAAE;YAC9C,QAAQ,WAAW,EAAE;gBACnB,KAAK,UAAU;oBACb,mBAAmB,GAAG,IAAI,CAAC;oBAC3B,QAAQ,GAAG,MAAM,mBAAmB,CAAC,qBAAqB,CAAC,WAAW,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC,CAAC;oBAC9F,MAAM;gBACR,KAAK,OAAO;oBACV,mBAAmB,GAAG,IAAI,CAAC;oBAC3B,KAAK,GAAG,MAAM,mBAAmB,CAAC,qBAAqB,CAAC,WAAW,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC,CAAC;oBAC3F,MAAM;gBACR;oBACE,OAAO,4BAAmB,CAAC,wBAAwB,CAAC;aACvD;SACF;KACF;IAED,OAAO,mBAAmB;QACxB,CAAC,CAAC;YACA,QAAQ;YACR,KAAK;SACN;QACD,CAAC,CAAC,4BAAmB,CAAC,UAAU,CAAC;AACrC,CAAC;AAYD,MAAM,8BAA8B;IAGlC,YAA6B,sBAA8C;QAA9C,2BAAsB,GAAtB,sBAAsB,CAAwB;QAF3D,YAAO,GAAG,iBAAiB,CAAC;IAG5C,CAAC;IAEM,KAAK,CAAC,KAAK,CAAC,GAAS;QAC1B,OAAO,GAAG,CAAC,MAAM,EAAE,CAAC,kBAAkB,CAAC;YACrC,YAAY,EAAE,IAAI,CAAC,sBAAsB,CAAC,YAAY;YACtD,QAAQ,EAAE,IAAI,CAAC,sBAAsB,CAAC,IAAI,CAAC,QAAQ;YACnD,KAAK,EAAE,IAAI,CAAC,sBAAsB,CAAC,IAAI,CAAC,KAAK;SAC9C,CAAC,CAAC,OAAO,EAAE,CAAC;IACf,CAAC;CACF","sourcesContent":["import { ISDK } from '../aws-auth';\nimport { assetMetadataChanged, ChangeHotswapImpact, ChangeHotswapResult, HotswapOperation, HotswappableChangeCandidate, establishResourcePhysicalName } from './common';\nimport { EvaluateCloudFormationTemplate } from './evaluate-cloudformation-template';\n\n/**\n * Returns `ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT` if the change cannot be short-circuited,\n * `ChangeHotswapImpact.IRRELEVANT` if the change is irrelevant from a short-circuit perspective\n * (like a change to CDKMetadata),\n * or a LambdaFunctionResource if the change can be short-circuited.\n */\nexport async function isHotswappableLambdaFunctionChange(\n  logicalId: string, change: HotswappableChangeCandidate, evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<ChangeHotswapResult> {\n  const lambdaCodeChange = await isLambdaFunctionCodeOnlyChange(change, evaluateCfnTemplate);\n  if (typeof lambdaCodeChange === 'string') {\n    return lambdaCodeChange;\n  } else {\n    // verify that the Asset changed - otherwise,\n    // it's a Code property-only change,\n    // but not to an asset change\n    // (for example, going from Code.fromAsset() to Code.fromInline())\n    if (!assetMetadataChanged(change)) {\n      return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n    }\n\n    const functionName = await establishResourcePhysicalName(logicalId, change.newValue.Properties?.FunctionName, evaluateCfnTemplate);\n    if (!functionName) {\n      return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n    }\n\n    return new LambdaFunctionHotswapOperation({\n      physicalName: functionName,\n      code: lambdaCodeChange,\n    });\n  }\n}\n\n/**\n * Returns `ChangeHotswapImpact.IRRELEVANT` if the change is not for a AWS::Lambda::Function,\n * but doesn't prevent short-circuiting\n * (like a change to CDKMetadata resource),\n * `ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT` if the change is to a AWS::Lambda::Function,\n * but not only to its Code property,\n * or a LambdaFunctionCode if the change is to a AWS::Lambda::Function,\n * and only affects its Code property.\n */\nasync function isLambdaFunctionCodeOnlyChange(\n  change: HotswappableChangeCandidate, evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<LambdaFunctionCode | ChangeHotswapImpact> {\n  const newResourceType = change.newValue.Type;\n  if (newResourceType !== 'AWS::Lambda::Function') {\n    return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n  }\n\n  /*\n   * On first glance, we would want to initialize these using the \"previous\" values (change.oldValue),\n   * in case only one of them changed, like the key, and the Bucket stayed the same.\n   * However, that actually fails for old-style synthesis, which uses CFN Parameters!\n   * Because the names of the Parameters depend on the hash of the Asset,\n   * the Parameters used for the \"old\" values no longer exist in `assetParams` at this point,\n   * which means we don't have the correct values available to evaluate the CFN expression with.\n   * Fortunately, the diff will always include both the s3Bucket and s3Key parts of the Lambda's Code property,\n   * even if only one of them was actually changed,\n   * which means we don't need the \"old\" values at all, and we can safely initialize these with just `''`.\n   */\n  let s3Bucket = '', s3Key = '';\n  let foundCodeDifference = false;\n  // Make sure only the code in the Lambda function changed\n  const propertyUpdates = change.propertyUpdates;\n  for (const updatedPropName in propertyUpdates) {\n    const updatedProp = propertyUpdates[updatedPropName];\n    for (const newPropName in updatedProp.newValue) {\n      switch (newPropName) {\n        case 'S3Bucket':\n          foundCodeDifference = true;\n          s3Bucket = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue[newPropName]);\n          break;\n        case 'S3Key':\n          foundCodeDifference = true;\n          s3Key = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue[newPropName]);\n          break;\n        default:\n          return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n      }\n    }\n  }\n\n  return foundCodeDifference\n    ? {\n      s3Bucket,\n      s3Key,\n    }\n    : ChangeHotswapImpact.IRRELEVANT;\n}\n\ninterface LambdaFunctionCode {\n  readonly s3Bucket: string;\n  readonly s3Key: string;\n}\n\ninterface LambdaFunctionResource {\n  readonly physicalName: string;\n  readonly code: LambdaFunctionCode;\n}\n\nclass LambdaFunctionHotswapOperation implements HotswapOperation {\n  public readonly service = 'lambda-function';\n\n  constructor(private readonly lambdaFunctionResource: LambdaFunctionResource) {\n  }\n\n  public async apply(sdk: ISDK): Promise<any> {\n    return sdk.lambda().updateFunctionCode({\n      FunctionName: this.lambdaFunctionResource.physicalName,\n      S3Bucket: this.lambdaFunctionResource.code.s3Bucket,\n      S3Key: this.lambdaFunctionResource.code.s3Key,\n    }).promise();\n  }\n}\n"]}
|
|
227
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lambda-functions.js","sourceRoot":"","sources":["lambda-functions.ts"],"names":[],"mappings":"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","sourcesContent":["import { flatMap } from '../../util';\nimport { ISDK } from '../aws-auth';\nimport { ChangeHotswapImpact, ChangeHotswapResult, establishResourcePhysicalName, HotswapOperation, HotswappableChangeCandidate } from './common';\nimport { EvaluateCloudFormationTemplate } from './evaluate-cloudformation-template';\n\n/**\n * Returns `ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT` if the change cannot be short-circuited,\n * `ChangeHotswapImpact.IRRELEVANT` if the change is irrelevant from a short-circuit perspective\n * (like a change to CDKMetadata),\n * or a LambdaFunctionResource if the change can be short-circuited.\n */\nexport async function isHotswappableLambdaFunctionChange(\n  logicalId: string, change: HotswappableChangeCandidate, evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<ChangeHotswapResult> {\n  // if the change is for a Lambda Version,\n  // ignore it by returning an empty hotswap operation -\n  // we will publish a new version when we get to hotswapping the actual Function this Version points to, below\n  // (Versions can't be changed in CloudFormation anyway, they're immutable)\n  if (change.newValue.Type === 'AWS::Lambda::Version') {\n    return ChangeHotswapImpact.IRRELEVANT;\n  }\n\n  // we handle Aliases specially too\n  if (change.newValue.Type === 'AWS::Lambda::Alias') {\n    return checkAliasHasVersionOnlyChange(change);\n  }\n\n  const lambdaCodeChange = await isLambdaFunctionCodeOnlyChange(change, evaluateCfnTemplate);\n  if (typeof lambdaCodeChange === 'string') {\n    return lambdaCodeChange;\n  }\n\n  const functionName = await establishResourcePhysicalName(logicalId, change.newValue.Properties?.FunctionName, evaluateCfnTemplate);\n  if (!functionName) {\n    return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n  }\n\n  const functionArn = await evaluateCfnTemplate.evaluateCfnExpression({\n    'Fn::Sub': 'arn:${AWS::Partition}:lambda:${AWS::Region}:${AWS::AccountId}:function:' + functionName,\n  });\n\n  // find all Lambda Versions that reference this Function\n  const versionsReferencingFunction = evaluateCfnTemplate.findReferencesTo(logicalId)\n    .filter(r => r.Type === 'AWS::Lambda::Version');\n  // find all Lambda Aliases that reference the above Versions\n  const aliasesReferencingVersions = flatMap(versionsReferencingFunction, v =>\n    evaluateCfnTemplate.findReferencesTo(v.LogicalId));\n  const aliasesNames = await Promise.all(aliasesReferencingVersions.map(a =>\n    evaluateCfnTemplate.evaluateCfnExpression(a.Properties?.Name)));\n\n  return new LambdaFunctionHotswapOperation({\n    physicalName: functionName,\n    functionArn: functionArn,\n    resource: lambdaCodeChange,\n    publishVersion: versionsReferencingFunction.length > 0,\n    aliasesNames,\n  });\n}\n\n/**\n * Returns  is a given Alias change is only in the 'FunctionVersion' property,\n * and `ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT` is the change is for any other property.\n */\nfunction checkAliasHasVersionOnlyChange(change: HotswappableChangeCandidate): ChangeHotswapResult {\n  for (const updatedPropName in change.propertyUpdates) {\n    if (updatedPropName !== 'FunctionVersion') {\n      return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n    }\n  }\n  return ChangeHotswapImpact.IRRELEVANT;\n}\n\n/**\n * Returns `ChangeHotswapImpact.IRRELEVANT` if the change is not for a AWS::Lambda::Function,\n * but doesn't prevent short-circuiting\n * (like a change to CDKMetadata resource),\n * `ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT` if the change is to a AWS::Lambda::Function,\n * but not only to its Code property,\n * or a LambdaFunctionCode if the change is to a AWS::Lambda::Function,\n * and only affects its Code property.\n */\nasync function isLambdaFunctionCodeOnlyChange(\n  change: HotswappableChangeCandidate, evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<LambdaFunctionChange | ChangeHotswapImpact> {\n  const newResourceType = change.newValue.Type;\n  if (newResourceType !== 'AWS::Lambda::Function') {\n    return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n  }\n\n  /*\n   * At first glance, we would want to initialize these using the \"previous\" values (change.oldValue),\n   * in case only one of them changed, like the key, and the Bucket stayed the same.\n   * However, that actually fails for old-style synthesis, which uses CFN Parameters!\n   * Because the names of the Parameters depend on the hash of the Asset,\n   * the Parameters used for the \"old\" values no longer exist in `assetParams` at this point,\n   * which means we don't have the correct values available to evaluate the CFN expression with.\n   * Fortunately, the diff will always include both the s3Bucket and s3Key parts of the Lambda's Code property,\n   * even if only one of them was actually changed,\n   * which means we don't need the \"old\" values at all, and we can safely initialize these with just `''`.\n   */\n  const propertyUpdates = change.propertyUpdates;\n  let code: LambdaFunctionCode | undefined = undefined;\n  let tags: LambdaFunctionTags | undefined = undefined;\n\n  for (const updatedPropName in propertyUpdates) {\n    const updatedProp = propertyUpdates[updatedPropName];\n\n    switch (updatedPropName) {\n      case 'Code':\n        let foundCodeDifference = false;\n        let s3Bucket = '', s3Key = '';\n\n        for (const newPropName in updatedProp.newValue) {\n          switch (newPropName) {\n            case 'S3Bucket':\n              foundCodeDifference = true;\n              s3Bucket = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue[newPropName]);\n              break;\n            case 'S3Key':\n              foundCodeDifference = true;\n              s3Key = await evaluateCfnTemplate.evaluateCfnExpression(updatedProp.newValue[newPropName]);\n              break;\n            default:\n              return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n          }\n        }\n        if (foundCodeDifference) {\n          code = {\n            s3Bucket,\n            s3Key,\n          };\n        }\n        break;\n      case 'Tags':\n        /*\n         * Tag updates are a bit odd; they manifest as two lists, are flagged only as\n         * `isDifferent`, and we have to reconcile them.\n         */\n        const tagUpdates: { [tag: string]: string | TagDeletion } = {};\n        if (updatedProp?.isDifferent) {\n          updatedProp.newValue.forEach((tag: CfnDiffTagValue) => {\n            tagUpdates[tag.Key] = tag.Value;\n          });\n\n          updatedProp.oldValue.forEach((tag: CfnDiffTagValue) => {\n            if (tagUpdates[tag.Key] === undefined) {\n              tagUpdates[tag.Key] = TagDeletion.DELETE;\n            }\n          });\n\n          tags = { tagUpdates };\n        }\n        break;\n      default:\n        return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n    }\n  }\n\n  return code || tags ? { code, tags } : ChangeHotswapImpact.IRRELEVANT;\n}\n\ninterface CfnDiffTagValue {\n  readonly Key: string;\n  readonly Value: string;\n}\n\ninterface LambdaFunctionCode {\n  readonly s3Bucket: string;\n  readonly s3Key: string;\n}\n\nenum TagDeletion {\n  DELETE = -1,\n}\n\ninterface LambdaFunctionTags {\n  readonly tagUpdates: { [tag : string] : string | TagDeletion };\n}\n\ninterface LambdaFunctionChange {\n  readonly code?: LambdaFunctionCode;\n  readonly tags?: LambdaFunctionTags;\n}\n\ninterface LambdaFunctionResource {\n  readonly physicalName: string;\n  readonly functionArn: string;\n  readonly resource: LambdaFunctionChange;\n  readonly publishVersion: boolean;\n  readonly aliasesNames: string[];\n}\n\nclass LambdaFunctionHotswapOperation implements HotswapOperation {\n  public readonly service = 'lambda-function';\n  public readonly resourceNames: string[];\n\n  constructor(private readonly lambdaFunctionResource: LambdaFunctionResource) {\n    this.resourceNames = [\n      `Lambda Function '${lambdaFunctionResource.physicalName}'`,\n      // add Version here if we're publishing a new one\n      ...(lambdaFunctionResource.publishVersion ? [`Lambda Version for Function '${lambdaFunctionResource.physicalName}'`] : []),\n      // add any Aliases that we are hotswapping here\n      ...lambdaFunctionResource.aliasesNames.map(alias => `Lambda Alias '${alias}' for Function '${lambdaFunctionResource.physicalName}'`),\n    ];\n  }\n\n  public async apply(sdk: ISDK): Promise<any> {\n    const lambda = sdk.lambda();\n    const resource = this.lambdaFunctionResource.resource;\n    const operations: Promise<any>[] = [];\n\n    if (resource.code !== undefined) {\n      const updateFunctionCodePromise = lambda.updateFunctionCode({\n        FunctionName: this.lambdaFunctionResource.physicalName,\n        S3Bucket: resource.code.s3Bucket,\n        S3Key: resource.code.s3Key,\n      }).promise();\n\n      // only if the code changed is there any point in publishing a new Version\n      if (this.lambdaFunctionResource.publishVersion) {\n        // we need to wait for the code update to be done before publishing a new Version\n        await updateFunctionCodePromise;\n        // if we don't wait for the Function to finish updating,\n        // we can get a \"The operation cannot be performed at this time. An update is in progress for resource:\"\n        // error when publishing a new Version\n        await lambda.waitFor('functionUpdated', {\n          FunctionName: this.lambdaFunctionResource.physicalName,\n        }).promise();\n\n        const publishVersionPromise = lambda.publishVersion({\n          FunctionName: this.lambdaFunctionResource.physicalName,\n        }).promise();\n\n        if (this.lambdaFunctionResource.aliasesNames.length > 0) {\n          // we need to wait for the Version to finish publishing\n          const versionUpdate = await publishVersionPromise;\n\n          for (const alias of this.lambdaFunctionResource.aliasesNames) {\n            operations.push(lambda.updateAlias({\n              FunctionName: this.lambdaFunctionResource.physicalName,\n              Name: alias,\n              FunctionVersion: versionUpdate.Version,\n            }).promise());\n          }\n        } else {\n          operations.push(publishVersionPromise);\n        }\n      } else {\n        operations.push(updateFunctionCodePromise);\n      }\n    }\n\n    if (resource.tags !== undefined) {\n      const tagsToDelete: string[] = Object.entries(resource.tags.tagUpdates)\n        .filter(([_key, val]) => val === TagDeletion.DELETE)\n        .map(([key, _val]) => key);\n\n      const tagsToSet: { [tag: string]: string } = {};\n      Object.entries(resource.tags!.tagUpdates)\n        .filter(([_key, val]) => val !== TagDeletion.DELETE)\n        .forEach(([tagName, tagValue]) => {\n          tagsToSet[tagName] = tagValue as string;\n        });\n\n      if (tagsToDelete.length > 0) {\n        operations.push(lambda.untagResource({\n          Resource: this.lambdaFunctionResource.functionArn,\n          TagKeys: tagsToDelete,\n        }).promise());\n      }\n\n      if (Object.keys(tagsToSet).length > 0) {\n        operations.push(lambda.tagResource({\n          Resource: this.lambdaFunctionResource.functionArn,\n          Tags: tagsToSet,\n        }).promise());\n      }\n    }\n\n    // run all of our updates in parallel\n    return Promise.all(operations);\n  }\n}\n"]}
|
|
@@ -34,6 +34,7 @@ class S3BucketDeploymentHotswapOperation {
|
|
|
34
34
|
this.functionName = functionName;
|
|
35
35
|
this.customResourceProperties = customResourceProperties;
|
|
36
36
|
this.service = 'custom-s3-deployment';
|
|
37
|
+
this.resourceNames = [`Contents of S3 Bucket '${this.customResourceProperties.DestinationBucketName}'`];
|
|
37
38
|
}
|
|
38
39
|
async apply(sdk) {
|
|
39
40
|
return sdk.lambda().invoke({
|
|
@@ -84,7 +85,7 @@ async function changeIsForS3DeployCustomResourcePolicy(iamPolicyLogicalId, chang
|
|
|
84
85
|
}
|
|
85
86
|
}
|
|
86
87
|
}
|
|
87
|
-
return
|
|
88
|
+
return common_1.ChangeHotswapImpact.IRRELEVANT;
|
|
88
89
|
}
|
|
89
90
|
function stringifyObject(obj) {
|
|
90
91
|
if (obj == null) {
|
|
@@ -102,12 +103,4 @@ function stringifyObject(obj) {
|
|
|
102
103
|
}
|
|
103
104
|
return ret;
|
|
104
105
|
}
|
|
105
|
-
class EmptyHotswapOperation {
|
|
106
|
-
constructor() {
|
|
107
|
-
this.service = 'empty';
|
|
108
|
-
}
|
|
109
|
-
async apply(sdk) {
|
|
110
|
-
return Promise.resolve(sdk);
|
|
111
|
-
}
|
|
112
|
-
}
|
|
113
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"s3-bucket-deployments.js","sourceRoot":"","sources":["s3-bucket-deployments.ts"],"names":[],"mappings":";;;AACA,qCAAsJ;AAGtJ;;;GAGG;AACU,QAAA,eAAe,GAAG,+BAA+B,CAAC;AAExD,KAAK,UAAU,sCAAsC,CAC1D,SAAiB,EAAE,MAAmC,EAAE,mBAAmD;;IAE3G,kGAAkG;IAClG,uFAAuF;IACvF,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,KAAK,kBAAkB,EAAE;QAC/C,OAAO,uCAAuC,CAAC,SAAS,EAAE,MAAM,EAAE,mBAAmB,CAAC,CAAC;KACxF;IAED,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,KAAK,6BAA6B,EAAE;QAC1D,OAAO,4BAAmB,CAAC,wBAAwB,CAAC;KACrD;IAED,wHAAwH;IACxH,MAAM,YAAY,GAAG,MAAM,mBAAmB,CAAC,qBAAqB,OAAC,MAAM,CAAC,QAAQ,CAAC,UAAU,0CAAE,YAAY,CAAC,CAAC;IAC/G,IAAI,CAAC,YAAY,EAAE;QACjB,OAAO,4BAAmB,CAAC,wBAAwB,CAAC;KACrD;IAED,MAAM,wBAAwB,GAAG,MAAM,mBAAmB,CAAC,qBAAqB,CAAC;QAC/E,GAAG,MAAM,CAAC,QAAQ,CAAC,UAAU;QAC7B,YAAY,EAAE,SAAS;KACxB,CAAC,CAAC;IAEH,OAAO,IAAI,kCAAkC,CAAC,YAAY,EAAE,wBAAwB,CAAC,CAAC;AACxF,CAAC;AAzBD,wFAyBC;AAED,MAAM,kCAAkC;IAGtC,YAA6B,YAAoB,EAAmB,wBAA6B;QAApE,iBAAY,GAAZ,YAAY,CAAQ;QAAmB,6BAAwB,GAAxB,wBAAwB,CAAK;QAFjF,YAAO,GAAG,sBAAsB,CAAC;IAGjD,CAAC;IAEM,KAAK,CAAC,KAAK,CAAC,GAAS;QAC1B,OAAO,GAAG,CAAC,MAAM,EAAE,CAAC,MAAM,CAAC;YACzB,YAAY,EAAE,IAAI,CAAC,YAAY;YAC/B,kFAAkF;YAClF,OAAO,EAAE,IAAI,CAAC,SAAS,CAAC;gBACtB,WAAW,EAAE,QAAQ;gBACrB,WAAW,EAAE,uBAAe;gBAC5B,kBAAkB,EAAE,uBAAe;gBACnC,OAAO,EAAE,uBAAe;gBACxB,SAAS,EAAE,uBAAe;gBAC1B,iBAAiB,EAAE,uBAAe;gBAClC,kBAAkB,EAAE,eAAe,CAAC,IAAI,CAAC,wBAAwB,CAAC;aACnE,CAAC;SACH,CAAC,CAAC,OAAO,EAAE,CAAC;IACf,CAAC;CACF;AAED,KAAK,UAAU,uCAAuC,CACpD,kBAA0B,EAAE,MAAmC,EAAE,mBAAmD;;IAEpH,MAAM,KAAK,SAAG,MAAM,CAAC,QAAQ,CAAC,UAAU,0CAAE,KAAK,CAAC;IAChD,IAAI,CAAC,KAAK,EAAE;QACV,OAAO,4BAAmB,CAAC,wBAAwB,CAAC;KACrD;IAED,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE;QACxB,MAAM,aAAa,GAAG,MAAM,mBAAmB,CAAC,4BAA4B,CAAC,MAAM,mBAAmB,CAAC,qBAAqB,CAAC,IAAI,CAAC,CAAC,CAAC;QACpI,IAAI,CAAC,aAAa,EAAE;YAClB,OAAO,4BAAmB,CAAC,wBAAwB,CAAC;SACrD;QAED,MAAM,QAAQ,GAAG,mBAAmB,CAAC,gBAAgB,CAAC,aAAa,CAAC,CAAC;QACrE,KAAK,MAAM,OAAO,IAAI,QAAQ,EAAE;YAC9B,IAAI,OAAO,CAAC,IAAI,KAAK,uBAAuB,EAAE;gBAC5C,MAAM,UAAU,GAAG,mBAAmB,CAAC,gBAAgB,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;gBAC3E,KAAK,MAAM,SAAS,IAAI,UAAU,EAAE;oBAClC,gGAAgG;oBAChG,kCAAkC;oBAClC,IAAI,SAAS,CAAC,IAAI,KAAK,6BAA6B,EAAE;wBACpD,OAAO,4BAAmB,CAAC,wBAAwB,CAAC;qBACrD;iBACF;aACF;iBAAM,IAAI,OAAO,CAAC,IAAI,KAAK,kBAAkB,EAAE;gBAC9C,IAAI,OAAO,CAAC,SAAS,KAAK,kBAAkB,EAAE;oBAC5C,OAAO,4BAAmB,CAAC,wBAAwB,CAAC;iBACrD;aACF;iBAAM;gBACL,OAAO,4BAAmB,CAAC,wBAAwB,CAAC;aACrD;SACF;KACF;IAED,OAAO,IAAI,qBAAqB,EAAE,CAAC;AACrC,CAAC;AAED,SAAS,eAAe,CAAC,GAAQ;IAC/B,IAAI,GAAG,IAAI,IAAI,EAAE;QACf,OAAO,GAAG,CAAC;KACZ;IACD,IAAI,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE;QACtB,OAAO,GAAG,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC;KACjC;IACD,IAAI,OAAO,GAAG,KAAK,QAAQ,EAAE;QAC3B,OAAO,GAAG,CAAC,QAAQ,EAAE,CAAC;KACvB;IAED,MAAM,GAAG,GAAyB,EAAE,CAAC;IACrC,KAAK,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE;QACxC,GAAG,CAAC,CAAC,CAAC,GAAG,eAAe,CAAC,CAAC,CAAC,CAAC;KAC7B;IACD,OAAO,GAAG,CAAC;AACb,CAAC;AAED,MAAM,qBAAqB;IAA3B;QACW,YAAO,GAAG,OAAO,CAAC;IAI7B,CAAC;IAHQ,KAAK,CAAC,KAAK,CAAC,GAAS;QAC1B,OAAO,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;IAC9B,CAAC;CACF","sourcesContent":["import { ISDK } from '../aws-auth';\nimport { ChangeHotswapImpact, ChangeHotswapResult, HotswapOperation, HotswappableChangeCandidate/*, establishResourcePhysicalName*/ } from './common';\nimport { EvaluateCloudFormationTemplate } from './evaluate-cloudformation-template';\n\n/**\n * This means that the value is required to exist by CloudFormation's API (or our S3 Bucket Deployment Lambda)\n * but the actual value specified is irrelevant\n */\nexport const REQUIRED_BY_CFN = 'required-to-be-present-by-cfn';\n\nexport async function isHotswappableS3BucketDeploymentChange(\n  logicalId: string, change: HotswappableChangeCandidate, evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<ChangeHotswapResult> {\n  // In old-style synthesis, the policy used by the lambda to copy assets Ref's the assets directly,\n  // meaning that the changes made to the Policy are artifacts that can be safely ignored\n  if (change.newValue.Type === 'AWS::IAM::Policy') {\n    return changeIsForS3DeployCustomResourcePolicy(logicalId, change, evaluateCfnTemplate);\n  }\n\n  if (change.newValue.Type !== 'Custom::CDKBucketDeployment') {\n    return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n  }\n\n  // note that this gives the ARN of the lambda, not the name. This is fine though, the invoke() sdk call will take either\n  const functionName = await evaluateCfnTemplate.evaluateCfnExpression(change.newValue.Properties?.ServiceToken);\n  if (!functionName) {\n    return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n  }\n\n  const customResourceProperties = await evaluateCfnTemplate.evaluateCfnExpression({\n    ...change.newValue.Properties,\n    ServiceToken: undefined,\n  });\n\n  return new S3BucketDeploymentHotswapOperation(functionName, customResourceProperties);\n}\n\nclass S3BucketDeploymentHotswapOperation implements HotswapOperation {\n  public readonly service = 'custom-s3-deployment';\n\n  constructor(private readonly functionName: string, private readonly customResourceProperties: any) {\n  }\n\n  public async apply(sdk: ISDK): Promise<any> {\n    return sdk.lambda().invoke({\n      FunctionName: this.functionName,\n      // Lambda refuses to take a direct JSON object and requires it to be stringify()'d\n      Payload: JSON.stringify({\n        RequestType: 'Update',\n        ResponseURL: REQUIRED_BY_CFN,\n        PhysicalResourceId: REQUIRED_BY_CFN,\n        StackId: REQUIRED_BY_CFN,\n        RequestId: REQUIRED_BY_CFN,\n        LogicalResourceId: REQUIRED_BY_CFN,\n        ResourceProperties: stringifyObject(this.customResourceProperties), // JSON.stringify() doesn't turn the actual objects to strings, but the lambda expects strings\n      }),\n    }).promise();\n  }\n}\n\nasync function changeIsForS3DeployCustomResourcePolicy(\n  iamPolicyLogicalId: string, change: HotswappableChangeCandidate, evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<ChangeHotswapResult> {\n  const roles = change.newValue.Properties?.Roles;\n  if (!roles) {\n    return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n  }\n\n  for (const role of roles) {\n    const roleLogicalId = await evaluateCfnTemplate.findLogicalIdForPhysicalName(await evaluateCfnTemplate.evaluateCfnExpression(role));\n    if (!roleLogicalId) {\n      return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n    }\n\n    const roleRefs = evaluateCfnTemplate.findReferencesTo(roleLogicalId);\n    for (const roleRef of roleRefs) {\n      if (roleRef.Type === 'AWS::Lambda::Function') {\n        const lambdaRefs = evaluateCfnTemplate.findReferencesTo(roleRef.LogicalId);\n        for (const lambdaRef of lambdaRefs) {\n          // If S3Deployment -> Lambda -> Role and IAM::Policy -> Role, then this IAM::Policy change is an\n          // artifact of old-style synthesis\n          if (lambdaRef.Type !== 'Custom::CDKBucketDeployment') {\n            return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n          }\n        }\n      } else if (roleRef.Type === 'AWS::IAM::Policy') {\n        if (roleRef.LogicalId !== iamPolicyLogicalId) {\n          return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n        }\n      } else {\n        return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n      }\n    }\n  }\n\n  return new EmptyHotswapOperation();\n}\n\nfunction stringifyObject(obj: any): any {\n  if (obj == null) {\n    return obj;\n  }\n  if (Array.isArray(obj)) {\n    return obj.map(stringifyObject);\n  }\n  if (typeof obj !== 'object') {\n    return obj.toString();\n  }\n\n  const ret: { [k: string]: any } = {};\n  for (const [k, v] of Object.entries(obj)) {\n    ret[k] = stringifyObject(v);\n  }\n  return ret;\n}\n\nclass EmptyHotswapOperation implements HotswapOperation {\n  readonly service = 'empty';\n  public async apply(sdk: ISDK): Promise<any> {\n    return Promise.resolve(sdk);\n  }\n}\n"]}
|
|
106
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"s3-bucket-deployments.js","sourceRoot":"","sources":["s3-bucket-deployments.ts"],"names":[],"mappings":"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","sourcesContent":["import { ISDK } from '../aws-auth';\nimport { ChangeHotswapImpact, ChangeHotswapResult, HotswapOperation, HotswappableChangeCandidate } from './common';\nimport { EvaluateCloudFormationTemplate } from './evaluate-cloudformation-template';\n\n/**\n * This means that the value is required to exist by CloudFormation's API (or our S3 Bucket Deployment Lambda)\n * but the actual value specified is irrelevant\n */\nexport const REQUIRED_BY_CFN = 'required-to-be-present-by-cfn';\n\nexport async function isHotswappableS3BucketDeploymentChange(\n  logicalId: string, change: HotswappableChangeCandidate, evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<ChangeHotswapResult> {\n  // In old-style synthesis, the policy used by the lambda to copy assets Ref's the assets directly,\n  // meaning that the changes made to the Policy are artifacts that can be safely ignored\n  if (change.newValue.Type === 'AWS::IAM::Policy') {\n    return changeIsForS3DeployCustomResourcePolicy(logicalId, change, evaluateCfnTemplate);\n  }\n\n  if (change.newValue.Type !== 'Custom::CDKBucketDeployment') {\n    return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n  }\n\n  // note that this gives the ARN of the lambda, not the name. This is fine though, the invoke() sdk call will take either\n  const functionName = await evaluateCfnTemplate.evaluateCfnExpression(change.newValue.Properties?.ServiceToken);\n  if (!functionName) {\n    return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n  }\n\n  const customResourceProperties = await evaluateCfnTemplate.evaluateCfnExpression({\n    ...change.newValue.Properties,\n    ServiceToken: undefined,\n  });\n\n  return new S3BucketDeploymentHotswapOperation(functionName, customResourceProperties);\n}\n\nclass S3BucketDeploymentHotswapOperation implements HotswapOperation {\n  public readonly service = 'custom-s3-deployment';\n  public readonly resourceNames: string[];\n\n  constructor(private readonly functionName: string, private readonly customResourceProperties: any) {\n    this.resourceNames = [`Contents of S3 Bucket '${this.customResourceProperties.DestinationBucketName}'`];\n  }\n\n  public async apply(sdk: ISDK): Promise<any> {\n    return sdk.lambda().invoke({\n      FunctionName: this.functionName,\n      // Lambda refuses to take a direct JSON object and requires it to be stringify()'d\n      Payload: JSON.stringify({\n        RequestType: 'Update',\n        ResponseURL: REQUIRED_BY_CFN,\n        PhysicalResourceId: REQUIRED_BY_CFN,\n        StackId: REQUIRED_BY_CFN,\n        RequestId: REQUIRED_BY_CFN,\n        LogicalResourceId: REQUIRED_BY_CFN,\n        ResourceProperties: stringifyObject(this.customResourceProperties), // JSON.stringify() doesn't turn the actual objects to strings, but the lambda expects strings\n      }),\n    }).promise();\n  }\n}\n\nasync function changeIsForS3DeployCustomResourcePolicy(\n  iamPolicyLogicalId: string, change: HotswappableChangeCandidate, evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n): Promise<ChangeHotswapResult> {\n  const roles = change.newValue.Properties?.Roles;\n  if (!roles) {\n    return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n  }\n\n  for (const role of roles) {\n    const roleLogicalId = await evaluateCfnTemplate.findLogicalIdForPhysicalName(await evaluateCfnTemplate.evaluateCfnExpression(role));\n    if (!roleLogicalId) {\n      return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n    }\n\n    const roleRefs = evaluateCfnTemplate.findReferencesTo(roleLogicalId);\n    for (const roleRef of roleRefs) {\n      if (roleRef.Type === 'AWS::Lambda::Function') {\n        const lambdaRefs = evaluateCfnTemplate.findReferencesTo(roleRef.LogicalId);\n        for (const lambdaRef of lambdaRefs) {\n          // If S3Deployment -> Lambda -> Role and IAM::Policy -> Role, then this IAM::Policy change is an\n          // artifact of old-style synthesis\n          if (lambdaRef.Type !== 'Custom::CDKBucketDeployment') {\n            return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n          }\n        }\n      } else if (roleRef.Type === 'AWS::IAM::Policy') {\n        if (roleRef.LogicalId !== iamPolicyLogicalId) {\n          return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n        }\n      } else {\n        return ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT;\n      }\n    }\n  }\n\n  return ChangeHotswapImpact.IRRELEVANT;\n}\n\nfunction stringifyObject(obj: any): any {\n  if (obj == null) {\n    return obj;\n  }\n  if (Array.isArray(obj)) {\n    return obj.map(stringifyObject);\n  }\n  if (typeof obj !== 'object') {\n    return obj.toString();\n  }\n\n  const ret: { [k: string]: any } = {};\n  for (const [k, v] of Object.entries(obj)) {\n    ret[k] = stringifyObject(v);\n  }\n  return ret;\n}\n"]}
|
|
@@ -42,6 +42,7 @@ class StateMachineHotswapOperation {
|
|
|
42
42
|
constructor(stepFunctionResource) {
|
|
43
43
|
this.stepFunctionResource = stepFunctionResource;
|
|
44
44
|
this.service = 'stepfunctions-state-machine';
|
|
45
|
+
this.resourceNames = [`StateMachine '${this.stepFunctionResource.stateMachineArn.split(':')[6]}'`];
|
|
45
46
|
}
|
|
46
47
|
async apply(sdk) {
|
|
47
48
|
// not passing the optional properties leaves them unchanged
|
|
@@ -51,4 +52,4 @@ class StateMachineHotswapOperation {
|
|
|
51
52
|
}).promise();
|
|
52
53
|
}
|
|
53
54
|
}
|
|
54
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
55
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RlcGZ1bmN0aW9ucy1zdGF0ZS1tYWNoaW5lcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInN0ZXBmdW5jdGlvbnMtc3RhdGUtbWFjaGluZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQ0EscUNBQW1IO0FBRzVHLEtBQUssVUFBVSxnQ0FBZ0MsQ0FDcEQsU0FBaUIsRUFBRSxNQUFtQyxFQUFFLG1CQUFtRDs7SUFFM0csTUFBTSw0QkFBNEIsR0FBRyxNQUFNLGtDQUFrQyxDQUFDLE1BQU0sRUFBRSxtQkFBbUIsQ0FBQyxDQUFDO0lBQzNHLElBQUksNEJBQTRCLEtBQUssNEJBQW1CLENBQUMsd0JBQXdCO1FBQzdFLDRCQUE0QixLQUFLLDRCQUFtQixDQUFDLFVBQVUsRUFBRTtRQUNuRSxPQUFPLDRCQUE0QixDQUFDO0tBQ3JDO0lBRUQsTUFBTSw2QkFBNkIsZUFBRyxNQUFNLENBQUMsUUFBUSwwQ0FBRSxVQUFVLDBDQUFFLGdCQUFnQixDQUFDO0lBQ3BGLE1BQU0sZUFBZSxHQUFHLDZCQUE2QjtRQUNuRCxDQUFDLENBQUMsTUFBTSxtQkFBbUIsQ0FBQyxxQkFBcUIsQ0FBQztZQUNoRCxTQUFTLEVBQUUsNkVBQTZFLEdBQUcsNkJBQTZCO1NBQ3pILENBQUM7UUFDRixDQUFDLENBQUMsTUFBTSxtQkFBbUIsQ0FBQyxtQkFBbUIsQ0FBQyxTQUFTLENBQUMsQ0FBQztJQUU3RCxJQUFJLENBQUMsZUFBZSxFQUFFO1FBQ3BCLE9BQU8sNEJBQW1CLENBQUMsd0JBQXdCLENBQUM7S0FDckQ7SUFFRCxPQUFPLElBQUksNEJBQTRCLENBQUM7UUFDdEMsVUFBVSxFQUFFLDRCQUE0QjtRQUN4QyxlQUFlLEVBQUUsZUFBZTtLQUNqQyxDQUFDLENBQUM7QUFDTCxDQUFDO0FBeEJELDRFQXdCQztBQUVELEtBQUssVUFBVSxrQ0FBa0MsQ0FDL0MsTUFBbUMsRUFBRSxtQkFBbUQ7SUFFeEYsTUFBTSxlQUFlLEdBQUcsTUFBTSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUM7SUFDN0MsSUFBSSxlQUFlLEtBQUssa0NBQWtDLEVBQUU7UUFDMUQsT0FBTyw0QkFBbUIsQ0FBQyx3QkFBd0IsQ0FBQztLQUNyRDtJQUVELE1BQU0sZUFBZSxHQUFHLE1BQU0sQ0FBQyxlQUFlLENBQUM7SUFDL0MsS0FBSyxNQUFNLGVBQWUsSUFBSSxlQUFlLEVBQUU7UUFDN0Msd0VBQXdFO1FBQ3hFLElBQUksZUFBZSxLQUFLLGtCQUFrQixFQUFFO1lBQzFDLE9BQU8sNEJBQW1CLENBQUMsd0JBQXdCLENBQUM7U0FDckQ7S0FDRjtJQUVELE9BQU8sbUJBQW1CLENBQUMscUJBQXFCLENBQUMsZUFBZSxDQUFDLGdCQUFnQixDQUFDLFFBQVEsQ0FBQyxDQUFDO0FBQzlGLENBQUM7QUFPRCxNQUFNLDRCQUE0QjtJQUloQyxZQUE2QixvQkFBMEM7UUFBMUMseUJBQW9CLEdBQXBCLG9CQUFvQixDQUFzQjtRQUh2RCxZQUFPLEdBQUcsNkJBQTZCLENBQUM7UUFJdEQsSUFBSSxDQUFDLGFBQWEsR0FBRyxDQUFDLGlCQUFpQixJQUFJLENBQUMsb0JBQW9CLENBQUMsZUFBZSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDckcsQ0FBQztJQUVNLEtBQUssQ0FBQyxLQUFLLENBQUMsR0FBUztRQUMxQiw0REFBNEQ7UUFDNUQsT0FBTyxHQUFHLENBQUMsYUFBYSxFQUFFLENBQUMsa0JBQWtCLENBQUM7WUFDNUMsZUFBZSxFQUFFLElBQUksQ0FBQyxvQkFBb0IsQ0FBQyxlQUFlO1lBQzFELFVBQVUsRUFBRSxJQUFJLENBQUMsb0JBQW9CLENBQUMsVUFBVTtTQUNqRCxDQUFDLENBQUMsT0FBTyxFQUFFLENBQUM7SUFDZixDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJU0RLIH0gZnJvbSAnLi4vYXdzLWF1dGgnO1xuaW1wb3J0IHsgQ2hhbmdlSG90c3dhcEltcGFjdCwgQ2hhbmdlSG90c3dhcFJlc3VsdCwgSG90c3dhcE9wZXJhdGlvbiwgSG90c3dhcHBhYmxlQ2hhbmdlQ2FuZGlkYXRlIH0gZnJvbSAnLi9jb21tb24nO1xuaW1wb3J0IHsgRXZhbHVhdGVDbG91ZEZvcm1hdGlvblRlbXBsYXRlIH0gZnJvbSAnLi9ldmFsdWF0ZS1jbG91ZGZvcm1hdGlvbi10ZW1wbGF0ZSc7XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBpc0hvdHN3YXBwYWJsZVN0YXRlTWFjaGluZUNoYW5nZShcbiAgbG9naWNhbElkOiBzdHJpbmcsIGNoYW5nZTogSG90c3dhcHBhYmxlQ2hhbmdlQ2FuZGlkYXRlLCBldmFsdWF0ZUNmblRlbXBsYXRlOiBFdmFsdWF0ZUNsb3VkRm9ybWF0aW9uVGVtcGxhdGUsXG4pOiBQcm9taXNlPENoYW5nZUhvdHN3YXBSZXN1bHQ+IHtcbiAgY29uc3Qgc3RhdGVNYWNoaW5lRGVmaW5pdGlvbkNoYW5nZSA9IGF3YWl0IGlzU3RhdGVNYWNoaW5lRGVmaW5pdGlvbk9ubHlDaGFuZ2UoY2hhbmdlLCBldmFsdWF0ZUNmblRlbXBsYXRlKTtcbiAgaWYgKHN0YXRlTWFjaGluZURlZmluaXRpb25DaGFuZ2UgPT09IENoYW5nZUhvdHN3YXBJbXBhY3QuUkVRVUlSRVNfRlVMTF9ERVBMT1lNRU5UIHx8XG4gICAgICBzdGF0ZU1hY2hpbmVEZWZpbml0aW9uQ2hhbmdlID09PSBDaGFuZ2VIb3Rzd2FwSW1wYWN0LklSUkVMRVZBTlQpIHtcbiAgICByZXR1cm4gc3RhdGVNYWNoaW5lRGVmaW5pdGlvbkNoYW5nZTtcbiAgfVxuXG4gIGNvbnN0IHN0YXRlTWFjaGluZU5hbWVJbkNmblRlbXBsYXRlID0gY2hhbmdlLm5ld1ZhbHVlPy5Qcm9wZXJ0aWVzPy5TdGF0ZU1hY2hpbmVOYW1lO1xuICBjb25zdCBzdGF0ZU1hY2hpbmVBcm4gPSBzdGF0ZU1hY2hpbmVOYW1lSW5DZm5UZW1wbGF0ZVxuICAgID8gYXdhaXQgZXZhbHVhdGVDZm5UZW1wbGF0ZS5ldmFsdWF0ZUNmbkV4cHJlc3Npb24oe1xuICAgICAgJ0ZuOjpTdWInOiAnYXJuOiR7QVdTOjpQYXJ0aXRpb259OnN0YXRlczoke0FXUzo6UmVnaW9ufToke0FXUzo6QWNjb3VudElkfTpzdGF0ZU1hY2hpbmU6JyArIHN0YXRlTWFjaGluZU5hbWVJbkNmblRlbXBsYXRlLFxuICAgIH0pXG4gICAgOiBhd2FpdCBldmFsdWF0ZUNmblRlbXBsYXRlLmZpbmRQaHlzaWNhbE5hbWVGb3IobG9naWNhbElkKTtcblxuICBpZiAoIXN0YXRlTWFjaGluZUFybikge1xuICAgIHJldHVybiBDaGFuZ2VIb3Rzd2FwSW1wYWN0LlJFUVVJUkVTX0ZVTExfREVQTE9ZTUVOVDtcbiAgfVxuXG4gIHJldHVybiBuZXcgU3RhdGVNYWNoaW5lSG90c3dhcE9wZXJhdGlvbih7XG4gICAgZGVmaW5pdGlvbjogc3RhdGVNYWNoaW5lRGVmaW5pdGlvbkNoYW5nZSxcbiAgICBzdGF0ZU1hY2hpbmVBcm46IHN0YXRlTWFjaGluZUFybixcbiAgfSk7XG59XG5cbmFzeW5jIGZ1bmN0aW9uIGlzU3RhdGVNYWNoaW5lRGVmaW5pdGlvbk9ubHlDaGFuZ2UoXG4gIGNoYW5nZTogSG90c3dhcHBhYmxlQ2hhbmdlQ2FuZGlkYXRlLCBldmFsdWF0ZUNmblRlbXBsYXRlOiBFdmFsdWF0ZUNsb3VkRm9ybWF0aW9uVGVtcGxhdGUsXG4pOiBQcm9taXNlPHN0cmluZyB8IENoYW5nZUhvdHN3YXBJbXBhY3Q+IHtcbiAgY29uc3QgbmV3UmVzb3VyY2VUeXBlID0gY2hhbmdlLm5ld1ZhbHVlLlR5cGU7XG4gIGlmIChuZXdSZXNvdXJjZVR5cGUgIT09ICdBV1M6OlN0ZXBGdW5jdGlvbnM6OlN0YXRlTWFjaGluZScpIHtcbiAgICByZXR1cm4gQ2hhbmdlSG90c3dhcEltcGFjdC5SRVFVSVJFU19GVUxMX0RFUExPWU1FTlQ7XG4gIH1cblxuICBjb25zdCBwcm9wZXJ0eVVwZGF0ZXMgPSBjaGFuZ2UucHJvcGVydHlVcGRhdGVzO1xuICBmb3IgKGNvbnN0IHVwZGF0ZWRQcm9wTmFtZSBpbiBwcm9wZXJ0eVVwZGF0ZXMpIHtcbiAgICAvLyBlbnN1cmUgdGhhdCBvbmx5IGNoYW5nZXMgdG8gdGhlIGRlZmluaXRpb24gc3RyaW5nIHJlc3VsdCBpbiBhIGhvdHN3YXBcbiAgICBpZiAodXBkYXRlZFByb3BOYW1lICE9PSAnRGVmaW5pdGlvblN0cmluZycpIHtcbiAgICAgIHJldHVybiBDaGFuZ2VIb3Rzd2FwSW1wYWN0LlJFUVVJUkVTX0ZVTExfREVQTE9ZTUVOVDtcbiAgICB9XG4gIH1cblxuICByZXR1cm4gZXZhbHVhdGVDZm5UZW1wbGF0ZS5ldmFsdWF0ZUNmbkV4cHJlc3Npb24ocHJvcGVydHlVcGRhdGVzLkRlZmluaXRpb25TdHJpbmcubmV3VmFsdWUpO1xufVxuXG5pbnRlcmZhY2UgU3RhdGVNYWNoaW5lUmVzb3VyY2Uge1xuICByZWFkb25seSBzdGF0ZU1hY2hpbmVBcm46IHN0cmluZztcbiAgcmVhZG9ubHkgZGVmaW5pdGlvbjogc3RyaW5nO1xufVxuXG5jbGFzcyBTdGF0ZU1hY2hpbmVIb3Rzd2FwT3BlcmF0aW9uIGltcGxlbWVudHMgSG90c3dhcE9wZXJhdGlvbiB7XG4gIHB1YmxpYyByZWFkb25seSBzZXJ2aWNlID0gJ3N0ZXBmdW5jdGlvbnMtc3RhdGUtbWFjaGluZSc7XG4gIHB1YmxpYyByZWFkb25seSByZXNvdXJjZU5hbWVzOiBzdHJpbmdbXTtcblxuICBjb25zdHJ1Y3Rvcihwcml2YXRlIHJlYWRvbmx5IHN0ZXBGdW5jdGlvblJlc291cmNlOiBTdGF0ZU1hY2hpbmVSZXNvdXJjZSkge1xuICAgIHRoaXMucmVzb3VyY2VOYW1lcyA9IFtgU3RhdGVNYWNoaW5lICcke3RoaXMuc3RlcEZ1bmN0aW9uUmVzb3VyY2Uuc3RhdGVNYWNoaW5lQXJuLnNwbGl0KCc6JylbNl19J2BdO1xuICB9XG5cbiAgcHVibGljIGFzeW5jIGFwcGx5KHNkazogSVNESyk6IFByb21pc2U8YW55PiB7XG4gICAgLy8gbm90IHBhc3NpbmcgdGhlIG9wdGlvbmFsIHByb3BlcnRpZXMgbGVhdmVzIHRoZW0gdW5jaGFuZ2VkXG4gICAgcmV0dXJuIHNkay5zdGVwRnVuY3Rpb25zKCkudXBkYXRlU3RhdGVNYWNoaW5lKHtcbiAgICAgIHN0YXRlTWFjaGluZUFybjogdGhpcy5zdGVwRnVuY3Rpb25SZXNvdXJjZS5zdGF0ZU1hY2hpbmVBcm4sXG4gICAgICBkZWZpbml0aW9uOiB0aGlzLnN0ZXBGdW5jdGlvblJlc291cmNlLmRlZmluaXRpb24sXG4gICAgfSkucHJvbWlzZSgpO1xuICB9XG59XG4iXX0=
|
|
@@ -1,8 +1,6 @@
|
|
|
1
|
-
import * as cfn_diff from '@aws-cdk/cloudformation-diff';
|
|
2
1
|
import * as cxapi from '@aws-cdk/cx-api';
|
|
3
2
|
import { SdkProvider } from './aws-auth';
|
|
4
3
|
import { DeployStackResult } from './deploy-stack';
|
|
5
|
-
import { ChangeHotswapImpact, HotswappableChangeCandidate } from './hotswap/common';
|
|
6
4
|
import { CloudFormationStack } from './util/cloudformation';
|
|
7
5
|
/**
|
|
8
6
|
* Perform a hotswap deployment,
|
|
@@ -14,8 +12,3 @@ import { CloudFormationStack } from './util/cloudformation';
|
|
|
14
12
|
export declare function tryHotswapDeployment(sdkProvider: SdkProvider, assetParams: {
|
|
15
13
|
[key: string]: string;
|
|
16
14
|
}, cloudFormationStack: CloudFormationStack, stackArtifact: cxapi.CloudFormationStackArtifact): Promise<DeployStackResult | undefined>;
|
|
17
|
-
/**
|
|
18
|
-
* returns `ChangeHotswapImpact.REQUIRES_FULL_DEPLOYMENT` if a resource was deleted, or a change that we cannot short-circuit occured.
|
|
19
|
-
* Returns `ChangeHotswapImpact.IRRELEVANT` if a change that does not impact shortcircuiting occured, such as a metadata change.
|
|
20
|
-
*/
|
|
21
|
-
export declare function isCandidateForHotswapping(change: cfn_diff.ResourceDifference): HotswappableChangeCandidate | ChangeHotswapImpact;
|