UNPKG

28.3 kBJavaScriptView Raw
1"use strict";
2var _a;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.VpcEndpointServiceDomainName = void 0;
5const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
6const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
7const crypto = require("crypto");
8const core_1 = require("@aws-cdk/core");
9const custom_resources_1 = require("@aws-cdk/custom-resources");
10const 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
13const core_2 = require("@aws-cdk/core");
14/**
15 * A Private DNS configuration for a VPC endpoint service.
16 */
17class 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}
174exports.VpcEndpointServiceDomainName = VpcEndpointServiceDomainName;
175_a = JSII_RTTI_SYMBOL_1;
176VpcEndpointServiceDomainName[_a] = { fqn: "@aws-cdk/aws-route53.VpcEndpointServiceDomainName", version: "1.160.0" };
177// Track all domain names created, so someone doesn't accidentally associate two domains with a single service
178VpcEndpointServiceDomainName.endpointServices = [];
179// Track all domain names created, so someone doesn't accidentally associate two domains with a single service
180VpcEndpointServiceDomainName.endpointServicesMap = {};
181/**
182 * Hash a string
183 */
184function 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