@trautonen/cdk-dns-validated-certificate 0.0.52 → 0.1.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/.jsii +90 -73
- package/API.md +129 -65
- package/README.md +31 -4
- package/assets/certificate-requestor.lambda/index.js +116 -37
- package/assets/certificate-requestor.lambda/index.js.map +2 -2
- package/lib/certificate-requestor.lambda.d.ts +8 -4
- package/lib/certificate-requestor.lambda.js +75 -33
- package/lib/dns-validated-certificate.d.ts +73 -40
- package/lib/dns-validated-certificate.js +73 -37
- package/lib/utils.d.ts +4 -0
- package/lib/utils.js +33 -2
- package/package.json +2 -2
|
@@ -39,19 +39,54 @@ const changeRecordSets = async (route53, action, records, hostedZoneId) => {
|
|
|
39
39
|
})),
|
|
40
40
|
},
|
|
41
41
|
};
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
42
|
+
try {
|
|
43
|
+
const { ChangeInfo } = await route53.send(new client_route_53_1.ChangeResourceRecordSetsCommand(changeRecordSetsInput));
|
|
44
|
+
const changeId = ChangeInfo?.Id;
|
|
45
|
+
const result = await (0, client_route_53_1.waitUntilResourceRecordSetsChanged)({ client: route53, maxWaitTime: 180 }, { Id: changeId });
|
|
46
|
+
if (result.state !== 'SUCCESS') {
|
|
47
|
+
throw new Error(`Validation records never changed for hosted zone ${hostedZoneId}: [${result.state}] ${result.reason ?? ''}`);
|
|
48
|
+
}
|
|
49
|
+
const operation = action === 'CREATE' || action === 'UPSERT' ? 'changed' : 'deleted';
|
|
50
|
+
const change = (0, utils_1.cleanChangeId)(changeId);
|
|
51
|
+
console.log(`Validation records succesfully ${operation} for hosted zone ${hostedZoneId} with change id ${change}`);
|
|
52
|
+
}
|
|
53
|
+
catch (error) {
|
|
54
|
+
if (action === 'DELETE' && error instanceof client_route_53_1.InvalidChangeBatch && error.message.includes('not found')) {
|
|
55
|
+
// there's a deletion race condition where some other certificate has already deleted the records
|
|
56
|
+
console.log(`All validation records have already been removed by some other certificate`);
|
|
57
|
+
}
|
|
58
|
+
else {
|
|
59
|
+
throw error;
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
};
|
|
63
|
+
const getRecordsForZoneNames = (records, zoneNames, result) => {
|
|
64
|
+
const [zoneName, ...restZoneNames] = zoneNames;
|
|
65
|
+
if (!zoneName) {
|
|
66
|
+
return result ?? {};
|
|
67
|
+
}
|
|
68
|
+
const matchingRecords = [];
|
|
69
|
+
const unmatchingRecords = [];
|
|
70
|
+
for (const record of records) {
|
|
71
|
+
const normalizedRecordName = (0, utils_1.cleanDomainName)(record.Name);
|
|
72
|
+
if (normalizedRecordName.endsWith(zoneName)) {
|
|
73
|
+
matchingRecords.push(record);
|
|
74
|
+
}
|
|
75
|
+
else {
|
|
76
|
+
unmatchingRecords.push(record);
|
|
77
|
+
}
|
|
46
78
|
}
|
|
47
|
-
return
|
|
79
|
+
return getRecordsForZoneNames(unmatchingRecords, restZoneNames, {
|
|
80
|
+
...(result ?? {}),
|
|
81
|
+
[zoneName]: matchingRecords,
|
|
82
|
+
});
|
|
48
83
|
};
|
|
49
84
|
const requestCertificate = async (acm, route53, requestId, properties) => {
|
|
50
|
-
const {
|
|
85
|
+
const { DomainName, AlternativeDomainNames, TransparencyLoggingEnabled } = properties;
|
|
51
86
|
console.log(`Requesting certificate for ${DomainName}`);
|
|
52
87
|
const requestCertificateInput = {
|
|
53
88
|
DomainName,
|
|
54
|
-
SubjectAlternativeNames:
|
|
89
|
+
SubjectAlternativeNames: AlternativeDomainNames,
|
|
55
90
|
IdempotencyToken: crypto.createHash('sha256').update(requestId).digest('hex').slice(0, 32),
|
|
56
91
|
ValidationMethod: 'DNS',
|
|
57
92
|
Options: {
|
|
@@ -69,10 +104,17 @@ const requestCertificate = async (acm, route53, requestId, properties) => {
|
|
|
69
104
|
const { Certificate } = await acm.send(new client_acm_1.DescribeCertificateCommand(describeCertificateInput));
|
|
70
105
|
return parseDomainValidationRecords(Certificate);
|
|
71
106
|
});
|
|
72
|
-
|
|
107
|
+
const hostedZones = Object.values(properties.ValidationHostedZones);
|
|
108
|
+
const hostedZoneIds = hostedZones.map((zone) => zone.HostedZoneId);
|
|
109
|
+
console.log(`Upserting ${validationRecords.length} validation record(s) into hosted zone(s) ${hostedZoneIds.join(', ')}:`);
|
|
73
110
|
validationRecords.forEach((record) => console.log(`${record.Name} ${record.Type} ${record.ResourceRecords?.map((rr) => rr.Value).join(',')}`));
|
|
74
|
-
const
|
|
75
|
-
|
|
111
|
+
const recordsForZoneNames = getRecordsForZoneNames(validationRecords, (0, utils_1.orderBySignificance)(Object.keys(properties.ValidationHostedZones)));
|
|
112
|
+
for (const hostedZone of hostedZones) {
|
|
113
|
+
const records = recordsForZoneNames[hostedZone.DomainName];
|
|
114
|
+
if (records.length > 0) {
|
|
115
|
+
await changeRecordSets(route53(hostedZone.ValidationRoleArn, hostedZone.ValidationExternalId), 'UPSERT', records, hostedZone.HostedZoneId);
|
|
116
|
+
}
|
|
117
|
+
}
|
|
76
118
|
console.log(`Waiting for certificate ${CertificateArn} to validate`);
|
|
77
119
|
const result = await (0, client_acm_1.waitUntilCertificateValidated)({ client: acm, maxWaitTime: 300 }, { CertificateArn });
|
|
78
120
|
if (result.state !== 'SUCCESS') {
|
|
@@ -81,7 +123,7 @@ const requestCertificate = async (acm, route53, requestId, properties) => {
|
|
|
81
123
|
console.log(`Certificate ${CertificateArn} successfully validated`);
|
|
82
124
|
return CertificateArn;
|
|
83
125
|
};
|
|
84
|
-
const deleteCertificate = async (acm, route53, certificateArn,
|
|
126
|
+
const deleteCertificate = async (acm, route53, certificateArn, properties) => {
|
|
85
127
|
console.log(`Waiting for certificate ${certificateArn} usage to drain before deletion`);
|
|
86
128
|
const waitUsageMaxSeconds = 600;
|
|
87
129
|
const waitUsageTimeoutError = `Certificate was still in use after ${waitUsageMaxSeconds} seconds`;
|
|
@@ -98,19 +140,15 @@ const deleteCertificate = async (acm, route53, certificateArn, hostedZoneId, cle
|
|
|
98
140
|
});
|
|
99
141
|
console.log('Certificate is unused and will be deleted');
|
|
100
142
|
const validationRecords = parseDomainValidationRecords(certificate);
|
|
101
|
-
if (validationRecords &&
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
if (
|
|
109
|
-
|
|
110
|
-
console.log(`All validation records have already been removed by some other certificate`);
|
|
111
|
-
}
|
|
112
|
-
else {
|
|
113
|
-
throw error;
|
|
143
|
+
if (validationRecords && (0, utils_1.stringToBoolean)(properties.CleanupValidationRecords)) {
|
|
144
|
+
const hostedZones = Object.values(properties.ValidationHostedZones);
|
|
145
|
+
const hostedZoneIds = hostedZones.map((zone) => zone.HostedZoneId);
|
|
146
|
+
console.log(`Deleting ${validationRecords.length} validation record(s) from hosted zone(s) ${hostedZoneIds.join(', ')}`);
|
|
147
|
+
const recordsForZoneNames = getRecordsForZoneNames(validationRecords, (0, utils_1.orderBySignificance)(Object.keys(properties.ValidationHostedZones)));
|
|
148
|
+
for (const hostedZone of hostedZones) {
|
|
149
|
+
const records = recordsForZoneNames[hostedZone.DomainName];
|
|
150
|
+
if (records.length > 0) {
|
|
151
|
+
await changeRecordSets(route53(hostedZone.ValidationRoleArn, hostedZone.ValidationExternalId), 'DELETE', records, hostedZone.HostedZoneId);
|
|
114
152
|
}
|
|
115
153
|
}
|
|
116
154
|
}
|
|
@@ -132,13 +170,15 @@ const addTags = async (acm, certificateArn, tags) => {
|
|
|
132
170
|
console.log(`All tags successfully added to certificate ${certificateArn}`);
|
|
133
171
|
};
|
|
134
172
|
const shouldRequestNew = (oldProperties, newProperties) => {
|
|
135
|
-
|
|
173
|
+
const oldHostedZoneIds = Object.values(oldProperties.ValidationHostedZones ?? {}).map((zone) => zone.HostedZoneId);
|
|
174
|
+
const newHostedZoneIds = Object.values(newProperties.ValidationHostedZones ?? {}).map((zone) => zone.HostedZoneId);
|
|
175
|
+
if (!(0, utils_1.containsSame)(oldHostedZoneIds, newHostedZoneIds))
|
|
136
176
|
return true;
|
|
137
177
|
if (oldProperties.DomainName !== newProperties.DomainName)
|
|
138
178
|
return true;
|
|
139
|
-
if (oldProperties.
|
|
179
|
+
if (!(0, utils_1.containsSame)(oldProperties.AlternativeDomainNames ?? [], newProperties.AlternativeDomainNames ?? []))
|
|
140
180
|
return true;
|
|
141
|
-
if (
|
|
181
|
+
if (oldProperties.CertificateRegion !== newProperties.CertificateRegion)
|
|
142
182
|
return true;
|
|
143
183
|
if (oldProperties.CleanupValidationRecords !== newProperties.CleanupValidationRecords)
|
|
144
184
|
return true;
|
|
@@ -171,10 +211,12 @@ const assumeRole = (roleArn, externalId) => {
|
|
|
171
211
|
const handler = async (event) => {
|
|
172
212
|
const properties = parseProperties(event.ResourceProperties);
|
|
173
213
|
const acm = new client_acm_1.ACMClient({ region: properties.CertificateRegion, retryMode: 'adaptive' });
|
|
174
|
-
const route53 =
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
214
|
+
const route53 = (roleArn, externalId) => {
|
|
215
|
+
return new client_route_53_1.Route53Client({
|
|
216
|
+
retryMode: 'adaptive',
|
|
217
|
+
credentials: assumeRole(roleArn, externalId),
|
|
218
|
+
});
|
|
219
|
+
};
|
|
178
220
|
switch (event.RequestType) {
|
|
179
221
|
case 'Create': {
|
|
180
222
|
console.log(`Requesting new certificate:\n${(0, utils_1.objectToString)(properties)}`);
|
|
@@ -209,7 +251,7 @@ const handler = async (event) => {
|
|
|
209
251
|
const certificateArn = event.PhysicalResourceId;
|
|
210
252
|
if (properties.RemovalPolicy === 'destroy') {
|
|
211
253
|
console.log(`Deleting old certificate as per removal policy:\n${(0, utils_1.objectToString)(properties)}`);
|
|
212
|
-
await deleteCertificate(acm, route53, certificateArn, properties
|
|
254
|
+
await deleteCertificate(acm, route53, certificateArn, properties);
|
|
213
255
|
}
|
|
214
256
|
return {
|
|
215
257
|
PhysicalResourceId: certificateArn,
|
|
@@ -222,4 +264,4 @@ const handler = async (event) => {
|
|
|
222
264
|
throw new Error(`Invalid request type`);
|
|
223
265
|
};
|
|
224
266
|
exports.handler = handler;
|
|
225
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"certificate-requestor.lambda.js","sourceRoot":"","sources":["../src/certificate-requestor.lambda.ts"],"names":[],"mappings":";;;AAAA,iCAAgC;AAChC,oDAY4B;AAC5B,8DAQiC;AACjC,oDAA0F;AAG1F,mCAA+E;AAe/E,MAAM,eAAe,GAAG,CAAC,UAA+B,EAAc,EAAE;IACtE,kDAAkD;IAClD,OAAO,UAAmC,CAAA;AAC5C,CAAC,CAAA;AAED,MAAM,4BAA4B,GAAG,CAAC,WAA8B,EAA8B,EAAE;IAClG,MAAM,OAAO,GAAG,WAAW,CAAC,uBAAuB,IAAI,EAAE,CAAA;IACzD,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC,IAAI,OAAO,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,cAAc,EAAE,IAAI,CAAC,EAAE;QAC1E,MAAM,aAAa,GAAG,CAAC,GAAG,IAAI,GAAG,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,cAAc,EAAE,IAAK,EAAE,GAAG,CAAC,cAAe,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAA;QACnH,OAAO,aAAa,CAAC,GAAG,CAAC,CAAC,MAAM,EAAE,EAAE;YAClC,OAAO;gBACL,IAAI,EAAE,MAAM,CAAC,IAAI;gBACjB,IAAI,EAAE,MAAM,CAAC,IAAI;gBACjB,GAAG,EAAE,EAAE;gBACP,eAAe,EAAE;oBACf;wBACE,KAAK,EAAE,MAAM,CAAC,KAAK;qBACpB;iBACF;aACF,CAAA;QACH,CAAC,CAAC,CAAA;KACH;IACD,OAAO,IAAI,CAAA;AACb,CAAC,CAAA;AAED,MAAM,gBAAgB,GAAG,KAAK,EAC5B,OAAsB,EACtB,MAAoB,EACpB,OAA4B,EAC5B,YAAoB,EACH,EAAE;IACnB,MAAM,qBAAqB,GAAyC;QAClE,YAAY,EAAE,YAAY;QAC1B,WAAW,EAAE;YACX,OAAO,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;gBAChC,MAAM,EAAE,MAAM;gBACd,iBAAiB,EAAE,MAAM;aAC1B,CAAC,CAAC;SACJ;KACF,CAAA;IACD,MAAM,EAAE,UAAU,EAAE,GAAG,MAAM,OAAO,CAAC,IAAI,CAAC,IAAI,iDAA+B,CAAC,qBAAqB,CAAC,CAAC,CAAA;IACrG,MAAM,MAAM,GAAG,MAAM,IAAA,oDAAkC,EAAC,EAAE,MAAM,EAAE,OAAO,EAAE,WAAW,EAAE,GAAG,EAAE,EAAE,EAAE,EAAE,EAAE,UAAU,EAAE,EAAE,EAAE,CAAC,CAAA;IACtH,IAAI,MAAM,CAAC,KAAK,KAAK,SAAS,EAAE;QAC9B,MAAM,IAAI,KAAK,CACb,6CAA6C,YAAY,MAAM,MAAM,CAAC,KAAK,KAAK,MAAM,CAAC,MAAM,IAAI,EAAE,EAAE,CACtG,CAAA;KACF;IACD,OAAO,UAAU,EAAE,EAAG,CAAA;AACxB,CAAC,CAAA;AAED,MAAM,kBAAkB,GAAG,KAAK,EAC9B,GAAc,EACd,OAAsB,EACtB,SAAiB,EACjB,UAAsB,EACL,EAAE;IACnB,MAAM,EAAE,YAAY,EAAE,UAAU,EAAE,uBAAuB,EAAE,0BAA0B,EAAE,GAAG,UAAU,CAAA;IAEpG,OAAO,CAAC,GAAG,CAAC,8BAA8B,UAAU,EAAE,CAAC,CAAA;IAEvD,MAAM,uBAAuB,GAAmC;QAC9D,UAAU;QACV,uBAAuB,EAAE,uBAAuB;QAChD,gBAAgB,EAAE,MAAM,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC;QAC1F,gBAAgB,EAAE,KAAK;QACvB,OAAO,EAAE;YACP,wCAAwC,EAAE,0BAA0B,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,UAAU;SAC9F;KACF,CAAA;IACD,MAAM,EAAE,cAAc,EAAE,GAAG,MAAM,GAAG,CAAC,IAAI,CAAC,IAAI,sCAAyB,CAAC,uBAAuB,CAAC,CAAC,CAAA;IAEjG,OAAO,CAAC,GAAG,CAAC,eAAe,cAAc,YAAY,CAAC,CAAA;IAEtD,MAAM,oBAAoB,GAAG,GAAG,CAAA;IAChC,MAAM,sBAAsB,GAAG,+CAA+C,oBAAoB,UAAU,CAAA;IAC5G,MAAM,iBAAiB,GAAG,MAAM,IAAA,cAAM,EAAC,oBAAoB,EAAE,sBAAsB,EAAE,KAAK,IAAI,EAAE;QAC9F,MAAM,wBAAwB,GAAoC;YAChE,cAAc;SACf,CAAA;QACD,MAAM,EAAE,WAAW,EAAE,GAAG,MAAM,GAAG,CAAC,IAAI,CAAC,IAAI,uCAA0B,CAAC,wBAAwB,CAAC,CAAC,CAAA;QAChG,OAAO,4BAA4B,CAAC,WAAY,CAAC,CAAA;IACnD,CAAC,CAAC,CAAA;IAEF,OAAO,CAAC,GAAG,CAAC,aAAa,iBAAiB,CAAC,MAAM,0CAA0C,YAAY,GAAG,CAAC,CAAA;IAC3G,iBAAiB,CAAC,OAAO,CAAC,CAAC,MAAM,EAAE,EAAE,CACnC,OAAO,CAAC,GAAG,CAAC,GAAG,MAAM,CAAC,IAAI,IAAI,MAAM,CAAC,IAAI,IAAI,MAAM,CAAC,eAAe,EAAE,GAAG,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,CACxG,CAAA;IACD,MAAM,QAAQ,GAAG,MAAM,gBAAgB,CAAC,OAAO,EAAE,QAAQ,EAAE,iBAAiB,EAAE,YAAY,CAAC,CAAA;IAC3F,OAAO,CAAC,GAAG,CAAC,4DAA4D,QAAQ,CAAC,OAAO,CAAC,UAAU,EAAE,EAAE,CAAC,EAAE,CAAC,CAAA;IAE3G,OAAO,CAAC,GAAG,CAAC,2BAA2B,cAAc,cAAc,CAAC,CAAA;IACpE,MAAM,MAAM,GAAG,MAAM,IAAA,0CAA6B,EAAC,EAAE,MAAM,EAAE,GAAG,EAAE,WAAW,EAAE,GAAG,EAAE,EAAE,EAAE,cAAc,EAAE,CAAC,CAAA;IACzG,IAAI,MAAM,CAAC,KAAK,KAAK,SAAS,EAAE;QAC9B,MAAM,IAAI,KAAK,CAAC,sBAAsB,cAAc,kBAAkB,MAAM,CAAC,KAAK,KAAK,MAAM,CAAC,MAAM,IAAI,EAAE,EAAE,CAAC,CAAA;KAC9G;IACD,OAAO,CAAC,GAAG,CAAC,eAAe,cAAc,yBAAyB,CAAC,CAAA;IACnE,OAAO,cAAe,CAAA;AACxB,CAAC,CAAA;AAED,MAAM,iBAAiB,GAAG,KAAK,EAC7B,GAAc,EACd,OAAsB,EACtB,cAAsB,EACtB,YAAoB,EACpB,wBAAiC,EAClB,EAAE;IACjB,OAAO,CAAC,GAAG,CAAC,2BAA2B,cAAc,iCAAiC,CAAC,CAAA;IAEvF,MAAM,mBAAmB,GAAG,GAAG,CAAA;IAC/B,MAAM,qBAAqB,GAAG,sCAAsC,mBAAmB,UAAU,CAAA;IACjG,MAAM,WAAW,GAAG,MAAM,IAAA,cAAM,EAAC,mBAAmB,EAAE,qBAAqB,EAAE,KAAK,IAAI,EAAE;QACtF,MAAM,wBAAwB,GAAoC;YAChE,cAAc,EAAE,cAAc;SAC/B,CAAA;QACD,MAAM,EAAE,WAAW,EAAE,GAAG,MAAM,GAAG,CAAC,IAAI,CAAC,IAAI,uCAA0B,CAAC,wBAAwB,CAAC,CAAC,CAAA;QAChG,MAAM,OAAO,GAAG,WAAW,EAAE,OAAO,IAAI,EAAE,CAAA;QAC1C,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE;YACtB,OAAO,IAAI,CAAA;SACZ;QACD,OAAO,WAAY,CAAA;IACrB,CAAC,CAAC,CAAA;IACF,OAAO,CAAC,GAAG,CAAC,2CAA2C,CAAC,CAAA;IAExD,MAAM,iBAAiB,GAAG,4BAA4B,CAAC,WAAW,CAAC,CAAA;IACnE,IAAI,iBAAiB,IAAI,wBAAwB,EAAE;QACjD,OAAO,CAAC,GAAG,CAAC,YAAY,iBAAiB,CAAC,MAAM,0CAA0C,YAAY,EAAE,CAAC,CAAA;QACzG,IAAI;YACF,MAAM,QAAQ,GAAG,MAAM,gBAAgB,CAAC,OAAO,EAAE,QAAQ,EAAE,iBAAiB,EAAE,YAAY,CAAC,CAAA;YAC3F,OAAO,CAAC,GAAG,CAAC,6DAA6D,QAAQ,CAAC,OAAO,CAAC,UAAU,EAAE,EAAE,CAAC,EAAE,CAAC,CAAA;SAC7G;QAAC,OAAO,KAAK,EAAE;YACd,IAAI,KAAK,YAAY,oCAAkB,IAAI,KAAK,CAAC,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAC,EAAE;gBAC9E,iGAAiG;gBACjG,OAAO,CAAC,GAAG,CAAC,4EAA4E,CAAC,CAAA;aAC1F;iBAAM;gBACL,MAAM,KAAK,CAAA;aACZ;SACF;KACF;IAED,OAAO,CAAC,GAAG,CAAC,wBAAwB,cAAc,WAAW,CAAC,CAAA;IAC9D,MAAM,sBAAsB,GAAkC;QAC5D,cAAc,EAAE,cAAc;KAC/B,CAAA;IACD,MAAM,GAAG,CAAC,IAAI,CAAC,IAAI,qCAAwB,CAAC,sBAAsB,CAAC,CAAC,CAAA;IACpE,OAAO,CAAC,GAAG,CAAC,eAAe,cAAc,uBAAuB,CAAC,CAAA;AACnE,CAAC,CAAA;AAED,MAAM,OAAO,GAAG,KAAK,EAAE,GAAc,EAAE,cAAsB,EAAE,IAA4B,EAAE,EAAE;IAC7F,MAAM,OAAO,GAAG,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,KAAK,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,CAAA;IACxF,MAAM,YAAY,GAAqC;QACrD,cAAc,EAAE,cAAc;QAC9B,IAAI,EAAE,OAAO;KACd,CAAA;IAED,OAAO,CAAC,GAAG,CAAC,UAAU,OAAO,CAAC,MAAM,wBAAwB,cAAc,EAAE,CAAC,CAAA;IAC7E,MAAM,GAAG,CAAC,IAAI,CAAC,IAAI,wCAA2B,CAAC,YAAY,CAAC,CAAC,CAAA;IAC7D,OAAO,CAAC,GAAG,CAAC,8CAA8C,cAAc,EAAE,CAAC,CAAA;AAC7E,CAAC,CAAA;AAED,MAAM,gBAAgB,GAAG,CAAC,aAAyB,EAAE,aAAyB,EAAW,EAAE;IACzF,IAAI,aAAa,CAAC,YAAY,KAAK,aAAa,CAAC,YAAY;QAAE,OAAO,IAAI,CAAA;IAC1E,IAAI,aAAa,CAAC,UAAU,KAAK,aAAa,CAAC,UAAU;QAAE,OAAO,IAAI,CAAA;IACtE,IAAI,aAAa,CAAC,iBAAiB,KAAK,aAAa,CAAC,iBAAiB;QAAE,OAAO,IAAI,CAAA;IACpF,IAAI,CAAC,IAAA,oBAAY,EAAC,aAAa,CAAC,uBAAuB,IAAI,EAAE,EAAE,aAAa,CAAC,uBAAuB,IAAI,EAAE,CAAC;QACzG,OAAO,IAAI,CAAA;IACb,IAAI,aAAa,CAAC,wBAAwB,KAAK,aAAa,CAAC,wBAAwB;QAAE,OAAO,IAAI,CAAA;IAClG,IAAI,aAAa,CAAC,0BAA0B,KAAK,aAAa,CAAC,0BAA0B;QAAE,OAAO,IAAI,CAAA;IACtG,IAAI,aAAa,CAAC,aAAa,KAAK,aAAa,CAAC,aAAa;QAAE,OAAO,IAAI,CAAA;IAC5E,OAAO,KAAK,CAAA;AACd,CAAC,CAAA;AAED,MAAM,UAAU,GAAG,CACjB,OAA2B,EAC3B,UAA8B,EACe,EAAE;IAC/C,IAAI,CAAC,OAAO,EAAE;QACZ,OAAO,SAAS,CAAA;KACjB;IACD,OAAO,KAAK,IAAI,EAAE;QAChB,MAAM,GAAG,GAAG,IAAI,sBAAS,CAAC,EAAE,SAAS,EAAE,UAAU,EAAE,CAAC,CAAA;QACpD,MAAM,eAAe,GAA2B;YAC9C,OAAO,EAAE,OAAO;YAChB,eAAe,EAAE,sBAAsB;YACvC,UAAU,EAAE,UAAU;SACvB,CAAA;QACD,MAAM,EAAE,WAAW,EAAE,GAAG,MAAM,GAAG,CAAC,IAAI,CAAC,IAAI,8BAAiB,CAAC,eAAe,CAAC,CAAC,CAAA;QAC9E,OAAO;YACL,WAAW,EAAE,WAAW,EAAE,WAAY;YACtC,eAAe,EAAE,WAAW,EAAE,eAAgB;YAC9C,YAAY,EAAE,WAAW,EAAE,YAAa;YACxC,UAAU,EAAE,WAAW,EAAE,UAAU;SACpC,CAAA;IACH,CAAC,CAAA;AACH,CAAC,CAAA;AAEM,MAAM,OAAO,GAAG,KAAK,EAAE,KAAwC,EAAE,EAAE;IACxE,MAAM,UAAU,GAAG,eAAe,CAAC,KAAK,CAAC,kBAAkB,CAAC,CAAA;IAE5D,MAAM,GAAG,GAAG,IAAI,sBAAS,CAAC,EAAE,MAAM,EAAE,UAAU,CAAC,iBAAiB,EAAE,SAAS,EAAE,UAAU,EAAE,CAAC,CAAA;IAC1F,MAAM,OAAO,GAAG,IAAI,+BAAa,CAAC;QAChC,SAAS,EAAE,UAAU;QACrB,WAAW,EAAE,UAAU,CAAC,UAAU,CAAC,iBAAiB,EAAE,UAAU,CAAC,oBAAoB,CAAC;KACvF,CAAC,CAAA;IAEF,QAAQ,KAAK,CAAC,WAAW,EAAE;QACzB,KAAK,QAAQ,CAAC,CAAC;YACb,OAAO,CAAC,GAAG,CAAC,gCAAgC,IAAA,sBAAc,EAAC,UAAU,CAAC,EAAE,CAAC,CAAA;YACzE,MAAM,cAAc,GAAG,MAAM,kBAAkB,CAAC,GAAG,EAAE,OAAO,EAAE,KAAK,CAAC,SAAS,EAAE,UAAU,CAAC,CAAA;YAC1F,IAAI,UAAU,CAAC,IAAI,IAAI,MAAM,CAAC,OAAO,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,MAAM,GAAG,CAAC,EAAE;gBACjE,MAAM,OAAO,CAAC,GAAG,EAAE,cAAc,EAAE,UAAU,CAAC,IAAI,CAAC,CAAA;aACpD;YACD,OAAO;gBACL,kBAAkB,EAAE,cAAc;gBAClC,IAAI,EAAE;oBACJ,GAAG,EAAE,cAAc;iBACpB;aACF,CAAA;SACF;QACD,KAAK,QAAQ,CAAC,CAAC;YACb,IAAI,cAAc,GAAG,KAAK,CAAC,kBAAkB,CAAA;YAC7C,IAAI,gBAAgB,CAAC,eAAe,CAAC,KAAK,CAAC,qBAAqB,CAAC,EAAE,UAAU,CAAC,EAAE;gBAC9E,OAAO,CAAC,GAAG,CAAC,4DAA4D,IAAA,sBAAc,EAAC,UAAU,CAAC,EAAE,CAAC,CAAA;gBACrG,cAAc,GAAG,MAAM,kBAAkB,CAAC,GAAG,EAAE,OAAO,EAAE,KAAK,CAAC,SAAS,EAAE,UAAU,CAAC,CAAA;aACrF;YACD,IAAI,UAAU,CAAC,IAAI,IAAI,MAAM,CAAC,OAAO,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,MAAM,GAAG,CAAC,EAAE;gBACjE,MAAM,OAAO,CAAC,GAAG,EAAE,cAAc,EAAE,UAAU,CAAC,IAAI,CAAC,CAAA;aACpD;YACD,OAAO;gBACL,kBAAkB,EAAE,cAAc;gBAClC,IAAI,EAAE;oBACJ,GAAG,EAAE,cAAc;iBACpB;aACF,CAAA;SACF;QACD,KAAK,QAAQ,CAAC,CAAC;YACb,MAAM,cAAc,GAAG,KAAK,CAAC,kBAAkB,CAAA;YAC/C,IAAI,UAAU,CAAC,aAAa,KAAK,SAAS,EAAE;gBAC1C,OAAO,CAAC,GAAG,CAAC,oDAAoD,IAAA,sBAAc,EAAC,UAAU,CAAC,EAAE,CAAC,CAAA;gBAC7F,MAAM,iBAAiB,CACrB,GAAG,EACH,OAAO,EACP,cAAc,EACd,UAAU,CAAC,YAAY,EACvB,IAAA,uBAAe,EAAC,UAAU,CAAC,wBAAwB,CAAC,CACrD,CAAA;aACF;YACD,OAAO;gBACL,kBAAkB,EAAE,cAAc;gBAClC,IAAI,EAAE;oBACJ,GAAG,EAAE,cAAc;iBACpB;aACF,CAAA;SACF;KACF;IACD,MAAM,IAAI,KAAK,CAAC,sBAAsB,CAAC,CAAA;AACzC,CAAC,CAAA;AA5DY,QAAA,OAAO,WA4DnB","sourcesContent":["import * as crypto from 'crypto'\nimport {\n  ACMClient,\n  AddTagsToCertificateCommand,\n  AddTagsToCertificateCommandInput,\n  CertificateDetail,\n  DeleteCertificateCommand,\n  DeleteCertificateCommandInput,\n  DescribeCertificateCommand,\n  DescribeCertificateCommandInput,\n  RequestCertificateCommand,\n  RequestCertificateCommandInput,\n  waitUntilCertificateValidated,\n} from '@aws-sdk/client-acm'\nimport {\n  ChangeAction,\n  ChangeResourceRecordSetsCommand,\n  ChangeResourceRecordSetsCommandInput,\n  InvalidChangeBatch,\n  ResourceRecordSet,\n  Route53Client,\n  waitUntilResourceRecordSetsChanged,\n} from '@aws-sdk/client-route-53'\nimport { AssumeRoleCommand, AssumeRoleCommandInput, STSClient } from '@aws-sdk/client-sts'\nimport type { AwsCredentialIdentity, Provider } from '@aws-sdk/types'\nimport type { CloudFormationCustomResourceEvent } from 'aws-lambda'\nimport { containsSame, objectToString, stringToBoolean, tryFor } from './utils'\n\nexport type Properties = {\n  HostedZoneId: string\n  DomainName: string\n  SubjectAlternativeNames?: string[]\n  CertificateRegion: string\n  ValidationRoleArn?: string\n  ValidationExternalId?: string\n  CleanupValidationRecords: string\n  TransparencyLoggingEnabled: string\n  Tags?: Record<string, string>\n  RemovalPolicy: string\n}\n\nconst parseProperties = (properties: Record<string, any>): Properties => {\n  // maybe should actually parse and not just assume\n  return properties as unknown as Properties\n}\n\nconst parseDomainValidationRecords = (certificate: CertificateDetail): ResourceRecordSet[] | null => {\n  const options = certificate.DomainValidationOptions ?? []\n  if (options.length > 0 && options.every((opt) => opt.ResourceRecord?.Name)) {\n    const uniqueRecords = [...new Map(options.map((opt) => [opt.ResourceRecord?.Name!, opt.ResourceRecord!])).values()]\n    return uniqueRecords.map((record) => {\n      return {\n        Name: record.Name,\n        Type: record.Type,\n        TTL: 30,\n        ResourceRecords: [\n          {\n            Value: record.Value,\n          },\n        ],\n      }\n    })\n  }\n  return null\n}\n\nconst changeRecordSets = async (\n  route53: Route53Client,\n  action: ChangeAction,\n  records: ResourceRecordSet[],\n  hostedZoneId: string\n): Promise<string> => {\n  const changeRecordSetsInput: ChangeResourceRecordSetsCommandInput = {\n    HostedZoneId: hostedZoneId,\n    ChangeBatch: {\n      Changes: records.map((record) => ({\n        Action: action,\n        ResourceRecordSet: record,\n      })),\n    },\n  }\n  const { ChangeInfo } = await route53.send(new ChangeResourceRecordSetsCommand(changeRecordSetsInput))\n  const result = await waitUntilResourceRecordSetsChanged({ client: route53, maxWaitTime: 180 }, { Id: ChangeInfo?.Id })\n  if (result.state !== 'SUCCESS') {\n    throw new Error(\n      `Record sets never changed for hosted zone ${hostedZoneId}: [${result.state}] ${result.reason ?? ''}`\n    )\n  }\n  return ChangeInfo?.Id!\n}\n\nconst requestCertificate = async (\n  acm: ACMClient,\n  route53: Route53Client,\n  requestId: string,\n  properties: Properties\n): Promise<string> => {\n  const { HostedZoneId, DomainName, SubjectAlternativeNames, TransparencyLoggingEnabled } = properties\n\n  console.log(`Requesting certificate for ${DomainName}`)\n\n  const requestCertificateInput: RequestCertificateCommandInput = {\n    DomainName,\n    SubjectAlternativeNames: SubjectAlternativeNames,\n    IdempotencyToken: crypto.createHash('sha256').update(requestId).digest('hex').slice(0, 32),\n    ValidationMethod: 'DNS',\n    Options: {\n      CertificateTransparencyLoggingPreference: TransparencyLoggingEnabled ? 'ENABLED' : 'DISABLED',\n    },\n  }\n  const { CertificateArn } = await acm.send(new RequestCertificateCommand(requestCertificateInput))\n\n  console.log(`Certificate ${CertificateArn} requested`)\n\n  const validationMaxSeconds = 180\n  const validationTimeoutError = `Domain validation options were not found in ${validationMaxSeconds} seconds`\n  const validationRecords = await tryFor(validationMaxSeconds, validationTimeoutError, async () => {\n    const describeCertificateInput: DescribeCertificateCommandInput = {\n      CertificateArn,\n    }\n    const { Certificate } = await acm.send(new DescribeCertificateCommand(describeCertificateInput))\n    return parseDomainValidationRecords(Certificate!)\n  })\n\n  console.log(`Upserting ${validationRecords.length} validation record(s) into hosted zone ${HostedZoneId}:`)\n  validationRecords.forEach((record) =>\n    console.log(`${record.Name} ${record.Type} ${record.ResourceRecords?.map((rr) => rr.Value).join(',')}`)\n  )\n  const changeId = await changeRecordSets(route53, 'UPSERT', validationRecords, HostedZoneId)\n  console.log(`All validation records changed succesfully for change id ${changeId.replace('/change/', '')}`)\n\n  console.log(`Waiting for certificate ${CertificateArn} to validate`)\n  const result = await waitUntilCertificateValidated({ client: acm, maxWaitTime: 300 }, { CertificateArn })\n  if (result.state !== 'SUCCESS') {\n    throw new Error(`Certificate failed ${CertificateArn} to validate: [${result.state}] ${result.reason ?? ''}`)\n  }\n  console.log(`Certificate ${CertificateArn} successfully validated`)\n  return CertificateArn!\n}\n\nconst deleteCertificate = async (\n  acm: ACMClient,\n  route53: Route53Client,\n  certificateArn: string,\n  hostedZoneId: string,\n  cleanupValidationRecords: boolean\n): Promise<void> => {\n  console.log(`Waiting for certificate ${certificateArn} usage to drain before deletion`)\n\n  const waitUsageMaxSeconds = 600\n  const waitUsageTimeoutError = `Certificate was still in use after ${waitUsageMaxSeconds} seconds`\n  const certificate = await tryFor(waitUsageMaxSeconds, waitUsageTimeoutError, async () => {\n    const describeCertificateInput: DescribeCertificateCommandInput = {\n      CertificateArn: certificateArn,\n    }\n    const { Certificate } = await acm.send(new DescribeCertificateCommand(describeCertificateInput))\n    const inUseBy = Certificate?.InUseBy ?? []\n    if (inUseBy.length > 0) {\n      return null\n    }\n    return Certificate!\n  })\n  console.log('Certificate is unused and will be deleted')\n\n  const validationRecords = parseDomainValidationRecords(certificate)\n  if (validationRecords && cleanupValidationRecords) {\n    console.log(`Deleting ${validationRecords.length} validation record(s) from hosted zone ${hostedZoneId}`)\n    try {\n      const changeId = await changeRecordSets(route53, 'DELETE', validationRecords, hostedZoneId)\n      console.log(`All validation records removed successfully for change id ${changeId.replace('/change/', '')}`)\n    } catch (error) {\n      if (error instanceof InvalidChangeBatch && error.message.includes('not found')) {\n        // there's a deletion race condition where some other certificate has already deleted the records\n        console.log(`All validation records have already been removed by some other certificate`)\n      } else {\n        throw error\n      }\n    }\n  }\n\n  console.log(`Deleting certificate ${certificateArn} from ACM`)\n  const deleteCertificateInput: DeleteCertificateCommandInput = {\n    CertificateArn: certificateArn,\n  }\n  await acm.send(new DeleteCertificateCommand(deleteCertificateInput))\n  console.log(`Certificate ${certificateArn} successfully deleted`)\n}\n\nconst addTags = async (acm: ACMClient, certificateArn: string, tags: Record<string, string>) => {\n  const tagList = Array.from(Object.entries(tags).map(([Key, Value]) => ({ Key, Value })))\n  const addTagsInput: AddTagsToCertificateCommandInput = {\n    CertificateArn: certificateArn,\n    Tags: tagList,\n  }\n\n  console.log(`Adding ${tagList.length} tags to certificate ${certificateArn}`)\n  await acm.send(new AddTagsToCertificateCommand(addTagsInput))\n  console.log(`All tags successfully added to certificate ${certificateArn}`)\n}\n\nconst shouldRequestNew = (oldProperties: Properties, newProperties: Properties): boolean => {\n  if (oldProperties.HostedZoneId !== newProperties.HostedZoneId) return true\n  if (oldProperties.DomainName !== newProperties.DomainName) return true\n  if (oldProperties.CertificateRegion !== newProperties.CertificateRegion) return true\n  if (!containsSame(oldProperties.SubjectAlternativeNames ?? [], newProperties.SubjectAlternativeNames ?? []))\n    return true\n  if (oldProperties.CleanupValidationRecords !== newProperties.CleanupValidationRecords) return true\n  if (oldProperties.TransparencyLoggingEnabled !== newProperties.TransparencyLoggingEnabled) return true\n  if (oldProperties.RemovalPolicy !== newProperties.RemovalPolicy) return true\n  return false\n}\n\nconst assumeRole = (\n  roleArn: string | undefined,\n  externalId: string | undefined\n): Provider<AwsCredentialIdentity> | undefined => {\n  if (!roleArn) {\n    return undefined\n  }\n  return async () => {\n    const sts = new STSClient({ retryMode: 'adaptive' })\n    const assumeRoleInput: AssumeRoleCommandInput = {\n      RoleArn: roleArn,\n      RoleSessionName: 'CertificateRequestor',\n      ExternalId: externalId,\n    }\n    const { Credentials } = await sts.send(new AssumeRoleCommand(assumeRoleInput))\n    return {\n      accessKeyId: Credentials?.AccessKeyId!,\n      secretAccessKey: Credentials?.SecretAccessKey!,\n      sessionToken: Credentials?.SessionToken!,\n      expiration: Credentials?.Expiration,\n    }\n  }\n}\n\nexport const handler = async (event: CloudFormationCustomResourceEvent) => {\n  const properties = parseProperties(event.ResourceProperties)\n\n  const acm = new ACMClient({ region: properties.CertificateRegion, retryMode: 'adaptive' })\n  const route53 = new Route53Client({\n    retryMode: 'adaptive',\n    credentials: assumeRole(properties.ValidationRoleArn, properties.ValidationExternalId),\n  })\n\n  switch (event.RequestType) {\n    case 'Create': {\n      console.log(`Requesting new certificate:\\n${objectToString(properties)}`)\n      const certificateArn = await requestCertificate(acm, route53, event.RequestId, properties)\n      if (properties.Tags && Object.entries(properties.Tags).length > 0) {\n        await addTags(acm, certificateArn, properties.Tags)\n      }\n      return {\n        PhysicalResourceId: certificateArn,\n        Data: {\n          Arn: certificateArn,\n        },\n      }\n    }\n    case 'Update': {\n      let certificateArn = event.PhysicalResourceId\n      if (shouldRequestNew(parseProperties(event.OldResourceProperties), properties)) {\n        console.log(`Requesting new certificate due to change of properties:\\n${objectToString(properties)}`)\n        certificateArn = await requestCertificate(acm, route53, event.RequestId, properties)\n      }\n      if (properties.Tags && Object.entries(properties.Tags).length > 0) {\n        await addTags(acm, certificateArn, properties.Tags)\n      }\n      return {\n        PhysicalResourceId: certificateArn,\n        Data: {\n          Arn: certificateArn,\n        },\n      }\n    }\n    case 'Delete': {\n      const certificateArn = event.PhysicalResourceId\n      if (properties.RemovalPolicy === 'destroy') {\n        console.log(`Deleting old certificate as per removal policy:\\n${objectToString(properties)}`)\n        await deleteCertificate(\n          acm,\n          route53,\n          certificateArn,\n          properties.HostedZoneId,\n          stringToBoolean(properties.CleanupValidationRecords)\n        )\n      }\n      return {\n        PhysicalResourceId: certificateArn,\n        Data: {\n          Arn: certificateArn,\n        },\n      }\n    }\n  }\n  throw new Error(`Invalid request type`)\n}\n"]}
|
|
267
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"certificate-requestor.lambda.js","sourceRoot":"","sources":["../src/certificate-requestor.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto'\nimport {\n  ACMClient,\n  AddTagsToCertificateCommand,\n  AddTagsToCertificateCommandInput,\n  CertificateDetail,\n  DeleteCertificateCommand,\n  DeleteCertificateCommandInput,\n  DescribeCertificateCommand,\n  DescribeCertificateCommandInput,\n  RequestCertificateCommand,\n  RequestCertificateCommandInput,\n  waitUntilCertificateValidated,\n} from '@aws-sdk/client-acm'\nimport {\n  ChangeAction,\n  ChangeResourceRecordSetsCommand,\n  ChangeResourceRecordSetsCommandInput,\n  InvalidChangeBatch,\n  ResourceRecordSet,\n  Route53Client,\n  waitUntilResourceRecordSetsChanged,\n} from '@aws-sdk/client-route-53'\nimport { AssumeRoleCommand, AssumeRoleCommandInput, STSClient } from '@aws-sdk/client-sts'\nimport type { AwsCredentialIdentity, Provider } from '@aws-sdk/types'\nimport type { CloudFormationCustomResourceEvent } from 'aws-lambda'\nimport {\n  cleanChangeId,\n  cleanDomainName,\n  containsSame,\n  objectToString,\n  orderBySignificance,\n  stringToBoolean,\n  tryFor,\n} from './utils'\n\nexport type ValidationHostedZoneProperties = {\n  DomainName: string\n  HostedZoneId: string\n  ValidationRoleArn?: string\n  ValidationExternalId?: string\n}\n\nexport type Properties = {\n  DomainName: string\n  AlternativeDomainNames?: string[]\n  ValidationHostedZones: Record<string, ValidationHostedZoneProperties>\n  CertificateRegion: string\n  CleanupValidationRecords: string\n  TransparencyLoggingEnabled: string\n  Tags?: Record<string, string>\n  RemovalPolicy: string\n}\n\ntype Route53Factory = (roleArn: string | undefined, externalId: string | undefined) => Route53Client\n\nconst parseProperties = (properties: Record<string, any>): Properties => {\n  // maybe should actually parse and not just assume\n  return properties as unknown as Properties\n}\n\nconst parseDomainValidationRecords = (certificate: CertificateDetail): ResourceRecordSet[] | null => {\n  const options = certificate.DomainValidationOptions ?? []\n  if (options.length > 0 && options.every((opt) => opt.ResourceRecord?.Name)) {\n    const uniqueRecords = [...new Map(options.map((opt) => [opt.ResourceRecord?.Name!, opt.ResourceRecord!])).values()]\n    return uniqueRecords.map((record) => {\n      return {\n        Name: record.Name,\n        Type: record.Type,\n        TTL: 30,\n        ResourceRecords: [\n          {\n            Value: record.Value,\n          },\n        ],\n      }\n    })\n  }\n  return null\n}\n\nconst changeRecordSets = async (\n  route53: Route53Client,\n  action: ChangeAction,\n  records: ResourceRecordSet[],\n  hostedZoneId: string\n): Promise<void> => {\n  const changeRecordSetsInput: ChangeResourceRecordSetsCommandInput = {\n    HostedZoneId: hostedZoneId,\n    ChangeBatch: {\n      Changes: records.map((record) => ({\n        Action: action,\n        ResourceRecordSet: record,\n      })),\n    },\n  }\n  try {\n    const { ChangeInfo } = await route53.send(new ChangeResourceRecordSetsCommand(changeRecordSetsInput))\n    const changeId = ChangeInfo?.Id!\n    const result = await waitUntilResourceRecordSetsChanged({ client: route53, maxWaitTime: 180 }, { Id: changeId })\n    if (result.state !== 'SUCCESS') {\n      throw new Error(\n        `Validation records never changed for hosted zone ${hostedZoneId}: [${result.state}] ${result.reason ?? ''}`\n      )\n    }\n    const operation = action === 'CREATE' || action === 'UPSERT' ? 'changed' : 'deleted'\n    const change = cleanChangeId(changeId)\n    console.log(`Validation records succesfully ${operation} for hosted zone ${hostedZoneId} with change id ${change}`)\n  } catch (error: unknown) {\n    if (action === 'DELETE' && error instanceof InvalidChangeBatch && error.message.includes('not found')) {\n      // there's a deletion race condition where some other certificate has already deleted the records\n      console.log(`All validation records have already been removed by some other certificate`)\n    } else {\n      throw error\n    }\n  }\n}\n\nconst getRecordsForZoneNames = (\n  records: ResourceRecordSet[],\n  zoneNames: string[],\n  result?: Record<string, ResourceRecordSet[]>\n): Record<string, ResourceRecordSet[]> => {\n  const [zoneName, ...restZoneNames] = zoneNames\n  if (!zoneName) {\n    return result ?? {}\n  }\n  const matchingRecords: ResourceRecordSet[] = []\n  const unmatchingRecords: ResourceRecordSet[] = []\n  for (const record of records) {\n    const normalizedRecordName = cleanDomainName(record.Name!)\n    if (normalizedRecordName.endsWith(zoneName)) {\n      matchingRecords.push(record)\n    } else {\n      unmatchingRecords.push(record)\n    }\n  }\n  return getRecordsForZoneNames(unmatchingRecords, restZoneNames, {\n    ...(result ?? {}),\n    [zoneName]: matchingRecords,\n  })\n}\n\nconst requestCertificate = async (\n  acm: ACMClient,\n  route53: Route53Factory,\n  requestId: string,\n  properties: Properties\n): Promise<string> => {\n  const { DomainName, AlternativeDomainNames, TransparencyLoggingEnabled } = properties\n\n  console.log(`Requesting certificate for ${DomainName}`)\n\n  const requestCertificateInput: RequestCertificateCommandInput = {\n    DomainName,\n    SubjectAlternativeNames: AlternativeDomainNames,\n    IdempotencyToken: crypto.createHash('sha256').update(requestId).digest('hex').slice(0, 32),\n    ValidationMethod: 'DNS',\n    Options: {\n      CertificateTransparencyLoggingPreference: TransparencyLoggingEnabled ? 'ENABLED' : 'DISABLED',\n    },\n  }\n  const { CertificateArn } = await acm.send(new RequestCertificateCommand(requestCertificateInput))\n\n  console.log(`Certificate ${CertificateArn} requested`)\n\n  const validationMaxSeconds = 180\n  const validationTimeoutError = `Domain validation options were not found in ${validationMaxSeconds} seconds`\n  const validationRecords = await tryFor(validationMaxSeconds, validationTimeoutError, async () => {\n    const describeCertificateInput: DescribeCertificateCommandInput = {\n      CertificateArn,\n    }\n    const { Certificate } = await acm.send(new DescribeCertificateCommand(describeCertificateInput))\n    return parseDomainValidationRecords(Certificate!)\n  })\n\n  const hostedZones = Object.values(properties.ValidationHostedZones)\n  const hostedZoneIds = hostedZones.map((zone) => zone.HostedZoneId)\n\n  console.log(\n    `Upserting ${validationRecords.length} validation record(s) into hosted zone(s) ${hostedZoneIds.join(', ')}:`\n  )\n  validationRecords.forEach((record) =>\n    console.log(`${record.Name} ${record.Type} ${record.ResourceRecords?.map((rr) => rr.Value).join(',')}`)\n  )\n\n  const recordsForZoneNames = getRecordsForZoneNames(\n    validationRecords,\n    orderBySignificance(Object.keys(properties.ValidationHostedZones))\n  )\n  for (const hostedZone of hostedZones) {\n    const records = recordsForZoneNames[hostedZone.DomainName]\n    if (records.length > 0) {\n      await changeRecordSets(\n        route53(hostedZone.ValidationRoleArn, hostedZone.ValidationExternalId),\n        'UPSERT',\n        records,\n        hostedZone.HostedZoneId\n      )\n    }\n  }\n\n  console.log(`Waiting for certificate ${CertificateArn} to validate`)\n  const result = await waitUntilCertificateValidated({ client: acm, maxWaitTime: 300 }, { CertificateArn })\n  if (result.state !== 'SUCCESS') {\n    throw new Error(`Certificate failed ${CertificateArn} to validate: [${result.state}] ${result.reason ?? ''}`)\n  }\n  console.log(`Certificate ${CertificateArn} successfully validated`)\n  return CertificateArn!\n}\n\nconst deleteCertificate = async (\n  acm: ACMClient,\n  route53: Route53Factory,\n  certificateArn: string,\n  properties: Properties\n): Promise<void> => {\n  console.log(`Waiting for certificate ${certificateArn} usage to drain before deletion`)\n\n  const waitUsageMaxSeconds = 600\n  const waitUsageTimeoutError = `Certificate was still in use after ${waitUsageMaxSeconds} seconds`\n  const certificate = await tryFor(waitUsageMaxSeconds, waitUsageTimeoutError, async () => {\n    const describeCertificateInput: DescribeCertificateCommandInput = {\n      CertificateArn: certificateArn,\n    }\n    const { Certificate } = await acm.send(new DescribeCertificateCommand(describeCertificateInput))\n    const inUseBy = Certificate?.InUseBy ?? []\n    if (inUseBy.length > 0) {\n      return null\n    }\n    return Certificate!\n  })\n  console.log('Certificate is unused and will be deleted')\n\n  const validationRecords = parseDomainValidationRecords(certificate)\n  if (validationRecords && stringToBoolean(properties.CleanupValidationRecords)) {\n    const hostedZones = Object.values(properties.ValidationHostedZones)\n    const hostedZoneIds = hostedZones.map((zone) => zone.HostedZoneId)\n\n    console.log(\n      `Deleting ${validationRecords.length} validation record(s) from hosted zone(s) ${hostedZoneIds.join(', ')}`\n    )\n\n    const recordsForZoneNames = getRecordsForZoneNames(\n      validationRecords,\n      orderBySignificance(Object.keys(properties.ValidationHostedZones))\n    )\n    for (const hostedZone of hostedZones) {\n      const records = recordsForZoneNames[hostedZone.DomainName]\n      if (records.length > 0) {\n        await changeRecordSets(\n          route53(hostedZone.ValidationRoleArn, hostedZone.ValidationExternalId),\n          'DELETE',\n          records,\n          hostedZone.HostedZoneId\n        )\n      }\n    }\n  }\n\n  console.log(`Deleting certificate ${certificateArn} from ACM`)\n  const deleteCertificateInput: DeleteCertificateCommandInput = {\n    CertificateArn: certificateArn,\n  }\n  await acm.send(new DeleteCertificateCommand(deleteCertificateInput))\n  console.log(`Certificate ${certificateArn} successfully deleted`)\n}\n\nconst addTags = async (acm: ACMClient, certificateArn: string, tags: Record<string, string>) => {\n  const tagList = Array.from(Object.entries(tags).map(([Key, Value]) => ({ Key, Value })))\n  const addTagsInput: AddTagsToCertificateCommandInput = {\n    CertificateArn: certificateArn,\n    Tags: tagList,\n  }\n\n  console.log(`Adding ${tagList.length} tags to certificate ${certificateArn}`)\n  await acm.send(new AddTagsToCertificateCommand(addTagsInput))\n  console.log(`All tags successfully added to certificate ${certificateArn}`)\n}\n\nconst shouldRequestNew = (oldProperties: Properties, newProperties: Properties): boolean => {\n  const oldHostedZoneIds = Object.values(oldProperties.ValidationHostedZones ?? {}).map((zone) => zone.HostedZoneId)\n  const newHostedZoneIds = Object.values(newProperties.ValidationHostedZones ?? {}).map((zone) => zone.HostedZoneId)\n  if (!containsSame(oldHostedZoneIds, newHostedZoneIds)) return true\n  if (oldProperties.DomainName !== newProperties.DomainName) return true\n  if (!containsSame(oldProperties.AlternativeDomainNames ?? [], newProperties.AlternativeDomainNames ?? [])) return true\n  if (oldProperties.CertificateRegion !== newProperties.CertificateRegion) return true\n  if (oldProperties.CleanupValidationRecords !== newProperties.CleanupValidationRecords) return true\n  if (oldProperties.TransparencyLoggingEnabled !== newProperties.TransparencyLoggingEnabled) return true\n  if (oldProperties.RemovalPolicy !== newProperties.RemovalPolicy) return true\n  return false\n}\n\nconst assumeRole = (\n  roleArn: string | undefined,\n  externalId: string | undefined\n): Provider<AwsCredentialIdentity> | undefined => {\n  if (!roleArn) {\n    return undefined\n  }\n  return async () => {\n    const sts = new STSClient({ retryMode: 'adaptive' })\n    const assumeRoleInput: AssumeRoleCommandInput = {\n      RoleArn: roleArn,\n      RoleSessionName: 'CertificateRequestor',\n      ExternalId: externalId,\n    }\n    const { Credentials } = await sts.send(new AssumeRoleCommand(assumeRoleInput))\n    return {\n      accessKeyId: Credentials?.AccessKeyId!,\n      secretAccessKey: Credentials?.SecretAccessKey!,\n      sessionToken: Credentials?.SessionToken!,\n      expiration: Credentials?.Expiration,\n    }\n  }\n}\n\nexport const handler = async (event: CloudFormationCustomResourceEvent) => {\n  const properties = parseProperties(event.ResourceProperties)\n\n  const acm = new ACMClient({ region: properties.CertificateRegion, retryMode: 'adaptive' })\n  const route53 = (roleArn: string | undefined, externalId: string | undefined): Route53Client => {\n    return new Route53Client({\n      retryMode: 'adaptive',\n      credentials: assumeRole(roleArn, externalId),\n    })\n  }\n\n  switch (event.RequestType) {\n    case 'Create': {\n      console.log(`Requesting new certificate:\\n${objectToString(properties)}`)\n      const certificateArn = await requestCertificate(acm, route53, event.RequestId, properties)\n      if (properties.Tags && Object.entries(properties.Tags).length > 0) {\n        await addTags(acm, certificateArn, properties.Tags)\n      }\n      return {\n        PhysicalResourceId: certificateArn,\n        Data: {\n          Arn: certificateArn,\n        },\n      }\n    }\n    case 'Update': {\n      let certificateArn = event.PhysicalResourceId\n      if (shouldRequestNew(parseProperties(event.OldResourceProperties), properties)) {\n        console.log(`Requesting new certificate due to change of properties:\\n${objectToString(properties)}`)\n        certificateArn = await requestCertificate(acm, route53, event.RequestId, properties)\n      }\n      if (properties.Tags && Object.entries(properties.Tags).length > 0) {\n        await addTags(acm, certificateArn, properties.Tags)\n      }\n      return {\n        PhysicalResourceId: certificateArn,\n        Data: {\n          Arn: certificateArn,\n        },\n      }\n    }\n    case 'Delete': {\n      const certificateArn = event.PhysicalResourceId\n      if (properties.RemovalPolicy === 'destroy') {\n        console.log(`Deleting old certificate as per removal policy:\\n${objectToString(properties)}`)\n        await deleteCertificate(acm, route53, certificateArn, properties)\n      }\n      return {\n        PhysicalResourceId: certificateArn,\n        Data: {\n          Arn: certificateArn,\n        },\n      }\n    }\n  }\n  throw new Error(`Invalid request type`)\n}\n"]}
|
|
@@ -3,6 +3,35 @@ import * as certificatemanager from 'aws-cdk-lib/aws-certificatemanager';
|
|
|
3
3
|
import * as iam from 'aws-cdk-lib/aws-iam';
|
|
4
4
|
import * as route53 from 'aws-cdk-lib/aws-route53';
|
|
5
5
|
import { Construct } from 'constructs';
|
|
6
|
+
export interface ValidationHostedZone {
|
|
7
|
+
/**
|
|
8
|
+
* Hosted zone to use for DNS validation. The zone name is matched to domain name to use the right
|
|
9
|
+
* hosted zone for validation.
|
|
10
|
+
*
|
|
11
|
+
* If the hosted zone is not managed by the CDK application, it needs to be provided via
|
|
12
|
+
* ``HostedZone.fromHostedZoneAttributes()``.
|
|
13
|
+
*/
|
|
14
|
+
readonly hostedZone: route53.IHostedZone;
|
|
15
|
+
/**
|
|
16
|
+
* The role that is assumed for DNS record changes for certificate validation.
|
|
17
|
+
*
|
|
18
|
+
* This role should exist in the same account as the hosted zone and include permissions to change the DNS records
|
|
19
|
+
* for the given ``hostedZone``. The ``customResourceRole`` or the default execution role is given permission to
|
|
20
|
+
* assume this role.
|
|
21
|
+
*
|
|
22
|
+
* @default - No separate role for DNS record changes. The given customResourceRole or the default role is used
|
|
23
|
+
* for DNS record changes.
|
|
24
|
+
*/
|
|
25
|
+
readonly validationRole?: iam.IRole;
|
|
26
|
+
/**
|
|
27
|
+
* External id for ``validationRole`` role assume verification.
|
|
28
|
+
*
|
|
29
|
+
* This should be used only when ``validationRole`` is given and the role expects an external id provided on assume.
|
|
30
|
+
*
|
|
31
|
+
* @default - No external id provided during assume.
|
|
32
|
+
*/
|
|
33
|
+
readonly validationExternalId?: string;
|
|
34
|
+
}
|
|
6
35
|
export interface DnsValidatedCertificateProps {
|
|
7
36
|
/**
|
|
8
37
|
* Fully-qualified domain name to request a certificate for.
|
|
@@ -11,12 +40,15 @@ export interface DnsValidatedCertificateProps {
|
|
|
11
40
|
*/
|
|
12
41
|
readonly domainName: string;
|
|
13
42
|
/**
|
|
14
|
-
*
|
|
43
|
+
* Fully-qualified alternative domain names to request a certificate for.
|
|
15
44
|
*
|
|
16
|
-
*
|
|
17
|
-
* ``HostedZone.fromHostedZoneAttributes()``.
|
|
45
|
+
* May contain wildcards, such as ``*.otherdomain.com``.
|
|
18
46
|
*/
|
|
19
|
-
readonly
|
|
47
|
+
readonly alternativeDomainNames?: string[];
|
|
48
|
+
/**
|
|
49
|
+
* List of hosted zones to use for validation. Hosted zones are mapped to domain names by the zone name.
|
|
50
|
+
*/
|
|
51
|
+
readonly validationHostedZones: ValidationHostedZone[];
|
|
20
52
|
/**
|
|
21
53
|
* AWS region where the certificate is deployed.
|
|
22
54
|
*
|
|
@@ -29,32 +61,13 @@ export interface DnsValidatedCertificateProps {
|
|
|
29
61
|
/**
|
|
30
62
|
* The role that is used for the custom resource Lambda execution.
|
|
31
63
|
*
|
|
32
|
-
* The role is given permissions to request certificates from ACM. If
|
|
33
|
-
* is also given permission to assume the ``validationRole``. Otherwise it is assumed that the hosted zone
|
|
34
|
-
* account and the execution role is given permissions to change DNS records for the given ``domainName``.
|
|
64
|
+
* The role is given permissions to request certificates from ACM. If there are any ``validationRole``s provided,
|
|
65
|
+
* this role is also given permission to assume the ``validationRole``. Otherwise it is assumed that the hosted zone
|
|
66
|
+
* is in same account and the execution role is given permissions to change DNS records for the given ``domainName``.
|
|
35
67
|
*
|
|
36
68
|
* @default - Lambda creates a default execution role.
|
|
37
69
|
*/
|
|
38
70
|
readonly customResourceRole?: iam.IRole;
|
|
39
|
-
/**
|
|
40
|
-
* The role that is assumed for DNS record changes for certificate validation.
|
|
41
|
-
*
|
|
42
|
-
* This role should exist in the same account as the hosted zone and include permissions to change the DNS records
|
|
43
|
-
* for the given ``hostedZone``. The ``customResourceRole`` or the default execution role is given permission to
|
|
44
|
-
* assume this role.
|
|
45
|
-
*
|
|
46
|
-
* @default - No separate role for DNS record changes. The given customResourceRole or the default role is used
|
|
47
|
-
* for DNS record changes.
|
|
48
|
-
*/
|
|
49
|
-
readonly validationRole?: iam.IRole;
|
|
50
|
-
/**
|
|
51
|
-
* External id for ``validationRole`` role assume verification.
|
|
52
|
-
*
|
|
53
|
-
* This should be used only when ``validationRole`` is given and the role expects an external id provided on assume.
|
|
54
|
-
*
|
|
55
|
-
* @default - No external id provided during assume
|
|
56
|
-
*/
|
|
57
|
-
readonly validationExternalId?: string;
|
|
58
71
|
/**
|
|
59
72
|
* Enable or disable cleaning of validation DNS records from the hosted zone.
|
|
60
73
|
*
|
|
@@ -99,32 +112,58 @@ export interface DnsValidatedCertificateProps {
|
|
|
99
112
|
* Please note that this construct does not support alternative names yet as it would require domain to role mapping.
|
|
100
113
|
*
|
|
101
114
|
* @example
|
|
102
|
-
* //
|
|
115
|
+
* // ### Cross-region certificate validation
|
|
103
116
|
* // hosted zone managed by the CDK application
|
|
104
117
|
* const hostedZone: route53.IHostedZone = ...
|
|
105
118
|
* // no separate validation role is needed
|
|
106
119
|
* const certificate = new DnsValidatedCertificate(this, 'CrossRegionCertificate', {
|
|
107
|
-
* hostedZone: hostedZone,
|
|
108
120
|
* domainName: 'example.com', // must be compatible with the hosted zone
|
|
121
|
+
* validationHostedZones: [{ // hosted zone used with the execution role's permissions
|
|
122
|
+
* hostedZone: hostedZone
|
|
123
|
+
* }],
|
|
109
124
|
* certificateRegion: 'us-east-1' // used by for example CloudFront
|
|
110
125
|
* })
|
|
111
|
-
* //
|
|
126
|
+
* // ### Cross-account certificate validation
|
|
112
127
|
* // external hosted zone
|
|
113
128
|
* const hostedZone: route53.IHostedZone =
|
|
114
129
|
* route53.HostedZone.fromHostedZoneAttributes(this, 'HostedZone', {
|
|
115
130
|
* hostedZoneId: 'Z532DGDEDFS123456789',
|
|
116
131
|
* zoneName: 'example.com'
|
|
117
132
|
* })
|
|
118
|
-
* // validation role
|
|
133
|
+
* // validation role in the same account as the hosted zone
|
|
119
134
|
* const roleArn = 'arn:aws:iam::123456789:role/ChangeDnsRecordsRole'
|
|
120
135
|
* const externalId = 'domain-assume'
|
|
121
136
|
* const validationRole: iam.IRole =
|
|
122
137
|
* iam.Role.fromRoleArn(this, 'ValidationRole', roleArn)
|
|
123
138
|
* const certificate = new DnsValidatedCertificate(this, 'CrossAccountCertificate', {
|
|
124
|
-
* hostedZone: hostedZone,
|
|
125
139
|
* domainName: 'example.com',
|
|
126
|
-
*
|
|
127
|
-
|
|
140
|
+
* validationHostedZones: [{
|
|
141
|
+
* hostedZone: hostedZone,
|
|
142
|
+
* validationRole: validationRole,
|
|
143
|
+
* validationExternalId: externalId
|
|
144
|
+
* }]
|
|
145
|
+
* })
|
|
146
|
+
* // ### Cross-account alternative name validation
|
|
147
|
+
* // example.com is validated on same account against managed hosted zone
|
|
148
|
+
* // and secondary.com is validated against external hosted zone on other account
|
|
149
|
+
* const hostedZoneForMain: route53.IHostedZone = ...
|
|
150
|
+
* const hostedZoneForAlternative: route53.IHostedZone =
|
|
151
|
+
* route53.HostedZone.fromHostedZoneAttributes(this, 'SecondaryHostedZone', {
|
|
152
|
+
* hostedZoneId: 'Z532DGDEDFS123456789',
|
|
153
|
+
* zoneName: 'secondary.com'
|
|
154
|
+
* })
|
|
155
|
+
* const certificate = new DnsValidatedCertificate(this, 'CrossAccountCertificate', {
|
|
156
|
+
* domainName: 'example.com',
|
|
157
|
+
* alternativeDomainNames: ['secondary.com'],
|
|
158
|
+
* validationHostedZones: [{
|
|
159
|
+
* hostedZone: hostedZoneForMain
|
|
160
|
+
* },{
|
|
161
|
+
* hostedZone: hostedZoneForAlternative,
|
|
162
|
+
* validationRole: iam.Role.fromRoleArn(
|
|
163
|
+
* this, 'SecondaryValidationRole', 'arn:aws:iam::123456789:role/ChangeDnsRecordsRole'
|
|
164
|
+
* ),
|
|
165
|
+
* validationExternalId: 'domain-assume'
|
|
166
|
+
* }]
|
|
128
167
|
* })
|
|
129
168
|
*
|
|
130
169
|
* @resource Custom::DnsValidatedCertificate
|
|
@@ -135,12 +174,6 @@ export declare class DnsValidatedCertificate extends cdk.Resource implements cer
|
|
|
135
174
|
readonly certificateArn: string;
|
|
136
175
|
/** The region where the certificate is deployed to */
|
|
137
176
|
readonly certificateRegion: string;
|
|
138
|
-
/** The hosted zone identifier authoritative for the certificate */
|
|
139
|
-
readonly hostedZoneId: string;
|
|
140
|
-
/** The hosted zone name authoritative for the certificate */
|
|
141
|
-
readonly hostedZoneName: string;
|
|
142
|
-
/** The domain name included in the certificate */
|
|
143
|
-
readonly domainName: string;
|
|
144
177
|
/** The tag manager to set, remove and format tags for the certificate */
|
|
145
178
|
readonly tags: cdk.TagManager;
|
|
146
179
|
/** The removal policy for the certificate */
|
|
@@ -158,5 +191,5 @@ export declare class DnsValidatedCertificate extends cdk.Resource implements cer
|
|
|
158
191
|
private normalizeDomainName;
|
|
159
192
|
private normalizeHostedZoneId;
|
|
160
193
|
private wildcardDomainName;
|
|
161
|
-
private
|
|
194
|
+
private validateDomainsToHostedZones;
|
|
162
195
|
}
|