@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.
@@ -39,19 +39,54 @@ const changeRecordSets = async (route53, action, records, hostedZoneId) => {
39
39
  })),
40
40
  },
41
41
  };
42
- const { ChangeInfo } = await route53.send(new client_route_53_1.ChangeResourceRecordSetsCommand(changeRecordSetsInput));
43
- const result = await (0, client_route_53_1.waitUntilResourceRecordSetsChanged)({ client: route53, maxWaitTime: 180 }, { Id: ChangeInfo?.Id });
44
- if (result.state !== 'SUCCESS') {
45
- throw new Error(`Record sets never changed for hosted zone ${hostedZoneId}: [${result.state}] ${result.reason ?? ''}`);
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 ChangeInfo?.Id;
79
+ return getRecordsForZoneNames(unmatchingRecords, restZoneNames, {
80
+ ...(result ?? {}),
81
+ [zoneName]: matchingRecords,
82
+ });
48
83
  };
49
84
  const requestCertificate = async (acm, route53, requestId, properties) => {
50
- const { HostedZoneId, DomainName, SubjectAlternativeNames, TransparencyLoggingEnabled } = properties;
85
+ const { DomainName, AlternativeDomainNames, TransparencyLoggingEnabled } = properties;
51
86
  console.log(`Requesting certificate for ${DomainName}`);
52
87
  const requestCertificateInput = {
53
88
  DomainName,
54
- SubjectAlternativeNames: 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
- console.log(`Upserting ${validationRecords.length} validation record(s) into hosted zone ${HostedZoneId}:`);
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 changeId = await changeRecordSets(route53, 'UPSERT', validationRecords, HostedZoneId);
75
- console.log(`All validation records changed succesfully for change id ${changeId.replace('/change/', '')}`);
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, hostedZoneId, cleanupValidationRecords) => {
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 && cleanupValidationRecords) {
102
- console.log(`Deleting ${validationRecords.length} validation record(s) from hosted zone ${hostedZoneId}`);
103
- try {
104
- const changeId = await changeRecordSets(route53, 'DELETE', validationRecords, hostedZoneId);
105
- console.log(`All validation records removed successfully for change id ${changeId.replace('/change/', '')}`);
106
- }
107
- catch (error) {
108
- if (error instanceof client_route_53_1.InvalidChangeBatch && error.message.includes('not found')) {
109
- // there's a deletion race condition where some other certificate has already deleted the records
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
- if (oldProperties.HostedZoneId !== newProperties.HostedZoneId)
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.CertificateRegion !== newProperties.CertificateRegion)
179
+ if (!(0, utils_1.containsSame)(oldProperties.AlternativeDomainNames ?? [], newProperties.AlternativeDomainNames ?? []))
140
180
  return true;
141
- if (!(0, utils_1.containsSame)(oldProperties.SubjectAlternativeNames ?? [], newProperties.SubjectAlternativeNames ?? []))
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 = new client_route_53_1.Route53Client({
175
- retryMode: 'adaptive',
176
- credentials: assumeRole(properties.ValidationRoleArn, properties.ValidationExternalId),
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.HostedZoneId, (0, utils_1.stringToBoolean)(properties.CleanupValidationRecords));
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":"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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 { 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
- * Hosted zone to use for DNS validation.
43
+ * Fully-qualified alternative domain names to request a certificate for.
15
44
  *
16
- * If the hosted zone is not managed by the CDK application, it needs to be provided via
17
- * ``HostedZone.fromHostedZoneAttributes()``.
45
+ * May contain wildcards, such as ``*.otherdomain.com``.
18
46
  */
19
- readonly hostedZone: route53.IHostedZone;
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 the ``validationRole`` is provided, this role
33
- * is also given permission to assume the ``validationRole``. Otherwise it is assumed that the hosted zone is in same
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
- * // # Cross-region certificate validation
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
- * // # Cross-account certificate validation
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 on the same account as the hosted zone
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
- * validationRole: validationRole,
127
- validationExternalId: externalId
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 validateDomainToHostedZone;
194
+ private validateDomainsToHostedZones;
162
195
  }