1 | ;
|
2 | var _a;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.VpcEndpointServiceDomainName = void 0;
|
5 | const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
|
6 | const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
7 | const crypto = require("crypto");
|
8 | const core_1 = require("@aws-cdk/core");
|
9 | const custom_resources_1 = require("@aws-cdk/custom-resources");
|
10 | const lib_1 = require("../lib");
|
11 | // v2 - keep this import as a separate section to reduce merge conflict when forward merging with the v2 branch.
|
12 | // eslint-disable-next-line
|
13 | const core_2 = require("@aws-cdk/core");
|
14 | /**
|
15 | * A Private DNS configuration for a VPC endpoint service.
|
16 | */
|
17 | class VpcEndpointServiceDomainName extends core_2.Construct {
|
18 | // The way this class works is by using three custom resources and a TxtRecord in conjunction
|
19 | // The first custom resource tells the VPC endpoint service to use the given DNS name
|
20 | // The VPC endpoint service will then say:
|
21 | // "ok, create a TXT record using these two values to prove you own the domain"
|
22 | // The second custom resource retrieves these two values from the service
|
23 | // The TxtRecord is created from these two values
|
24 | // The third custom resource tells the VPC Endpoint Service to verify the domain ownership
|
25 | constructor(scope, id, props) {
|
26 | super(scope, id);
|
27 | try {
|
28 | jsiiDeprecationWarnings._aws_cdk_aws_route53_VpcEndpointServiceDomainNameProps(props);
|
29 | }
|
30 | catch (error) {
|
31 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
32 | Error.captureStackTrace(error, VpcEndpointServiceDomainName);
|
33 | }
|
34 | throw error;
|
35 | }
|
36 | const serviceUniqueId = core_1.Names.nodeUniqueId(props.endpointService.node);
|
37 | const serviceId = props.endpointService.vpcEndpointServiceId;
|
38 | this.domainName = props.domainName;
|
39 | // Make sure a user doesn't accidentally add multiple domains
|
40 | this.validateProps(props);
|
41 | VpcEndpointServiceDomainName.endpointServicesMap[serviceUniqueId] = this.domainName;
|
42 | VpcEndpointServiceDomainName.endpointServices.push(props.endpointService);
|
43 | // Enable Private DNS on the endpoint service and retrieve the AWS-generated configuration
|
44 | const privateDnsConfiguration = this.getPrivateDnsConfiguration(serviceUniqueId, serviceId, this.domainName);
|
45 | // Tell AWS to verify that this account owns the domain attached to the service
|
46 | this.verifyPrivateDnsConfiguration(privateDnsConfiguration, props.publicHostedZone);
|
47 | // Finally, don't do any of the above before the endpoint service is created
|
48 | this.node.addDependency(props.endpointService);
|
49 | }
|
50 | validateProps(props) {
|
51 | const serviceUniqueId = core_1.Names.nodeUniqueId(props.endpointService.node);
|
52 | if (serviceUniqueId in VpcEndpointServiceDomainName.endpointServicesMap) {
|
53 | const endpoint = VpcEndpointServiceDomainName.endpointServicesMap[serviceUniqueId];
|
54 | throw new Error(`Cannot create a VpcEndpointServiceDomainName for service ${serviceUniqueId}, another VpcEndpointServiceDomainName (${endpoint}) is already associated with it`);
|
55 | }
|
56 | }
|
57 | /**
|
58 | * Sets up Custom Resources to make AWS calls to set up Private DNS on an endpoint service,
|
59 | * returning the values to use in a TxtRecord, which AWS uses to verify domain ownership.
|
60 | */
|
61 | getPrivateDnsConfiguration(serviceUniqueId, serviceId, privateDnsName) {
|
62 | // The custom resource which tells AWS to enable Private DNS on the given service, using the given domain name
|
63 | // AWS will generate a name/value pair for use in a TxtRecord, which is used to verify domain ownership.
|
64 | const enablePrivateDnsAction = {
|
65 | service: 'EC2',
|
66 | action: 'modifyVpcEndpointServiceConfiguration',
|
67 | parameters: {
|
68 | ServiceId: serviceId,
|
69 | PrivateDnsName: privateDnsName,
|
70 | },
|
71 | physicalResourceId: custom_resources_1.PhysicalResourceId.of(serviceUniqueId),
|
72 | };
|
73 | const removePrivateDnsAction = {
|
74 | service: 'EC2',
|
75 | action: 'modifyVpcEndpointServiceConfiguration',
|
76 | parameters: {
|
77 | ServiceId: serviceId,
|
78 | RemovePrivateDnsName: true,
|
79 | },
|
80 | };
|
81 | const enable = new custom_resources_1.AwsCustomResource(this, 'EnableDns', {
|
82 | onCreate: enablePrivateDnsAction,
|
83 | onUpdate: enablePrivateDnsAction,
|
84 | onDelete: removePrivateDnsAction,
|
85 | policy: custom_resources_1.AwsCustomResourcePolicy.fromSdkCalls({
|
86 | resources: [
|
87 | core_1.Fn.join(':', [
|
88 | 'arn',
|
89 | core_1.Stack.of(this).partition,
|
90 | 'ec2',
|
91 | core_1.Stack.of(this).region,
|
92 | core_1.Stack.of(this).account,
|
93 | core_1.Fn.join('/', [
|
94 | 'vpc-endpoint-service',
|
95 | serviceId,
|
96 | ]),
|
97 | ]),
|
98 | ],
|
99 | }),
|
100 | });
|
101 | // Look up the name/value pair if the domain changes, or the service changes,
|
102 | // which would cause the values to be different. If the unique ID changes,
|
103 | // the resource may be entirely recreated, so we will need to look it up again.
|
104 | const lookup = hashcode(core_1.Names.uniqueId(this) + serviceUniqueId + privateDnsName);
|
105 | // Create the custom resource to look up the name/value pair generated by AWS
|
106 | // after the previous API call
|
107 | const retrieveNameValuePairAction = {
|
108 | service: 'EC2',
|
109 | action: 'describeVpcEndpointServiceConfigurations',
|
110 | parameters: {
|
111 | ServiceIds: [serviceId],
|
112 | },
|
113 | physicalResourceId: custom_resources_1.PhysicalResourceId.of(lookup),
|
114 | };
|
115 | const getNames = new custom_resources_1.AwsCustomResource(this, 'GetNames', {
|
116 | onCreate: retrieveNameValuePairAction,
|
117 | onUpdate: retrieveNameValuePairAction,
|
118 | // describeVpcEndpointServiceConfigurations can't take an ARN for granular permissions
|
119 | policy: custom_resources_1.AwsCustomResourcePolicy.fromSdkCalls({
|
120 | resources: custom_resources_1.AwsCustomResourcePolicy.ANY_RESOURCE,
|
121 | }),
|
122 | });
|
123 | // We only want to call and get the name/value pair after we've told AWS to enable Private DNS
|
124 | // If we call before then, we'll get an empty pair of values.
|
125 | getNames.node.addDependency(enable);
|
126 | // Get the references to the name/value pair associated with the endpoint service
|
127 | const name = getNames.getResponseField('ServiceConfigurations.0.PrivateDnsNameConfiguration.Name');
|
128 | const value = getNames.getResponseField('ServiceConfigurations.0.PrivateDnsNameConfiguration.Value');
|
129 | return { name, value, serviceId };
|
130 | }
|
131 | /**
|
132 | * Creates a Route53 entry and a Custom Resource which explicitly tells AWS to verify ownership
|
133 | * of the domain name attached to an endpoint service.
|
134 | */
|
135 | verifyPrivateDnsConfiguration(config, publicHostedZone) {
|
136 | // Create the TXT record in the provided hosted zone
|
137 | const verificationRecord = new lib_1.TxtRecord(this, 'DnsVerificationRecord', {
|
138 | recordName: config.name,
|
139 | values: [config.value],
|
140 | zone: publicHostedZone,
|
141 | });
|
142 | // Tell the endpoint service to verify the domain ownership
|
143 | const startVerificationAction = {
|
144 | service: 'EC2',
|
145 | action: 'startVpcEndpointServicePrivateDnsVerification',
|
146 | parameters: {
|
147 | ServiceId: config.serviceId,
|
148 | },
|
149 | physicalResourceId: custom_resources_1.PhysicalResourceId.of(core_1.Fn.join(':', [config.name, config.value])),
|
150 | };
|
151 | const startVerification = new custom_resources_1.AwsCustomResource(this, 'StartVerification', {
|
152 | onCreate: startVerificationAction,
|
153 | onUpdate: startVerificationAction,
|
154 | policy: custom_resources_1.AwsCustomResourcePolicy.fromSdkCalls({
|
155 | resources: [
|
156 | core_1.Fn.join(':', [
|
157 | 'arn',
|
158 | core_1.Stack.of(this).partition,
|
159 | 'ec2',
|
160 | core_1.Stack.of(this).region,
|
161 | core_1.Stack.of(this).account,
|
162 | core_1.Fn.join('/', [
|
163 | 'vpc-endpoint-service',
|
164 | config.serviceId,
|
165 | ]),
|
166 | ]),
|
167 | ],
|
168 | }),
|
169 | });
|
170 | // Only verify after the record has been created
|
171 | startVerification.node.addDependency(verificationRecord);
|
172 | }
|
173 | }
|
174 | exports.VpcEndpointServiceDomainName = VpcEndpointServiceDomainName;
|
175 | _a = JSII_RTTI_SYMBOL_1;
|
176 | VpcEndpointServiceDomainName[_a] = { fqn: "@aws-cdk/aws-route53.VpcEndpointServiceDomainName", version: "1.204.0" };
|
177 | // Track all domain names created, so someone doesn't accidentally associate two domains with a single service
|
178 | VpcEndpointServiceDomainName.endpointServices = [];
|
179 | // Track all domain names created, so someone doesn't accidentally associate two domains with a single service
|
180 | VpcEndpointServiceDomainName.endpointServicesMap = {};
|
181 | /**
|
182 | * Hash a string
|
183 | */
|
184 | function hashcode(s) {
|
185 | const hash = crypto.createHash('md5');
|
186 | hash.update(s);
|
187 | return hash.digest('hex');
|
188 | }
|
189 | ;
|
190 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"vpc-endpoint-service-domain-name.js","sourceRoot":"","sources":["vpc-endpoint-service-domain-name.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\nimport { IVpcEndpointService } from '@aws-cdk/aws-ec2';\nimport { Fn, Names, Stack } from '@aws-cdk/core';\nimport { AwsCustomResource, AwsCustomResourcePolicy, PhysicalResourceId } from '@aws-cdk/custom-resources';\nimport { Construct } from 'constructs';\nimport { IPublicHostedZone, TxtRecord } from '../lib';\n\n// v2 - keep this import as a separate section to reduce merge conflict when forward merging with the v2 branch.\n// eslint-disable-next-line\nimport { Construct as CoreConstruct } from '@aws-cdk/core';\n\n/**\n * Properties to configure a VPC Endpoint Service domain name\n */\nexport interface VpcEndpointServiceDomainNameProps {\n\n  /**\n   * The VPC Endpoint Service to configure Private DNS for\n   */\n  readonly endpointService: IVpcEndpointService;\n\n  /**\n   * The domain name to use.\n   *\n   * This domain name must be owned by this account (registered through Route53),\n   * or delegated to this account. Domain ownership will be verified by AWS before\n   * private DNS can be used.\n   * @see https://docs.aws.amazon.com/vpc/latest/userguide/endpoint-services-dns-validation.html\n   */\n  readonly domainName: string;\n\n  /**\n   * The public hosted zone to use for the domain.\n   */\n  readonly publicHostedZone: IPublicHostedZone;\n}\n\n/**\n * A Private DNS configuration for a VPC endpoint service.\n */\nexport class VpcEndpointServiceDomainName extends CoreConstruct {\n\n  // Track all domain names created, so someone doesn't accidentally associate two domains with a single service\n  private static readonly endpointServices: IVpcEndpointService[] = [];\n\n  // Track all domain names created, so someone doesn't accidentally associate two domains with a single service\n  private static readonly endpointServicesMap: { [endpointService: string]: string} = {};\n\n  /**\n   * The domain name associated with the private DNS configuration\n   */\n  public domainName: string;\n\n  // The way this class works is by using three custom resources and a TxtRecord in conjunction\n  // The first custom resource tells the VPC endpoint service to use the given DNS name\n  // The VPC endpoint service will then say:\n  // \"ok, create a TXT record using these two values to prove you own the domain\"\n  // The second custom resource retrieves these two values from the service\n  // The TxtRecord is created from these two values\n  // The third custom resource tells the VPC Endpoint Service to verify the domain ownership\n  constructor(scope: Construct, id: string, props: VpcEndpointServiceDomainNameProps) {\n    super(scope, id);\n\n    const serviceUniqueId = Names.nodeUniqueId(props.endpointService.node);\n    const serviceId = props.endpointService.vpcEndpointServiceId;\n    this.domainName = props.domainName;\n\n    // Make sure a user doesn't accidentally add multiple domains\n    this.validateProps(props);\n\n    VpcEndpointServiceDomainName.endpointServicesMap[serviceUniqueId] = this.domainName;\n    VpcEndpointServiceDomainName.endpointServices.push(props.endpointService);\n\n    // Enable Private DNS on the endpoint service and retrieve the AWS-generated configuration\n    const privateDnsConfiguration = this.getPrivateDnsConfiguration(serviceUniqueId, serviceId, this.domainName);\n\n    // Tell AWS to verify that this account owns the domain attached to the service\n    this.verifyPrivateDnsConfiguration(privateDnsConfiguration, props.publicHostedZone);\n\n    // Finally, don't do any of the above before the endpoint service is created\n    this.node.addDependency(props.endpointService);\n  }\n\n  private validateProps(props: VpcEndpointServiceDomainNameProps): void {\n    const serviceUniqueId = Names.nodeUniqueId(props.endpointService.node);\n    if (serviceUniqueId in VpcEndpointServiceDomainName.endpointServicesMap) {\n      const endpoint = VpcEndpointServiceDomainName.endpointServicesMap[serviceUniqueId];\n      throw new Error(\n        `Cannot create a VpcEndpointServiceDomainName for service ${serviceUniqueId}, another VpcEndpointServiceDomainName (${endpoint}) is already associated with it`);\n    }\n  }\n\n  /**\n   * Sets up Custom Resources to make AWS calls to set up Private DNS on an endpoint service,\n   * returning the values to use in a TxtRecord, which AWS uses to verify domain ownership.\n   */\n  private getPrivateDnsConfiguration(serviceUniqueId: string, serviceId: string, privateDnsName: string): PrivateDnsConfiguration {\n\n    // The custom resource which tells AWS to enable Private DNS on the given service, using the given domain name\n    // AWS will generate a name/value pair for use in a TxtRecord, which is used to verify domain ownership.\n    const enablePrivateDnsAction = {\n      service: 'EC2',\n      action: 'modifyVpcEndpointServiceConfiguration',\n      parameters: {\n        ServiceId: serviceId,\n        PrivateDnsName: privateDnsName,\n      },\n      physicalResourceId: PhysicalResourceId.of(serviceUniqueId),\n    };\n    const removePrivateDnsAction = {\n      service: 'EC2',\n      action: 'modifyVpcEndpointServiceConfiguration',\n      parameters: {\n        ServiceId: serviceId,\n        RemovePrivateDnsName: true,\n      },\n    };\n    const enable = new AwsCustomResource(this, 'EnableDns', {\n      onCreate: enablePrivateDnsAction,\n      onUpdate: enablePrivateDnsAction,\n      onDelete: removePrivateDnsAction,\n      policy: AwsCustomResourcePolicy.fromSdkCalls({\n        resources: [\n          Fn.join(':', [\n            'arn',\n            Stack.of(this).partition,\n            'ec2',\n            Stack.of(this).region,\n            Stack.of(this).account,\n            Fn.join('/', [\n              'vpc-endpoint-service',\n              serviceId,\n            ]),\n          ]),\n        ],\n      }),\n    });\n\n    // Look up the name/value pair if the domain changes, or the service changes,\n    // which would cause the values to be different. If the unique ID changes,\n    // the resource may be entirely recreated, so we will need to look it up again.\n    const lookup = hashcode(Names.uniqueId(this) + serviceUniqueId + privateDnsName);\n\n    // Create the custom resource to look up the name/value pair generated by AWS\n    // after the previous API call\n    const retrieveNameValuePairAction = {\n      service: 'EC2',\n      action: 'describeVpcEndpointServiceConfigurations',\n      parameters: {\n        ServiceIds: [serviceId],\n      },\n      physicalResourceId: PhysicalResourceId.of(lookup),\n    };\n    const getNames = new AwsCustomResource(this, 'GetNames', {\n      onCreate: retrieveNameValuePairAction,\n      onUpdate: retrieveNameValuePairAction,\n      // describeVpcEndpointServiceConfigurations can't take an ARN for granular permissions\n      policy: AwsCustomResourcePolicy.fromSdkCalls({\n        resources: AwsCustomResourcePolicy.ANY_RESOURCE,\n      }),\n    });\n\n    // We only want to call and get the name/value pair after we've told AWS to enable Private DNS\n    // If we call before then, we'll get an empty pair of values.\n    getNames.node.addDependency(enable);\n\n    // Get the references to the name/value pair associated with the endpoint service\n    const name = getNames.getResponseField('ServiceConfigurations.0.PrivateDnsNameConfiguration.Name');\n    const value = getNames.getResponseField('ServiceConfigurations.0.PrivateDnsNameConfiguration.Value');\n\n    return { name, value, serviceId };\n  }\n\n  /**\n   * Creates a Route53 entry and a Custom Resource which explicitly tells AWS to verify ownership\n   * of the domain name attached to an endpoint service.\n   */\n  private verifyPrivateDnsConfiguration(config: PrivateDnsConfiguration, publicHostedZone: IPublicHostedZone) {\n    // Create the TXT record in the provided hosted zone\n    const verificationRecord = new TxtRecord(this, 'DnsVerificationRecord', {\n      recordName: config.name,\n      values: [config.value],\n      zone: publicHostedZone,\n    });\n\n    // Tell the endpoint service to verify the domain ownership\n    const startVerificationAction = {\n      service: 'EC2',\n      action: 'startVpcEndpointServicePrivateDnsVerification',\n      parameters: {\n        ServiceId: config.serviceId,\n      },\n      physicalResourceId: PhysicalResourceId.of(Fn.join(':', [config.name, config.value])),\n    };\n    const startVerification = new AwsCustomResource(this, 'StartVerification', {\n      onCreate: startVerificationAction,\n      onUpdate: startVerificationAction,\n      policy: AwsCustomResourcePolicy.fromSdkCalls({\n        resources: [\n          Fn.join(':', [\n            'arn',\n            Stack.of(this).partition,\n            'ec2',\n            Stack.of(this).region,\n            Stack.of(this).account,\n            Fn.join('/', [\n              'vpc-endpoint-service',\n              config.serviceId,\n            ]),\n          ]),\n        ],\n      }),\n    });\n    // Only verify after the record has been created\n    startVerification.node.addDependency(verificationRecord);\n  }\n}\n\n/**\n * Represent the name/value pair associated with a Private DNS enabled endpoint service\n */\ninterface PrivateDnsConfiguration {\n  readonly name: string;\n  readonly value: string;\n  readonly serviceId: string;\n}\n\n/**\n * Hash a string\n */\nfunction hashcode(s: string): string {\n  const hash = crypto.createHash('md5');\n  hash.update(s);\n  return hash.digest('hex');\n};"]} |
\ | No newline at end of file |