@trautonen/cdk-dns-validated-certificate 0.0.2
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/.gitattributes +25 -0
- package/.jsii +3894 -0
- package/.prettierignore +1 -0
- package/.prettierrc.json +7 -0
- package/.projenrc.ts +49 -0
- package/API.md +560 -0
- package/LICENSE +202 -0
- package/README.md +40 -0
- package/lib/dns-validated-certificate.d.ts +161 -0
- package/lib/dns-validated-certificate.js +179 -0
- package/lib/index.d.ts +1 -0
- package/lib/index.js +18 -0
- package/lib/lambda/handler.d.ts +19 -0
- package/lib/lambda/handler.js +243 -0
- package/package.json +140 -0
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import type { CloudFormationCustomResourceEvent } from 'aws-lambda';
|
|
2
|
+
export type Properties = {
|
|
3
|
+
HostedZoneId: string;
|
|
4
|
+
DomainName: string;
|
|
5
|
+
SubjectAlternativeNames?: string[];
|
|
6
|
+
CertificateRegion: string;
|
|
7
|
+
ValidationRoleArn?: string;
|
|
8
|
+
ValidationExternalId?: string;
|
|
9
|
+
CleanupValidationRecords: boolean;
|
|
10
|
+
TransparencyLoggingEnabled: boolean;
|
|
11
|
+
Tags?: Record<string, string>;
|
|
12
|
+
RemovalPolicy: string;
|
|
13
|
+
};
|
|
14
|
+
export declare const handler: (event: CloudFormationCustomResourceEvent) => Promise<{
|
|
15
|
+
PhysicalResourceId: string;
|
|
16
|
+
Data: {
|
|
17
|
+
Arn: string;
|
|
18
|
+
};
|
|
19
|
+
}>;
|
|
@@ -0,0 +1,243 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.handler = void 0;
|
|
4
|
+
const crypto = require("crypto");
|
|
5
|
+
const client_acm_1 = require("@aws-sdk/client-acm");
|
|
6
|
+
const client_route_53_1 = require("@aws-sdk/client-route-53");
|
|
7
|
+
const client_sts_1 = require("@aws-sdk/client-sts");
|
|
8
|
+
const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
|
|
9
|
+
const containsSame = (array1, array2) => {
|
|
10
|
+
if (array1.length !== array2.length)
|
|
11
|
+
return false;
|
|
12
|
+
return array1.every((v1) => array2.includes(v1));
|
|
13
|
+
};
|
|
14
|
+
const tryFor = async (maxSeconds, timeoutError, fn) => {
|
|
15
|
+
const startTime = Date.now();
|
|
16
|
+
// eslint-disable-next-line no-constant-condition
|
|
17
|
+
for (let i = 0; true; i++) {
|
|
18
|
+
if (Date.now() > startTime + maxSeconds * 1000) {
|
|
19
|
+
throw new Error(timeoutError);
|
|
20
|
+
}
|
|
21
|
+
const result = await fn();
|
|
22
|
+
if (result !== null) {
|
|
23
|
+
return result;
|
|
24
|
+
}
|
|
25
|
+
const base = Math.pow(2, i);
|
|
26
|
+
await sleep(Math.random() * base * 50 + base * 150);
|
|
27
|
+
}
|
|
28
|
+
};
|
|
29
|
+
const parseProperties = (properties) => {
|
|
30
|
+
// maybe should actually parse and not just assume
|
|
31
|
+
return properties;
|
|
32
|
+
};
|
|
33
|
+
const parseDomainValidationRecords = (certificate) => {
|
|
34
|
+
const options = certificate.DomainValidationOptions ?? [];
|
|
35
|
+
console.log('options: ', options);
|
|
36
|
+
if (options.length > 0 && options.every((opt) => opt.ResourceRecord?.Name)) {
|
|
37
|
+
const uniqueRecords = [...new Map(options.map((opt) => [opt.ResourceRecord?.Name, opt.ResourceRecord])).values()];
|
|
38
|
+
return uniqueRecords.map((record) => {
|
|
39
|
+
return {
|
|
40
|
+
Name: record.Name,
|
|
41
|
+
Type: record.Type,
|
|
42
|
+
TTL: 30,
|
|
43
|
+
ResourceRecords: [
|
|
44
|
+
{
|
|
45
|
+
Value: record.Value,
|
|
46
|
+
},
|
|
47
|
+
],
|
|
48
|
+
};
|
|
49
|
+
});
|
|
50
|
+
}
|
|
51
|
+
return null;
|
|
52
|
+
};
|
|
53
|
+
const changeRecordSets = async (route53, action, records, hostedZoneId) => {
|
|
54
|
+
const changeRecordSetsInput = {
|
|
55
|
+
HostedZoneId: hostedZoneId,
|
|
56
|
+
ChangeBatch: {
|
|
57
|
+
Changes: records.map((record) => ({
|
|
58
|
+
Action: action,
|
|
59
|
+
ResourceRecordSet: record,
|
|
60
|
+
})),
|
|
61
|
+
},
|
|
62
|
+
};
|
|
63
|
+
const { ChangeInfo } = await route53.send(new client_route_53_1.ChangeResourceRecordSetsCommand(changeRecordSetsInput));
|
|
64
|
+
const result = await (0, client_route_53_1.waitUntilResourceRecordSetsChanged)({ client: route53, maxWaitTime: 180 }, { Id: ChangeInfo?.Id });
|
|
65
|
+
if (result.state !== 'SUCCESS') {
|
|
66
|
+
throw new Error(`Record sets never changed for hosted zone ${hostedZoneId}: [${result.state}] ${result.reason ?? ''}`);
|
|
67
|
+
}
|
|
68
|
+
return ChangeInfo?.Id;
|
|
69
|
+
};
|
|
70
|
+
const requestCertificate = async (acm, route53, requestId, properties) => {
|
|
71
|
+
const { HostedZoneId, DomainName, SubjectAlternativeNames, TransparencyLoggingEnabled } = properties;
|
|
72
|
+
console.log(`Requesting certificate for ${DomainName}`);
|
|
73
|
+
const requestCertificateInput = {
|
|
74
|
+
DomainName,
|
|
75
|
+
SubjectAlternativeNames: SubjectAlternativeNames,
|
|
76
|
+
IdempotencyToken: crypto.createHash('sha256').update(requestId).digest('hex').slice(0, 32),
|
|
77
|
+
ValidationMethod: 'DNS',
|
|
78
|
+
Options: {
|
|
79
|
+
CertificateTransparencyLoggingPreference: TransparencyLoggingEnabled ? 'ENABLED' : 'DISABLED',
|
|
80
|
+
},
|
|
81
|
+
};
|
|
82
|
+
const { CertificateArn } = await acm.send(new client_acm_1.RequestCertificateCommand(requestCertificateInput));
|
|
83
|
+
console.log(`Certificate ${CertificateArn} requested`);
|
|
84
|
+
const validationMaxSeconds = 180;
|
|
85
|
+
const validationTimeoutError = `Domain validation options were not found in ${validationMaxSeconds} seconds`;
|
|
86
|
+
const validationRecords = await tryFor(validationMaxSeconds, validationTimeoutError, async () => {
|
|
87
|
+
const describeCertificateInput = {
|
|
88
|
+
CertificateArn,
|
|
89
|
+
};
|
|
90
|
+
const { Certificate } = await acm.send(new client_acm_1.DescribeCertificateCommand(describeCertificateInput));
|
|
91
|
+
return parseDomainValidationRecords(Certificate);
|
|
92
|
+
});
|
|
93
|
+
console.log(`Upserting ${validationRecords.length} validation record(s) into hosted zone ${HostedZoneId}:`);
|
|
94
|
+
validationRecords.forEach((record) => console.log(`${record.Name} ${record.Type} ${record.ResourceRecords?.map((rr) => rr.Value).join(',')}`));
|
|
95
|
+
const changeId = await changeRecordSets(route53, 'UPSERT', validationRecords, HostedZoneId);
|
|
96
|
+
console.log(`All validation records changed succesfully for change id ${changeId}`);
|
|
97
|
+
console.log(`Waiting for certificate ${CertificateArn} to validate`);
|
|
98
|
+
const result = await (0, client_acm_1.waitUntilCertificateValidated)({ client: acm, maxWaitTime: 300 }, { CertificateArn });
|
|
99
|
+
if (result.state !== 'SUCCESS') {
|
|
100
|
+
throw new Error(`Certificate failed ${CertificateArn} to validate: [${result.state}] ${result.reason ?? ''}`);
|
|
101
|
+
}
|
|
102
|
+
console.log(`Certificate ${CertificateArn} successfully validated`);
|
|
103
|
+
return CertificateArn;
|
|
104
|
+
};
|
|
105
|
+
const deleteCertificate = async (acm, route53, certificateArn, hostedZoneId, cleanupValidationRecords) => {
|
|
106
|
+
console.log(`Waiting for certificate ${certificateArn} usage to drain before deletion`);
|
|
107
|
+
const waitUsageMaxSeconds = 600;
|
|
108
|
+
const waitUsageTimeoutError = `Certificate was still in use after ${waitUsageMaxSeconds} seconds`;
|
|
109
|
+
const certificate = await tryFor(waitUsageMaxSeconds, waitUsageTimeoutError, async () => {
|
|
110
|
+
const describeCertificateInput = {
|
|
111
|
+
CertificateArn: certificateArn,
|
|
112
|
+
};
|
|
113
|
+
const { Certificate } = await acm.send(new client_acm_1.DescribeCertificateCommand(describeCertificateInput));
|
|
114
|
+
const inUseBy = Certificate?.InUseBy ?? [];
|
|
115
|
+
if (inUseBy.length > 0) {
|
|
116
|
+
return null;
|
|
117
|
+
}
|
|
118
|
+
return Certificate;
|
|
119
|
+
});
|
|
120
|
+
console.log('Certificate is unused and will be deleted');
|
|
121
|
+
const validationRecords = parseDomainValidationRecords(certificate);
|
|
122
|
+
if (validationRecords && cleanupValidationRecords) {
|
|
123
|
+
console.log(`Deleting ${validationRecords.length} validation record(s) from hosted zone ${hostedZoneId}`);
|
|
124
|
+
try {
|
|
125
|
+
const changeId = await changeRecordSets(route53, 'DELETE', validationRecords, hostedZoneId);
|
|
126
|
+
console.log(`All validation records removed successfully for change id ${changeId}`);
|
|
127
|
+
}
|
|
128
|
+
catch (error) {
|
|
129
|
+
if (error instanceof client_route_53_1.InvalidChangeBatch && error.message.includes('not found')) {
|
|
130
|
+
// there's a deletion race condition where some other certificate has already deleted the records
|
|
131
|
+
console.log(`All validation records have already been removed by some other certificate`);
|
|
132
|
+
}
|
|
133
|
+
else {
|
|
134
|
+
throw error;
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
console.log(`Deleting certificate ${certificateArn}`);
|
|
139
|
+
const deleteCertificateInput = {
|
|
140
|
+
CertificateArn: certificateArn,
|
|
141
|
+
};
|
|
142
|
+
await acm.send(new client_acm_1.DeleteCertificateCommand(deleteCertificateInput));
|
|
143
|
+
console.log(`Certificate ${certificateArn} successfully deleted`);
|
|
144
|
+
};
|
|
145
|
+
const addTags = async (acm, certificateArn, tags) => {
|
|
146
|
+
const tagList = Array.from(Object.entries(tags).map(([Key, Value]) => ({ Key, Value })));
|
|
147
|
+
const addTagsInput = {
|
|
148
|
+
CertificateArn: certificateArn,
|
|
149
|
+
Tags: tagList,
|
|
150
|
+
};
|
|
151
|
+
console.log(`Adding ${tagList.length} tags to certificate ${certificateArn}`);
|
|
152
|
+
await acm.send(new client_acm_1.AddTagsToCertificateCommand(addTagsInput));
|
|
153
|
+
console.log(`All tags successfully added to certificate ${certificateArn}`);
|
|
154
|
+
};
|
|
155
|
+
const shouldRequestNew = (oldProperties, newProperties) => {
|
|
156
|
+
if (oldProperties.HostedZoneId !== newProperties.HostedZoneId)
|
|
157
|
+
return true;
|
|
158
|
+
if (oldProperties.DomainName !== newProperties.DomainName)
|
|
159
|
+
return true;
|
|
160
|
+
if (oldProperties.CertificateRegion !== newProperties.CertificateRegion)
|
|
161
|
+
return true;
|
|
162
|
+
if (!containsSame(oldProperties.SubjectAlternativeNames ?? [], newProperties.SubjectAlternativeNames ?? []))
|
|
163
|
+
return true;
|
|
164
|
+
if (oldProperties.CleanupValidationRecords !== newProperties.CleanupValidationRecords)
|
|
165
|
+
return true;
|
|
166
|
+
if (oldProperties.TransparencyLoggingEnabled !== newProperties.TransparencyLoggingEnabled)
|
|
167
|
+
return true;
|
|
168
|
+
if (oldProperties.RemovalPolicy !== newProperties.RemovalPolicy)
|
|
169
|
+
return true;
|
|
170
|
+
return false;
|
|
171
|
+
};
|
|
172
|
+
const assumeRole = (roleArn, externalId) => {
|
|
173
|
+
if (!roleArn) {
|
|
174
|
+
return undefined;
|
|
175
|
+
}
|
|
176
|
+
return async () => {
|
|
177
|
+
const sts = new client_sts_1.STSClient({ retryMode: 'adaptive' });
|
|
178
|
+
const assumeRoleInput = {
|
|
179
|
+
RoleArn: roleArn,
|
|
180
|
+
RoleSessionName: 'CertificateRequestor',
|
|
181
|
+
ExternalId: externalId,
|
|
182
|
+
};
|
|
183
|
+
const { Credentials } = await sts.send(new client_sts_1.AssumeRoleCommand(assumeRoleInput));
|
|
184
|
+
return {
|
|
185
|
+
accessKeyId: Credentials?.AccessKeyId,
|
|
186
|
+
secretAccessKey: Credentials?.SecretAccessKey,
|
|
187
|
+
sessionToken: Credentials?.SessionToken,
|
|
188
|
+
expiration: Credentials?.Expiration,
|
|
189
|
+
};
|
|
190
|
+
};
|
|
191
|
+
};
|
|
192
|
+
const handler = async (event) => {
|
|
193
|
+
const properties = parseProperties(event.ResourceProperties);
|
|
194
|
+
const acm = new client_acm_1.ACMClient({ region: properties.CertificateRegion, retryMode: 'adaptive' });
|
|
195
|
+
const route53 = new client_route_53_1.Route53Client({
|
|
196
|
+
retryMode: 'adaptive',
|
|
197
|
+
credentials: assumeRole(properties.ValidationRoleArn, properties.ValidationExternalId),
|
|
198
|
+
});
|
|
199
|
+
switch (event.RequestType) {
|
|
200
|
+
case 'Create': {
|
|
201
|
+
const certificateArn = await requestCertificate(acm, route53, event.RequestId, properties);
|
|
202
|
+
if (properties.Tags && Object.entries(properties.Tags).length > 0) {
|
|
203
|
+
await addTags(acm, certificateArn, properties.Tags);
|
|
204
|
+
}
|
|
205
|
+
return {
|
|
206
|
+
PhysicalResourceId: certificateArn,
|
|
207
|
+
Data: {
|
|
208
|
+
Arn: certificateArn,
|
|
209
|
+
},
|
|
210
|
+
};
|
|
211
|
+
}
|
|
212
|
+
case 'Update': {
|
|
213
|
+
let certificateArn = event.PhysicalResourceId;
|
|
214
|
+
if (shouldRequestNew(parseProperties(event.OldResourceProperties), properties)) {
|
|
215
|
+
certificateArn = await requestCertificate(acm, route53, event.RequestId, properties);
|
|
216
|
+
}
|
|
217
|
+
if (properties.Tags && Object.entries(properties.Tags).length > 0) {
|
|
218
|
+
await addTags(acm, certificateArn, properties.Tags);
|
|
219
|
+
}
|
|
220
|
+
return {
|
|
221
|
+
PhysicalResourceId: certificateArn,
|
|
222
|
+
Data: {
|
|
223
|
+
Arn: certificateArn,
|
|
224
|
+
},
|
|
225
|
+
};
|
|
226
|
+
}
|
|
227
|
+
case 'Delete': {
|
|
228
|
+
const certificateArn = event.PhysicalResourceId;
|
|
229
|
+
if (properties.RemovalPolicy === 'destroy') {
|
|
230
|
+
await deleteCertificate(acm, route53, certificateArn, properties.HostedZoneId, properties.CleanupValidationRecords);
|
|
231
|
+
}
|
|
232
|
+
return {
|
|
233
|
+
PhysicalResourceId: certificateArn,
|
|
234
|
+
Data: {
|
|
235
|
+
Arn: certificateArn,
|
|
236
|
+
},
|
|
237
|
+
};
|
|
238
|
+
}
|
|
239
|
+
}
|
|
240
|
+
throw new Error(`Invalid request type`);
|
|
241
|
+
};
|
|
242
|
+
exports.handler = handler;
|
|
243
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"handler.js","sourceRoot":"","sources":["../../src/lambda/handler.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'\n\nexport type Properties = {\n  HostedZoneId: string\n  DomainName: string\n  SubjectAlternativeNames?: string[]\n  CertificateRegion: string\n  ValidationRoleArn?: string\n  ValidationExternalId?: string\n  CleanupValidationRecords: boolean\n  TransparencyLoggingEnabled: boolean\n  Tags?: Record<string, string>\n  RemovalPolicy: string\n}\n\nconst sleep = (ms: number) => new Promise((resolve) => setTimeout(resolve, ms))\n\nconst containsSame = <T>(array1: T[], array2: T[]): boolean => {\n  if (array1.length !== array2.length) return false\n  return array1.every((v1) => array2.includes(v1))\n}\n\nconst tryFor = async <T>(maxSeconds: number, timeoutError: string, fn: () => Promise<T | null>): Promise<T> => {\n  const startTime = Date.now()\n  // eslint-disable-next-line no-constant-condition\n  for (let i = 0; true; i++) {\n    if (Date.now() > startTime + maxSeconds * 1000) {\n      throw new Error(timeoutError)\n    }\n    const result = await fn()\n    if (result !== null) {\n      return result\n    }\n    const base = Math.pow(2, i)\n    await sleep(Math.random() * base * 50 + base * 150)\n  }\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  console.log('options: ', options)\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}`)\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}`)\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}`)\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      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        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        await deleteCertificate(\n          acm,\n          route53,\n          certificateArn,\n          properties.HostedZoneId,\n          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"]}
|
package/package.json
ADDED
|
@@ -0,0 +1,140 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@trautonen/cdk-dns-validated-certificate",
|
|
3
|
+
"description": "CDK certificate construct that supports cross-region and cross-account DNS validation",
|
|
4
|
+
"repository": {
|
|
5
|
+
"type": "git",
|
|
6
|
+
"url": "https://github.com/trautonen/cdk-dns-validated-certificate.git"
|
|
7
|
+
},
|
|
8
|
+
"scripts": {
|
|
9
|
+
"build": "npx projen build",
|
|
10
|
+
"bump": "npx projen bump",
|
|
11
|
+
"clobber": "npx projen clobber",
|
|
12
|
+
"compat": "npx projen compat",
|
|
13
|
+
"compile": "npx projen compile",
|
|
14
|
+
"default": "npx projen default",
|
|
15
|
+
"docgen": "npx projen docgen",
|
|
16
|
+
"eject": "npx projen eject",
|
|
17
|
+
"eslint": "npx projen eslint",
|
|
18
|
+
"package": "npx projen package",
|
|
19
|
+
"package-all": "npx projen package-all",
|
|
20
|
+
"package:js": "npx projen package:js",
|
|
21
|
+
"post-compile": "npx projen post-compile",
|
|
22
|
+
"post-upgrade": "npx projen post-upgrade",
|
|
23
|
+
"pre-compile": "npx projen pre-compile",
|
|
24
|
+
"release": "npx projen release",
|
|
25
|
+
"test": "npx projen test",
|
|
26
|
+
"test:watch": "npx projen test:watch",
|
|
27
|
+
"unbump": "npx projen unbump",
|
|
28
|
+
"upgrade": "npx projen upgrade",
|
|
29
|
+
"watch": "npx projen watch",
|
|
30
|
+
"projen": "npx projen"
|
|
31
|
+
},
|
|
32
|
+
"author": {
|
|
33
|
+
"name": "Tapio Rautonen",
|
|
34
|
+
"email": "trautonen@users.noreply.github.com",
|
|
35
|
+
"organization": false
|
|
36
|
+
},
|
|
37
|
+
"devDependencies": {
|
|
38
|
+
"@aws-sdk/client-acm": "^3.0.0",
|
|
39
|
+
"@aws-sdk/client-route-53": "^3.0.0",
|
|
40
|
+
"@aws-sdk/client-sts": "^3.0.0",
|
|
41
|
+
"@aws-sdk/types": "^3.0.0",
|
|
42
|
+
"@types/aws-lambda": "^8.10.117",
|
|
43
|
+
"@types/jest": "^29.5.2",
|
|
44
|
+
"@types/node": "^16",
|
|
45
|
+
"@typescript-eslint/eslint-plugin": "^5",
|
|
46
|
+
"@typescript-eslint/parser": "^5",
|
|
47
|
+
"aws-cdk-lib": "2.83.1",
|
|
48
|
+
"aws-lambda": "^1.0.7",
|
|
49
|
+
"constructs": "10.0.5",
|
|
50
|
+
"esbuild": "^0.18.2",
|
|
51
|
+
"eslint": "^8",
|
|
52
|
+
"eslint-config-prettier": "^8.8.0",
|
|
53
|
+
"eslint-import-resolver-node": "^0.3.7",
|
|
54
|
+
"eslint-import-resolver-typescript": "^3.5.5",
|
|
55
|
+
"eslint-plugin-import": "^2.27.5",
|
|
56
|
+
"eslint-plugin-prettier": "^4.2.1",
|
|
57
|
+
"jest": "^29.5.0",
|
|
58
|
+
"jest-junit": "^15",
|
|
59
|
+
"jsii": "~5.0.0",
|
|
60
|
+
"jsii-diff": "^1.83.0",
|
|
61
|
+
"jsii-docgen": "^8.0.44",
|
|
62
|
+
"jsii-pacmak": "^1.83.0",
|
|
63
|
+
"npm-check-updates": "^16",
|
|
64
|
+
"prettier": "^2.8.8",
|
|
65
|
+
"projen": "^0.71.87",
|
|
66
|
+
"standard-version": "^9",
|
|
67
|
+
"ts-jest": "^29.1.0",
|
|
68
|
+
"ts-node": "^10.9.1",
|
|
69
|
+
"typescript": "^5.1.3"
|
|
70
|
+
},
|
|
71
|
+
"peerDependencies": {
|
|
72
|
+
"aws-cdk-lib": "^2.83.1",
|
|
73
|
+
"constructs": "^10.0.5"
|
|
74
|
+
},
|
|
75
|
+
"keywords": [
|
|
76
|
+
"aws",
|
|
77
|
+
"cdk",
|
|
78
|
+
"certificate",
|
|
79
|
+
"cross-account",
|
|
80
|
+
"cross-region",
|
|
81
|
+
"dns"
|
|
82
|
+
],
|
|
83
|
+
"main": "lib/index.js",
|
|
84
|
+
"license": "Apache-2.0",
|
|
85
|
+
"publishConfig": {
|
|
86
|
+
"access": "public"
|
|
87
|
+
},
|
|
88
|
+
"version": "0.0.2",
|
|
89
|
+
"jest": {
|
|
90
|
+
"testMatch": [
|
|
91
|
+
"<rootDir>/src/**/__tests__/**/*.ts?(x)",
|
|
92
|
+
"<rootDir>/(test|src)/**/*(*.)@(spec|test).ts?(x)"
|
|
93
|
+
],
|
|
94
|
+
"clearMocks": true,
|
|
95
|
+
"collectCoverage": true,
|
|
96
|
+
"coverageReporters": [
|
|
97
|
+
"json",
|
|
98
|
+
"lcov",
|
|
99
|
+
"clover",
|
|
100
|
+
"cobertura",
|
|
101
|
+
"text"
|
|
102
|
+
],
|
|
103
|
+
"coverageDirectory": "coverage",
|
|
104
|
+
"coveragePathIgnorePatterns": [
|
|
105
|
+
"/node_modules/"
|
|
106
|
+
],
|
|
107
|
+
"testPathIgnorePatterns": [
|
|
108
|
+
"/node_modules/"
|
|
109
|
+
],
|
|
110
|
+
"watchPathIgnorePatterns": [
|
|
111
|
+
"/node_modules/"
|
|
112
|
+
],
|
|
113
|
+
"reporters": [
|
|
114
|
+
"default",
|
|
115
|
+
[
|
|
116
|
+
"jest-junit",
|
|
117
|
+
{
|
|
118
|
+
"outputDirectory": "test-reports"
|
|
119
|
+
}
|
|
120
|
+
]
|
|
121
|
+
],
|
|
122
|
+
"preset": "ts-jest",
|
|
123
|
+
"globals": {
|
|
124
|
+
"ts-jest": {
|
|
125
|
+
"tsconfig": "tsconfig.dev.json"
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
},
|
|
129
|
+
"types": "lib/index.d.ts",
|
|
130
|
+
"stability": "stable",
|
|
131
|
+
"jsii": {
|
|
132
|
+
"outdir": "dist",
|
|
133
|
+
"targets": {},
|
|
134
|
+
"tsc": {
|
|
135
|
+
"outDir": "lib",
|
|
136
|
+
"rootDir": "src"
|
|
137
|
+
}
|
|
138
|
+
},
|
|
139
|
+
"//": "~~ Generated by projen. To modify, edit .projenrc.ts and run \"npx projen\"."
|
|
140
|
+
}
|