From 174515f9287ed6f665dca633a9764714b906686c Mon Sep 17 00:00:00 2001 From: Jonathan Goldwasser Date: Tue, 4 Jun 2019 16:31:57 +0200 Subject: [PATCH 1/6] feat(route53): improve constructs for basic records Constructs for basic records (CNAME, TXT, etc.) now extend the `BasicRecord` construct and offer better typed properties interfaces. Add constructs for A, AAAA, CAA, MX and SRV records. Add support for multiple values in basic records. Make `recordName` optional with default to zone root. Add a "security" `CaaAmazonRecord` construct to easily restrict certificate authorities allowed to issue certificates for a domain to Amazon only. BREAKING CHANGE: `recordValue: string` prop in `route53.TxtRecord` changed to `values: string[]` * `recordValue` prop in `route53.CnameRecord` renamed to `domainName` --- packages/@aws-cdk/aws-route53/README.md | 33 +- .../@aws-cdk/aws-route53/lib/records/basic.ts | 362 ++++++++++++++++++ .../@aws-cdk/aws-route53/lib/records/cname.ts | 47 --- .../@aws-cdk/aws-route53/lib/records/index.ts | 3 +- .../@aws-cdk/aws-route53/lib/records/txt.ts | 49 --- .../test/integ.route53.expected.json | 342 ++--------------- .../aws-route53/test/integ.route53.ts | 24 +- .../aws-route53/test/test.basic-record.ts | 355 +++++++++++++++++ .../aws-route53/test/test.cname-record.ts | 67 ---- .../@aws-cdk/aws-route53/test/test.route53.ts | 2 +- .../aws-route53/test/test.txt-record.ts | 50 --- 11 files changed, 793 insertions(+), 541 deletions(-) create mode 100644 packages/@aws-cdk/aws-route53/lib/records/basic.ts delete mode 100644 packages/@aws-cdk/aws-route53/lib/records/cname.ts delete mode 100644 packages/@aws-cdk/aws-route53/lib/records/txt.ts create mode 100644 packages/@aws-cdk/aws-route53/test/test.basic-record.ts delete mode 100644 packages/@aws-cdk/aws-route53/test/test.cname-record.ts delete mode 100644 packages/@aws-cdk/aws-route53/test/test.txt-record.ts diff --git a/packages/@aws-cdk/aws-route53/README.md b/packages/@aws-cdk/aws-route53/README.md index 336cf7c0dfb6b..44489dc7da90e 100644 --- a/packages/@aws-cdk/aws-route53/README.md +++ b/packages/@aws-cdk/aws-route53/README.md @@ -6,7 +6,7 @@ To add a public hosted zone: import route53 = require('@aws-cdk/aws-route53'); new route53.PublicHostedZone(this, 'HostedZone', { - zoneName: 'fully.qualified.domain.com' + zoneName: 'fully.qualified.domain.com' }); ``` @@ -21,8 +21,8 @@ import route53 = require('@aws-cdk/aws-route53'); const vpc = new ec2.VpcNetwork(this, 'VPC'); const zone = new route53.PrivateHostedZone(this, 'HostedZone', { - zoneName: 'fully.qualified.domain.com', - vpc // At least one VPC has to be added to a Private Hosted Zone. + zoneName: 'fully.qualified.domain.com', + vpc // At least one VPC has to be added to a Private Hosted Zone. }); ``` @@ -34,15 +34,24 @@ To add a TXT record to your zone: ```ts import route53 = require('@aws-cdk/aws-route53'); -new route53.TxtRecord(zone, 'TXTRecord', { - recordName: '_foo', // If the name ends with a ".", it will be used as-is; - // if it ends with a "." followed by the zone name, a trailing "." will be added automatically; - // otherwise, a ".", the zone name, and a trailing "." will be added automatically. - recordValue: 'Bar!', // Will be quoted for you, and " will be escaped automatically. - ttl: 90, // Optional - default is 1800 +new route53.TxtRecord(this, 'TXTRecord', { + zone: myZone, + recordName: '_foo', // If the name ends with a ".", it will be used as-is; + // if it ends with a "." followed by the zone name, a trailing "." will be added automatically; + // otherwise, a ".", the zone name, and a trailing "." will be added automatically. + // Defaults to zone root if not specified. + values: [ // Will be quoted for you, and " will be escaped automatically. + 'Bar!', + 'Baz?' + ], + ttl: 90, // Optional - default is 1800 }); ``` +Constructs are available for A, AAAA, CAA, CNAME, MX, SRV and TXT records. + +Use the `CaaAmazonRecord` construct to easily restrict certificate authorities +allowed to issue certificates for a domain to Amazon only. ### Adding records to existing hosted zones @@ -50,8 +59,8 @@ If you know the ID and Name of a Hosted Zone, you can import it directly: ```ts const zone = HostedZone.import(this, 'MyZone', { - zoneName: 'example.com', - hostedZoneId: 'ZOJJZC49E0EPZ', + zoneName: 'example.com', + hostedZoneId: 'ZOJJZC49E0EPZ', }); ``` @@ -60,6 +69,6 @@ to discover and import it: ```ts const zone = new HostedZoneProvider(this, { - domainName: 'example.com' + domainName: 'example.com' }).findAndImport(this, 'MyZone'); ``` diff --git a/packages/@aws-cdk/aws-route53/lib/records/basic.ts b/packages/@aws-cdk/aws-route53/lib/records/basic.ts new file mode 100644 index 0000000000000..e38315c4cad02 --- /dev/null +++ b/packages/@aws-cdk/aws-route53/lib/records/basic.ts @@ -0,0 +1,362 @@ +import { Construct, Resource } from '@aws-cdk/cdk'; +import { IHostedZone } from '../hosted-zone-ref'; +import { CfnRecordSet } from '../route53.generated'; +import { determineFullyQualifiedDomainName } from './_util'; + +/** + * The record type. + */ +export enum RecordType { + A = 'A', + AAAA = 'AAAA', + CAA = 'CAA', + CNAME = 'CNAME', + MX = 'MX', + NAPTR = 'NAPTR', + NS = 'NS', + PTR = 'PTR', + SOA = 'SOA', + SPF = 'SPF', + SRV = 'SRV', + TXT = 'TXT' +} + +/** + * Propreties for a record. + */ +export interface RecordProps { + /** + * The hosted zone in which to define the new record. + */ + readonly zone: IHostedZone; + + /** + * The domain name for this record. + * + * @default zone root + */ + readonly recordName?: string; + + /** + * The resource record cache time to live (TTL) in seconds. + * + * @default 1800 seconds + */ + readonly ttl?: number; +} + +/** + * Construction properties for a BasicRecord. + */ +export interface BasicRecordProps extends RecordProps { + /** + * The record type. + */ + readonly recordType: RecordType; + + /** + * The values for this record. + */ + readonly recordValues: string[]; +} + +/** + * A basic record + * + * @resource AWS::Route53::RecordSet + * + * @see https://docs.aws.amazon.com/Route53/latest/DeveloperGuide/resource-record-sets-values-basic.html + */ +export class BasicRecord extends Resource { + constructor(scope: Construct, id: string, props: BasicRecordProps) { + super(scope, id); + + const ttl = props.ttl === undefined ? 1800 : props.ttl; + + new CfnRecordSet(this, 'Resource', { + hostedZoneId: props.zone.hostedZoneId, + name: determineFullyQualifiedDomainName(props.recordName || props.zone.zoneName, props.zone), + type: props.recordType, + resourceRecords: props.recordValues, + ttl: ttl.toString() + }); + } +} + +/** + * Construction properties for a ARecord. + */ +export interface ARecordProps extends RecordProps { + /** + * The IP addresses. + */ + readonly ipAddresses: string[]; +} + +/** + * A DNS A record + * + * @resource AWS::Route53::RecordSet + */ +export class ARecord extends BasicRecord { + constructor(scope: Construct, id: string, props: ARecordProps) { + super(scope, id, { + ...props, + recordType: RecordType.A, + recordValues: props.ipAddresses, + }); + } +} + +/** + * Construction properties for a AaaaRecord. + */ +export interface AaaaRecordProps extends RecordProps, ARecordProps {} + +/** + * A DNS AAAA record + * + * @resource AWS::Route53::RecordSet + */ +export class AaaaRecord extends BasicRecord { + constructor(scope: Construct, id: string, props: AaaaRecordProps) { + super(scope, id, { + ...props, + recordType: RecordType.AAAA, + recordValues: props.ipAddresses, + }); + } +} + +/** + * Construction properties for a CnameRecord. + */ +export interface CnameRecordProps extends RecordProps { + /** + * The domain name. + */ + readonly domainName: string; +} + +/** + * A DNS CNAME record + * + * @resource AWS::Route53::RecordSet + */ +export class CnameRecord extends BasicRecord { + constructor(scope: Construct, id: string, props: CnameRecordProps) { + super(scope, id, { + ...props, + recordType: RecordType.CNAME, + recordValues: [props.domainName] + }); + } +} + +/** + * Construction properties for a TxtRecord. + */ +export interface TxtRecordProps extends RecordProps { + /** + * The text values. + */ + readonly values: string[]; +} + +/** + * A DNS TXT record + * + * @resource AWS::Route53::RecordSet + */ +export class TxtRecord extends BasicRecord { + constructor(scope: Construct, id: string, props: TxtRecordProps) { + super(scope, id, { + ...props, + recordType: RecordType.TXT, + recordValues: props.values && props.values.map(v => JSON.stringify(v)), + }); + } +} + +/** + * Properties for a SRV record value. + */ +export interface SrvRecordValue { + /** + * The priority. + */ + readonly priority: number; + + /** + * The weight. + */ + readonly weight: number; + + /** + * The port. + */ + readonly port: number; + + /** + * The server host name. + */ + readonly hostName: string; +} +/** + * Construction properties for a SrvRecord. + */ +export interface SrvRecordProps extends RecordProps { + /** + * The values. + */ + readonly values: SrvRecordValue[]; +} + +/** + * A DNS SRV record + * + * @resource AWS::Route53::RecordSet + */ +export class SrvRecord extends BasicRecord { + constructor(scope: Construct, id: string, props: SrvRecordProps) { + super(scope, id, { + ...props, + recordType: RecordType.SRV, + recordValues: props.values.map(v => `${v.priority} ${v.weight} ${v.port} ${v.hostName}`), + }); + } +} + +/** + * The CAA tag. + */ +export enum CaaTag { + /** + * Explicity authorizes a single certificate authority to issue a + * certificate (any type) for the hostname. + */ + ISSUE = 'issue', + + /** + * Explicity authorizes a single certificate authority to issue a + * wildcard certificate (and only wildcard) for the hostname. + */ + ISSUEWILD = 'issuewild', + + /** + * Specifies a URL to which a certificate authority may report policy + * violations. + */ + IODEF = 'iodef', +} + +/** + * Properties for a CAA record value. + */ +export interface CaaRecordValue { + /** + * The flag. + */ + readonly flag: number; + + /** + * The tag. + */ + readonly tag: CaaTag; + + /** + * The value associated with the tag. + */ + readonly value: string; +} + +/** + * Construction properties for a CaaRecord. + */ +export interface CaaRecordProps extends RecordProps { + /** + * The values. + */ + readonly values: CaaRecordValue[]; +} + +/** + * A DNS CAA record + * + * @resource AWS::Route53::RecordSet + */ +export class CaaRecord extends BasicRecord { + constructor(scope: Construct, id: string, props: CaaRecordProps) { + super(scope, id, { + ...props, + recordType: RecordType.CAA, + recordValues: props.values.map(v => `${v.flag} ${v.tag} "${v.value}"`), + }); + } +} + +/** + * Construction properties for a CaaAmazonRecord. + */ +export interface CaaAmazonRecordProps extends RecordProps {} + +/** + * A DNS Amazon CAA record + * + * @resource AWS::Route53::RecordSet + */ +export class CaaAmazonRecord extends CaaRecord { + constructor(scope: Construct, id: string, props: CaaAmazonRecordProps) { + super(scope, id, { + ...props, + values: [ + { + flag: 0, + tag: CaaTag.ISSUE, + value: 'amazon.com' + } + ], + recordName: props.zone.zoneName + }); + } +} + +/** + * Properties for a MX record value. + */ +export interface MxRecordValue { + /** + * The priority. + */ + readonly priority: number; + + /** + * The mail server host name. + */ + readonly hostName: string; +} + +/** + * Construction properties for a MxRecord. + */ +export interface MxRecordProps extends RecordProps { + /** + * The values. + */ + readonly values: MxRecordValue[]; +} + +/** + * A DNS MX record + * + * @resource AWS::Route53::RecordSet + */ +export class MxRecord extends BasicRecord { + constructor(scope: Construct, id: string, props: MxRecordProps) { + super(scope, id, { + ...props, + recordType: RecordType.MX, + recordValues: props.values.map(v => `${v.priority} ${v.hostName}`) + }); + } +} diff --git a/packages/@aws-cdk/aws-route53/lib/records/cname.ts b/packages/@aws-cdk/aws-route53/lib/records/cname.ts deleted file mode 100644 index 55e9a7bd3f32a..0000000000000 --- a/packages/@aws-cdk/aws-route53/lib/records/cname.ts +++ /dev/null @@ -1,47 +0,0 @@ -import { Construct } from '@aws-cdk/cdk'; -import { IHostedZone } from '../hosted-zone-ref'; -import { CfnRecordSet } from '../route53.generated'; -import { determineFullyQualifiedDomainName } from './_util'; - -export interface CnameRecordProps { - /** - * The hosted zone in which to define the new TXT record. - */ - readonly zone: IHostedZone; - - /** - * The domain name for this record set. - */ - readonly recordName: string; - - /** - * The value for this record set. - */ - readonly recordValue: string; - - /** - * The resource record cache time to live (TTL) in seconds. - * - * @default 1800 seconds - */ - readonly ttl?: number; -} - -/** - * A DNS CNAME record - */ -export class CnameRecord extends Construct { - constructor(scope: Construct, id: string, props: CnameRecordProps) { - super(scope, id); - - const ttl = props.ttl === undefined ? 1800 : props.ttl; - - new CfnRecordSet(this, 'Resource', { - hostedZoneId: props.zone.hostedZoneId, - name: determineFullyQualifiedDomainName(props.recordName, props.zone), - type: 'CNAME', - resourceRecords: [ props.recordValue ], - ttl: ttl.toString(), - }); - } -} diff --git a/packages/@aws-cdk/aws-route53/lib/records/index.ts b/packages/@aws-cdk/aws-route53/lib/records/index.ts index 1643af3bdaadb..3ecd42b96de83 100644 --- a/packages/@aws-cdk/aws-route53/lib/records/index.ts +++ b/packages/@aws-cdk/aws-route53/lib/records/index.ts @@ -1,4 +1,3 @@ export * from './alias'; -export * from './txt'; -export * from './cname'; +export * from './basic'; export * from './zone-delegation'; diff --git a/packages/@aws-cdk/aws-route53/lib/records/txt.ts b/packages/@aws-cdk/aws-route53/lib/records/txt.ts deleted file mode 100644 index 64bd769848640..0000000000000 --- a/packages/@aws-cdk/aws-route53/lib/records/txt.ts +++ /dev/null @@ -1,49 +0,0 @@ -import { Construct } from '@aws-cdk/cdk'; -import { IHostedZone } from '../hosted-zone-ref'; -import { CfnRecordSet } from '../route53.generated'; -import { determineFullyQualifiedDomainName } from './_util'; - -export interface TxtRecordProps { - /** - * The hosted zone in which to define the new TXT record. - */ - readonly zone: IHostedZone; - - /** - * The domain name for this record set. - */ - readonly recordName: string; - - /** - * The value for this record set. - */ - readonly recordValue: string; - - /** - * The resource record cache time to live (TTL) in seconds. - * - * @default 1800 seconds - */ - readonly ttl?: number; -} - -/** - * A DNS TXT record - */ -export class TxtRecord extends Construct { - constructor(scope: Construct, id: string, props: TxtRecordProps) { - super(scope, id); - - // JSON.stringify conveniently wraps strings in " and escapes ". - const recordValue = JSON.stringify(props.recordValue); - const ttl = props.ttl === undefined ? 1800 : props.ttl; - - new CfnRecordSet(this, 'Resource', { - hostedZoneId: props.zone.hostedZoneId, - name: determineFullyQualifiedDomainName(props.recordName, props.zone), - type: 'TXT', - resourceRecords: [recordValue], - ttl: ttl.toString() - }); - } -} diff --git a/packages/@aws-cdk/aws-route53/test/integ.route53.expected.json b/packages/@aws-cdk/aws-route53/test/integ.route53.expected.json index 9cde321023c0f..4a1797a096e06 100644 --- a/packages/@aws-cdk/aws-route53/test/integ.route53.expected.json +++ b/packages/@aws-cdk/aws-route53/test/integ.route53.expected.json @@ -18,7 +18,7 @@ "VPCPublicSubnet1SubnetB4246D30": { "Type": "AWS::EC2::Subnet", "Properties": { - "CidrBlock": "10.0.0.0/19", + "CidrBlock": "10.0.0.0/17", "VpcId": { "Ref": "VPCB9E5F0B4" }, @@ -106,192 +106,10 @@ ] } }, - "VPCPublicSubnet2Subnet74179F39": { - "Type": "AWS::EC2::Subnet", - "Properties": { - "CidrBlock": "10.0.32.0/19", - "VpcId": { - "Ref": "VPCB9E5F0B4" - }, - "AvailabilityZone": "test-region-1b", - "MapPublicIpOnLaunch": true, - "Tags": [ - { - "Key": "Name", - "Value": "aws-cdk-route53-integ/VPC/PublicSubnet2" - }, - { - "Key": "aws-cdk:subnet-name", - "Value": "Public" - }, - { - "Key": "aws-cdk:subnet-type", - "Value": "Public" - } - ] - } - }, - "VPCPublicSubnet2RouteTable6F1A15F1": { - "Type": "AWS::EC2::RouteTable", - "Properties": { - "VpcId": { - "Ref": "VPCB9E5F0B4" - }, - "Tags": [ - { - "Key": "Name", - "Value": "aws-cdk-route53-integ/VPC/PublicSubnet2" - } - ] - } - }, - "VPCPublicSubnet2RouteTableAssociation5A808732": { - "Type": "AWS::EC2::SubnetRouteTableAssociation", - "Properties": { - "RouteTableId": { - "Ref": "VPCPublicSubnet2RouteTable6F1A15F1" - }, - "SubnetId": { - "Ref": "VPCPublicSubnet2Subnet74179F39" - } - } - }, - "VPCPublicSubnet2DefaultRouteB7481BBA": { - "Type": "AWS::EC2::Route", - "Properties": { - "RouteTableId": { - "Ref": "VPCPublicSubnet2RouteTable6F1A15F1" - }, - "DestinationCidrBlock": "0.0.0.0/0", - "GatewayId": { - "Ref": "VPCIGWB7E252D3" - } - }, - "DependsOn": [ - "VPCVPCGW99B986DC" - ] - }, - "VPCPublicSubnet2EIP4947BC00": { - "Type": "AWS::EC2::EIP", - "Properties": { - "Domain": "vpc" - } - }, - "VPCPublicSubnet2NATGateway3C070193": { - "Type": "AWS::EC2::NatGateway", - "Properties": { - "AllocationId": { - "Fn::GetAtt": [ - "VPCPublicSubnet2EIP4947BC00", - "AllocationId" - ] - }, - "SubnetId": { - "Ref": "VPCPublicSubnet2Subnet74179F39" - }, - "Tags": [ - { - "Key": "Name", - "Value": "aws-cdk-route53-integ/VPC/PublicSubnet2" - } - ] - } - }, - "VPCPublicSubnet3Subnet631C5E25": { - "Type": "AWS::EC2::Subnet", - "Properties": { - "CidrBlock": "10.0.64.0/19", - "VpcId": { - "Ref": "VPCB9E5F0B4" - }, - "AvailabilityZone": "test-region-1c", - "MapPublicIpOnLaunch": true, - "Tags": [ - { - "Key": "Name", - "Value": "aws-cdk-route53-integ/VPC/PublicSubnet3" - }, - { - "Key": "aws-cdk:subnet-name", - "Value": "Public" - }, - { - "Key": "aws-cdk:subnet-type", - "Value": "Public" - } - ] - } - }, - "VPCPublicSubnet3RouteTable98AE0E14": { - "Type": "AWS::EC2::RouteTable", - "Properties": { - "VpcId": { - "Ref": "VPCB9E5F0B4" - }, - "Tags": [ - { - "Key": "Name", - "Value": "aws-cdk-route53-integ/VPC/PublicSubnet3" - } - ] - } - }, - "VPCPublicSubnet3RouteTableAssociation427FE0C6": { - "Type": "AWS::EC2::SubnetRouteTableAssociation", - "Properties": { - "RouteTableId": { - "Ref": "VPCPublicSubnet3RouteTable98AE0E14" - }, - "SubnetId": { - "Ref": "VPCPublicSubnet3Subnet631C5E25" - } - } - }, - "VPCPublicSubnet3DefaultRouteA0D29D46": { - "Type": "AWS::EC2::Route", - "Properties": { - "RouteTableId": { - "Ref": "VPCPublicSubnet3RouteTable98AE0E14" - }, - "DestinationCidrBlock": "0.0.0.0/0", - "GatewayId": { - "Ref": "VPCIGWB7E252D3" - } - }, - "DependsOn": [ - "VPCVPCGW99B986DC" - ] - }, - "VPCPublicSubnet3EIPAD4BC883": { - "Type": "AWS::EC2::EIP", - "Properties": { - "Domain": "vpc" - } - }, - "VPCPublicSubnet3NATGatewayD3048F5C": { - "Type": "AWS::EC2::NatGateway", - "Properties": { - "AllocationId": { - "Fn::GetAtt": [ - "VPCPublicSubnet3EIPAD4BC883", - "AllocationId" - ] - }, - "SubnetId": { - "Ref": "VPCPublicSubnet3Subnet631C5E25" - }, - "Tags": [ - { - "Key": "Name", - "Value": "aws-cdk-route53-integ/VPC/PublicSubnet3" - } - ] - } - }, "VPCPrivateSubnet1Subnet8BCA10E0": { "Type": "AWS::EC2::Subnet", "Properties": { - "CidrBlock": "10.0.96.0/19", + "CidrBlock": "10.0.128.0/17", "VpcId": { "Ref": "VPCB9E5F0B4" }, @@ -350,130 +168,6 @@ } } }, - "VPCPrivateSubnet2SubnetCFCDAA7A": { - "Type": "AWS::EC2::Subnet", - "Properties": { - "CidrBlock": "10.0.128.0/19", - "VpcId": { - "Ref": "VPCB9E5F0B4" - }, - "AvailabilityZone": "test-region-1b", - "MapPublicIpOnLaunch": false, - "Tags": [ - { - "Key": "Name", - "Value": "aws-cdk-route53-integ/VPC/PrivateSubnet2" - }, - { - "Key": "aws-cdk:subnet-name", - "Value": "Private" - }, - { - "Key": "aws-cdk:subnet-type", - "Value": "Private" - } - ] - } - }, - "VPCPrivateSubnet2RouteTable0A19E10E": { - "Type": "AWS::EC2::RouteTable", - "Properties": { - "VpcId": { - "Ref": "VPCB9E5F0B4" - }, - "Tags": [ - { - "Key": "Name", - "Value": "aws-cdk-route53-integ/VPC/PrivateSubnet2" - } - ] - } - }, - "VPCPrivateSubnet2RouteTableAssociation0C73D413": { - "Type": "AWS::EC2::SubnetRouteTableAssociation", - "Properties": { - "RouteTableId": { - "Ref": "VPCPrivateSubnet2RouteTable0A19E10E" - }, - "SubnetId": { - "Ref": "VPCPrivateSubnet2SubnetCFCDAA7A" - } - } - }, - "VPCPrivateSubnet2DefaultRouteF4F5CFD2": { - "Type": "AWS::EC2::Route", - "Properties": { - "RouteTableId": { - "Ref": "VPCPrivateSubnet2RouteTable0A19E10E" - }, - "DestinationCidrBlock": "0.0.0.0/0", - "NatGatewayId": { - "Ref": "VPCPublicSubnet2NATGateway3C070193" - } - } - }, - "VPCPrivateSubnet3Subnet3EDCD457": { - "Type": "AWS::EC2::Subnet", - "Properties": { - "CidrBlock": "10.0.160.0/19", - "VpcId": { - "Ref": "VPCB9E5F0B4" - }, - "AvailabilityZone": "test-region-1c", - "MapPublicIpOnLaunch": false, - "Tags": [ - { - "Key": "Name", - "Value": "aws-cdk-route53-integ/VPC/PrivateSubnet3" - }, - { - "Key": "aws-cdk:subnet-name", - "Value": "Private" - }, - { - "Key": "aws-cdk:subnet-type", - "Value": "Private" - } - ] - } - }, - "VPCPrivateSubnet3RouteTable192186F8": { - "Type": "AWS::EC2::RouteTable", - "Properties": { - "VpcId": { - "Ref": "VPCB9E5F0B4" - }, - "Tags": [ - { - "Key": "Name", - "Value": "aws-cdk-route53-integ/VPC/PrivateSubnet3" - } - ] - } - }, - "VPCPrivateSubnet3RouteTableAssociationC28D144E": { - "Type": "AWS::EC2::SubnetRouteTableAssociation", - "Properties": { - "RouteTableId": { - "Ref": "VPCPrivateSubnet3RouteTable192186F8" - }, - "SubnetId": { - "Ref": "VPCPrivateSubnet3Subnet3EDCD457" - } - } - }, - "VPCPrivateSubnet3DefaultRoute27F311AE": { - "Type": "AWS::EC2::Route", - "Properties": { - "RouteTableId": { - "Ref": "VPCPrivateSubnet3RouteTable192186F8" - }, - "DestinationCidrBlock": "0.0.0.0/0", - "NatGatewayId": { - "Ref": "VPCPublicSubnet3NATGatewayD3048F5C" - } - } - }, "VPCIGWB7E252D3": { "Type": "AWS::EC2::InternetGateway", "Properties": { @@ -521,7 +215,8 @@ "Ref": "PrivateZone27242E85" }, "ResourceRecords": [ - "\"Bar!\"" + "\"Bar!\"", + "\"Baz?\"" ], "TTL": "60" } @@ -568,6 +263,35 @@ ], "TTL": "1800" } + }, + "ACCC8ACD5": { + "Type": "AWS::Route53::RecordSet", + "Properties": { + "Name": "test.cdk.local.", + "Type": "A", + "HostedZoneId": { + "Ref": "PrivateZone27242E85" + }, + "ResourceRecords": [ + "1.2.3.4", + "5.6.7.8" + ], + "TTL": "1800" + } + }, + "CaaAmazon40DF725F": { + "Type": "AWS::Route53::RecordSet", + "Properties": { + "Name": "cdk.test.", + "Type": "CAA", + "HostedZoneId": { + "Ref": "PublicZone2E1C4E34" + }, + "ResourceRecords": [ + "0 issue \"amazon.com\"" + ], + "TTL": "1800" + } } }, "Outputs": { diff --git a/packages/@aws-cdk/aws-route53/test/integ.route53.ts b/packages/@aws-cdk/aws-route53/test/integ.route53.ts index 625357e5717b6..cefbc108d3bb0 100644 --- a/packages/@aws-cdk/aws-route53/test/integ.route53.ts +++ b/packages/@aws-cdk/aws-route53/test/integ.route53.ts @@ -1,12 +1,12 @@ import ec2 = require('@aws-cdk/aws-ec2'); import cdk = require('@aws-cdk/cdk'); -import { CnameRecord, PrivateHostedZone, PublicHostedZone, TxtRecord } from '../lib'; +import { ARecord, CaaAmazonRecord, CnameRecord, PrivateHostedZone, PublicHostedZone, TxtRecord } from '../lib'; const app = new cdk.App(); const stack = new cdk.Stack(app, 'aws-cdk-route53-integ'); -const vpc = new ec2.Vpc(stack, 'VPC'); +const vpc = new ec2.Vpc(stack, 'VPC', { maxAZs: 1 }); const privateZone = new PrivateHostedZone(stack, 'PrivateZone', { zoneName: 'cdk.local', vpc @@ -23,14 +23,30 @@ publicZone.addDelegation(publicSubZone); new TxtRecord(privateZone, 'TXT', { zone: privateZone, recordName: '_foo', - recordValue: 'Bar!', + values: [ + 'Bar!', + 'Baz?' + ], ttl: 60 }); new CnameRecord(stack, 'CNAME', { zone: privateZone, recordName: 'www', - recordValue: 'server' + domainName: 'server' +}); + +new ARecord(stack, 'A', { + zone: privateZone, + recordName: 'test', + ipAddresses: [ + '1.2.3.4', + '5.6.7.8' + ] +}); + +new CaaAmazonRecord(stack, 'CaaAmazon', { + zone: publicZone }); new cdk.CfnOutput(stack, 'PrivateZoneId', { value: privateZone.hostedZoneId }); diff --git a/packages/@aws-cdk/aws-route53/test/test.basic-record.ts b/packages/@aws-cdk/aws-route53/test/test.basic-record.ts new file mode 100644 index 0000000000000..8d339013b61ec --- /dev/null +++ b/packages/@aws-cdk/aws-route53/test/test.basic-record.ts @@ -0,0 +1,355 @@ +import { expect, haveResource } from '@aws-cdk/assert'; +import { Stack } from '@aws-cdk/cdk'; +import { Test } from 'nodeunit'; +import route53 = require('../lib'); + +export = { + 'with default ttl'(test: Test) { + // GIVEN + const stack = new Stack(); + + // WHEN + const zone = new route53.HostedZone(stack, 'HostedZone', { + zoneName: 'myzone' + }); + + new route53.BasicRecord(stack, 'Basic', { + zone, + recordName: 'www', + recordType: route53.RecordType.CNAME, + recordValues: ['zzz'] + }); + + // THEN + expect(stack).to(haveResource('AWS::Route53::RecordSet', { + Name: "www.myzone.", + Type: "CNAME", + HostedZoneId: { + Ref: "HostedZoneDB99F866" + }, + ResourceRecords: [ + "zzz" + ], + TTL: "1800" + })); + test.done(); + }, + + 'with custom ttl'(test: Test) { + // GIVEN + const stack = new Stack(); + + // WHEN + const zone = new route53.HostedZone(stack, 'HostedZone', { + zoneName: 'myzone' + }); + + new route53.BasicRecord(stack, 'Basic', { + zone, + recordName: 'aa', + recordType: route53.RecordType.CNAME, + recordValues: ['bbb'], + ttl: 6077 + }); + + // THEN + expect(stack).to(haveResource('AWS::Route53::RecordSet', { + Name: "aa.myzone.", + Type: "CNAME", + HostedZoneId: { + Ref: "HostedZoneDB99F866" + }, + ResourceRecords: [ + "bbb" + ], + TTL: "6077" + })); + test.done(); + }, + + 'defaults to zone root'(test: Test) { + // GIVEN + const stack = new Stack(); + + // WHEN + const zone = new route53.HostedZone(stack, 'HostedZone', { + zoneName: 'myzone' + }); + + new route53.BasicRecord(stack, 'Basic', { + zone, + recordType: route53.RecordType.A, + recordValues: ['1.2.3.4'], + }); + + // THEN + expect(stack).to(haveResource('AWS::Route53::RecordSet', { + Name: "myzone.", + Type: "A", + HostedZoneId: { + Ref: "HostedZoneDB99F866" + }, + ResourceRecords: [ + "1.2.3.4" + ], + })); + test.done(); + }, + + 'A record'(test: Test) { + // GIVEN + const stack = new Stack(); + + // WHEN + const zone = new route53.HostedZone(stack, 'HostedZone', { + zoneName: 'myzone' + }); + + new route53.ARecord(stack, 'A', { + zone, + recordName: 'www', + ipAddresses: [ + '1.2.3.4', + '5.6.7.8' + ], + }); + + // THEN + expect(stack).to(haveResource('AWS::Route53::RecordSet', { + Name: "www.myzone.", + Type: "A", + HostedZoneId: { + Ref: "HostedZoneDB99F866" + }, + ResourceRecords: [ + "1.2.3.4", + "5.6.7.8" + ], + TTL: "1800" + })); + test.done(); + }, + + 'AAAA record'(test: Test) { + // GIVEN + const stack = new Stack(); + + // WHEN + const zone = new route53.HostedZone(stack, 'HostedZone', { + zoneName: 'myzone' + }); + + new route53.AaaaRecord(stack, 'AAAA', { + zone, + recordName: 'www', + ipAddresses: [ + '2001:0db8:85a3:0000:0000:8a2e:0370:7334' + ], + }); + + // THEN + expect(stack).to(haveResource('AWS::Route53::RecordSet', { + Name: "www.myzone.", + Type: "AAAA", + HostedZoneId: { + Ref: "HostedZoneDB99F866" + }, + ResourceRecords: [ + "2001:0db8:85a3:0000:0000:8a2e:0370:7334" + ], + TTL: "1800" + })); + test.done(); + }, + + 'CNAME record'(test: Test) { + // GIVEN + const stack = new Stack(); + + // WHEN + const zone = new route53.HostedZone(stack, 'HostedZone', { + zoneName: 'myzone' + }); + + new route53.CnameRecord(stack, 'CNAME', { + zone, + recordName: 'www', + domainName: 'hello', + }); + + // THEN + expect(stack).to(haveResource('AWS::Route53::RecordSet', { + Name: "www.myzone.", + Type: "CNAME", + HostedZoneId: { + Ref: "HostedZoneDB99F866" + }, + ResourceRecords: [ + "hello" + ], + TTL: "1800" + })); + test.done(); + }, + + 'TXT record'(test: Test) { + // GIVEN + const stack = new Stack(); + + // WHEN + const zone = new route53.HostedZone(stack, 'HostedZone', { + zoneName: 'myzone' + }); + + new route53.TxtRecord(stack, 'TXT', { + zone, + recordName: 'www', + values: ['should be enclosed with double quotes'] + }); + + // THEN + expect(stack).to(haveResource('AWS::Route53::RecordSet', { + Name: "www.myzone.", + Type: "TXT", + HostedZoneId: { + Ref: "HostedZoneDB99F866" + }, + ResourceRecords: [ + '"should be enclosed with double quotes"' + ], + TTL: "1800" + })); + test.done(); + }, + + 'SRV record'(test: Test) { + // GIVEN + const stack = new Stack(); + + // WHEN + const zone = new route53.HostedZone(stack, 'HostedZone', { + zoneName: 'myzone' + }); + + new route53.SrvRecord(stack, 'SRV', { + zone, + recordName: 'www', + values: [{ + hostName: 'aws.com', + port: 8080, + priority: 10, + weight: 5 + }] + }); + + // THEN + expect(stack).to(haveResource('AWS::Route53::RecordSet', { + Name: "www.myzone.", + Type: "SRV", + HostedZoneId: { + Ref: "HostedZoneDB99F866" + }, + ResourceRecords: [ + '10 5 8080 aws.com' + ], + TTL: "1800" + })); + test.done(); + }, + + 'CAA record'(test: Test) { + // GIVEN + const stack = new Stack(); + + // WHEN + const zone = new route53.HostedZone(stack, 'HostedZone', { + zoneName: 'myzone' + }); + + new route53.CaaRecord(stack, 'CAA', { + zone, + recordName: 'www', + values: [{ + flag: 0, + tag: route53.CaaTag.ISSUE, + value: 'ssl.com' + }] + }); + + // THEN + expect(stack).to(haveResource('AWS::Route53::RecordSet', { + Name: "www.myzone.", + Type: "CAA", + HostedZoneId: { + Ref: "HostedZoneDB99F866" + }, + ResourceRecords: [ + '0 issue "ssl.com"' + ], + TTL: "1800" + })); + test.done(); + }, + + 'CAA Amazon record'(test: Test) { + // GIVEN + const stack = new Stack(); + + // WHEN + const zone = new route53.HostedZone(stack, 'HostedZone', { + zoneName: 'myzone' + }); + + new route53.CaaAmazonRecord(stack, 'CAAAmazon', { + zone, + recordName: 'www', // should have no effect + }); + + // THEN + expect(stack).to(haveResource('AWS::Route53::RecordSet', { + Name: "myzone.", + Type: "CAA", + HostedZoneId: { + Ref: "HostedZoneDB99F866" + }, + ResourceRecords: [ + '0 issue "amazon.com"' + ], + TTL: "1800" + })); + test.done(); + }, + + 'MX record'(test: Test) { + // GIVEN + const stack = new Stack(); + + // WHEN + const zone = new route53.HostedZone(stack, 'HostedZone', { + zoneName: 'myzone' + }); + + new route53.MxRecord(stack, 'MX', { + zone, + recordName: 'mail', + values: [{ + hostName: 'workmail.aws', + priority: 10 + }] + }); + + // THEN + expect(stack).to(haveResource('AWS::Route53::RecordSet', { + Name: "mail.myzone.", + Type: "MX", + HostedZoneId: { + Ref: "HostedZoneDB99F866" + }, + ResourceRecords: [ + '10 workmail.aws' + ], + TTL: "1800" + })); + test.done(); + } +}; diff --git a/packages/@aws-cdk/aws-route53/test/test.cname-record.ts b/packages/@aws-cdk/aws-route53/test/test.cname-record.ts deleted file mode 100644 index 92d1b84b478fe..0000000000000 --- a/packages/@aws-cdk/aws-route53/test/test.cname-record.ts +++ /dev/null @@ -1,67 +0,0 @@ -import { expect, haveResource } from '@aws-cdk/assert'; -import { Stack } from '@aws-cdk/cdk'; -import { Test } from 'nodeunit'; -import route53 = require('../lib'); - -export = { - 'with default ttl'(test: Test) { - // GIVEN - const stack = new Stack(); - - // WHEN - const zone = new route53.HostedZone(stack, 'HostedZone', { - zoneName: 'myzone' - }); - - new route53.CnameRecord(stack, 'MyCname', { - zone, - recordName: 'www', - recordValue: 'zzz', - }); - - // THEN - expect(stack).to(haveResource('AWS::Route53::RecordSet', { - Name: "www.myzone.", - Type: "CNAME", - HostedZoneId: { - Ref: "HostedZoneDB99F866" - }, - ResourceRecords: [ - "zzz" - ], - TTL: "1800" - })); - test.done(); - }, - - 'with custom ttl'(test: Test) { - // GIVEN - const stack = new Stack(); - - // WHEN - const zone = new route53.HostedZone(stack, 'HostedZone', { - zoneName: 'myzone' - }); - - new route53.CnameRecord(stack, 'MyCname', { - zone, - recordName: 'aa', - recordValue: 'bbb', - ttl: 6077, - }); - - // THEN - expect(stack).to(haveResource('AWS::Route53::RecordSet', { - Name: "aa.myzone.", - Type: "CNAME", - HostedZoneId: { - Ref: "HostedZoneDB99F866" - }, - ResourceRecords: [ - "bbb" - ], - TTL: "6077" - })); - test.done(); - } -}; \ No newline at end of file diff --git a/packages/@aws-cdk/aws-route53/test/test.route53.ts b/packages/@aws-cdk/aws-route53/test/test.route53.ts index f2efdd0b53ad9..f4945795d5594 100644 --- a/packages/@aws-cdk/aws-route53/test/test.route53.ts +++ b/packages/@aws-cdk/aws-route53/test/test.route53.ts @@ -80,7 +80,7 @@ export = { new TxtRecord(importedZone as any, 'Record', { zone: importedZone, recordName: 'lookHere', - recordValue: 'SeeThere' + values: ['SeeThere'] }); expect(stack2).to(haveResource("AWS::Route53::RecordSet", { diff --git a/packages/@aws-cdk/aws-route53/test/test.txt-record.ts b/packages/@aws-cdk/aws-route53/test/test.txt-record.ts deleted file mode 100644 index c213b798813c2..0000000000000 --- a/packages/@aws-cdk/aws-route53/test/test.txt-record.ts +++ /dev/null @@ -1,50 +0,0 @@ -import { exactlyMatchTemplate, expect } from '@aws-cdk/assert'; -import { App, Stack } from '@aws-cdk/cdk'; -import { Test } from 'nodeunit'; -import { PublicHostedZone, TxtRecord } from '../lib'; - -export = { - 'TXT records': { - TXT(test: Test) { - const app = new TestApp(); - const zone = new PublicHostedZone(app.stack, 'HostedZone', { zoneName: 'test.public' }); - new TxtRecord(zone, 'TXT', { zone, recordName: '_foo', recordValue: 'Bar!' }); - expect(app.stack).to(exactlyMatchTemplate({ - Resources: { - HostedZoneDB99F866: { - Type: 'AWS::Route53::HostedZone', - Properties: { - Name: 'test.public.' - } - }, - HostedZoneTXT69C29760: { - Type: 'AWS::Route53::RecordSet', - Properties: { - HostedZoneId: { - Ref: 'HostedZoneDB99F866' - }, - Name: '_foo.test.public.', - ResourceRecords: ['"Bar!"'], - Type: 'TXT', - TTL: '1800' - } - } - } - })); - test.done(); - } - } -}; - -class TestApp { - public readonly stack: Stack; - private readonly app = new App(); - - constructor() { - const account = '123456789012'; - const region = 'bermuda-triangle'; - this.app.node.setContext(`availability-zones:${account}:${region}`, - [`${region}-1a`]); - this.stack = new Stack(this.app, 'MyStack', { env: { account, region } }); - } -} From c1c2263fcdb6a00af9f6aab57b2124f04ff9dcee Mon Sep 17 00:00:00 2001 From: Jonathan Goldwasser Date: Tue, 4 Jun 2019 21:00:45 +0200 Subject: [PATCH 2/6] caaAmazon for PublicHostedZone --- .../@aws-cdk/aws-route53/lib/hosted-zone.ts | 22 +++++++++++++++++-- .../@aws-cdk/aws-route53/lib/records/basic.ts | 5 ++++- .../@aws-cdk/aws-route53/test/test.route53.ts | 21 ++++++++++++++++++ 3 files changed, 45 insertions(+), 3 deletions(-) diff --git a/packages/@aws-cdk/aws-route53/lib/hosted-zone.ts b/packages/@aws-cdk/aws-route53/lib/hosted-zone.ts index af99ddb1efbb0..e84429c582a54 100644 --- a/packages/@aws-cdk/aws-route53/lib/hosted-zone.ts +++ b/packages/@aws-cdk/aws-route53/lib/hosted-zone.ts @@ -1,7 +1,7 @@ import ec2 = require('@aws-cdk/aws-ec2'); import { Construct, Resource, Token } from '@aws-cdk/cdk'; import { HostedZoneAttributes, IHostedZone } from './hosted-zone-ref'; -import { ZoneDelegationRecord } from './records'; +import { CaaAmazonRecord, ZoneDelegationRecord } from './records'; import { CfnHostedZone } from './route53.generated'; import { validateZoneName } from './util'; @@ -107,7 +107,19 @@ export class HostedZone extends Resource implements IHostedZone { } } -export interface PublicHostedZoneProps extends CommonHostedZoneProps { } +/** + * Construction properties for a PublicHostedZone. + */ +export interface PublicHostedZoneProps extends CommonHostedZoneProps { + /** + * Whether to create a CAA record to restrict certificate authorities allowed + * to issue certificates for this domain to Amazon only. + * + * @default false + */ + readonly caaAmazon?: boolean; +} + export interface IPublicHostedZone extends IHostedZone { } /** @@ -127,6 +139,12 @@ export class PublicHostedZone extends HostedZone implements IPublicHostedZone { constructor(scope: Construct, id: string, props: PublicHostedZoneProps) { super(scope, id, props); + + if (props.caaAmazon) { + new CaaAmazonRecord(this, 'CaaAmazon', { + zone: this + }); + } } public addVpc(_vpc: ec2.IVpc) { diff --git a/packages/@aws-cdk/aws-route53/lib/records/basic.ts b/packages/@aws-cdk/aws-route53/lib/records/basic.ts index e38315c4cad02..c9fd9622b139d 100644 --- a/packages/@aws-cdk/aws-route53/lib/records/basic.ts +++ b/packages/@aws-cdk/aws-route53/lib/records/basic.ts @@ -301,7 +301,10 @@ export class CaaRecord extends BasicRecord { export interface CaaAmazonRecordProps extends RecordProps {} /** - * A DNS Amazon CAA record + * A DNS Amazon CAA record. + * + * A CAA record to restrict certificate authorities allowed + * to issue certificates for a domain to Amazon only. * * @resource AWS::Route53::RecordSet */ diff --git a/packages/@aws-cdk/aws-route53/test/test.route53.ts b/packages/@aws-cdk/aws-route53/test/test.route53.ts index f4945795d5594..0daf47e07a08f 100644 --- a/packages/@aws-cdk/aws-route53/test/test.route53.ts +++ b/packages/@aws-cdk/aws-route53/test/test.route53.ts @@ -194,6 +194,27 @@ export = { })); test.done(); }, + + 'public hosted zone wiht caaAmazon set to true'(test: Test) { + // GIVEN + const stack = new cdk.Stack(); + + // WHEN + new PublicHostedZone(stack, 'MyHostedZone', { + zoneName: 'protected.com', + caaAmazon: true + }); + + // THEN + expect(stack).to(haveResource('AWS::Route53::RecordSet', { + Type: 'CAA', + Name: 'protected.com.', + ResourceRecords: [ + '0 issue "amazon.com"' + ] + })); + test.done(); + } }; class TestApp { From b1490a6ed9e2a1fb49d4b4181c15e1c99f506c99 Mon Sep 17 00:00:00 2001 From: Jonathan Goldwasser Date: Wed, 5 Jun 2019 18:08:52 +0200 Subject: [PATCH 3/6] RecordSet, refactor and clean up --- .../fargate/load-balanced-fargate-service.ts | 6 +- .../lib/cloudfront-target.ts | 2 +- .../lib/load-balancer-target.ts | 4 +- .../test/cloudfront-target.test.ts | 6 +- .../test/integ.alb-alias-target.ts | 4 +- .../test/integ.cloudfront-alias-target.ts | 4 +- .../test/load-balancer-target.test.ts | 4 +- packages/@aws-cdk/aws-route53/README.md | 23 ++- .../aws-route53/lib/alias-record-target.ts | 6 +- .../@aws-cdk/aws-route53/lib/hosted-zone.ts | 4 +- packages/@aws-cdk/aws-route53/lib/index.ts | 2 +- .../lib/{records/basic.ts => record-set.ts} | 153 +++++++++++++----- .../@aws-cdk/aws-route53/lib/records/_util.ts | 29 ---- .../@aws-cdk/aws-route53/lib/records/alias.ts | 51 ------ .../@aws-cdk/aws-route53/lib/records/index.ts | 3 - .../lib/records/zone-delegation.ts | 44 ----- packages/@aws-cdk/aws-route53/lib/util.ts | 30 ++++ .../aws-route53/test/integ.route53.ts | 7 +- .../aws-route53/test/test.alias-record.ts | 79 --------- ...est.basic-record.ts => test.record-set.ts} | 147 ++++++++++++++--- .../test/test.zone-delegation-record.ts | 54 ------- 21 files changed, 314 insertions(+), 348 deletions(-) rename packages/@aws-cdk/aws-route53/lib/{records/basic.ts => record-set.ts} (58%) delete mode 100644 packages/@aws-cdk/aws-route53/lib/records/_util.ts delete mode 100644 packages/@aws-cdk/aws-route53/lib/records/alias.ts delete mode 100644 packages/@aws-cdk/aws-route53/lib/records/index.ts delete mode 100644 packages/@aws-cdk/aws-route53/lib/records/zone-delegation.ts delete mode 100644 packages/@aws-cdk/aws-route53/test/test.alias-record.ts rename packages/@aws-cdk/aws-route53/test/{test.basic-record.ts => test.record-set.ts} (71%) delete mode 100644 packages/@aws-cdk/aws-route53/test/test.zone-delegation-record.ts diff --git a/packages/@aws-cdk/aws-ecs-patterns/lib/fargate/load-balanced-fargate-service.ts b/packages/@aws-cdk/aws-ecs-patterns/lib/fargate/load-balanced-fargate-service.ts index 0b2ddba2634f2..ff799522ca8d6 100644 --- a/packages/@aws-cdk/aws-ecs-patterns/lib/fargate/load-balanced-fargate-service.ts +++ b/packages/@aws-cdk/aws-ecs-patterns/lib/fargate/load-balanced-fargate-service.ts @@ -1,5 +1,5 @@ import ecs = require('@aws-cdk/aws-ecs'); -import { AliasRecord, IHostedZone } from '@aws-cdk/aws-route53'; +import { ARecord, IHostedZone, RecordTarget } from '@aws-cdk/aws-route53'; import targets = require('@aws-cdk/aws-route53-targets'); import cdk = require('@aws-cdk/cdk'); import { LoadBalancedServiceBase, LoadBalancedServiceBaseProps } from '../base/load-balanced-service-base'; @@ -120,10 +120,10 @@ export class LoadBalancedFargateService extends LoadBalancedServiceBase { throw new Error('A Route53 hosted domain zone name is required to configure the specified domain name'); } - new AliasRecord(this, "DNS", { + new ARecord(this, "DNS", { zone: props.domainZone, recordName: props.domainName, - target: new targets.LoadBalancerTarget(this.loadBalancer), + target: RecordTarget.fromAliasRecordTarget(new targets.LoadBalancerTarget(this.loadBalancer)), }); } } diff --git a/packages/@aws-cdk/aws-route53-targets/lib/cloudfront-target.ts b/packages/@aws-cdk/aws-route53-targets/lib/cloudfront-target.ts index eacb1285dd40e..b0e2aab31ead7 100644 --- a/packages/@aws-cdk/aws-route53-targets/lib/cloudfront-target.ts +++ b/packages/@aws-cdk/aws-route53-targets/lib/cloudfront-target.ts @@ -14,7 +14,7 @@ export class CloudFrontTarget implements route53.IAliasRecordTarget { constructor(private readonly distribution: cloudfront.CloudFrontWebDistribution) { } - public bind(_record: route53.IAliasRecord): route53.AliasRecordTargetProps { + public bind(_record: route53.IRecordSet): route53.AliasRecordTargetProps { return { hostedZoneId: CLOUDFRONT_ZONE_ID, dnsName: this.distribution.domainName diff --git a/packages/@aws-cdk/aws-route53-targets/lib/load-balancer-target.ts b/packages/@aws-cdk/aws-route53-targets/lib/load-balancer-target.ts index 274f4bb7587e2..c2d1e33126831 100644 --- a/packages/@aws-cdk/aws-route53-targets/lib/load-balancer-target.ts +++ b/packages/@aws-cdk/aws-route53-targets/lib/load-balancer-target.ts @@ -8,10 +8,10 @@ export class LoadBalancerTarget implements route53.IAliasRecordTarget { constructor(private readonly loadBalancer: elbv2.ILoadBalancerV2) { } - public bind(_record: route53.IAliasRecord): route53.AliasRecordTargetProps { + public bind(_record: route53.IRecordSet): route53.AliasRecordTargetProps { return { hostedZoneId: this.loadBalancer.loadBalancerCanonicalHostedZoneId, dnsName: this.loadBalancer.loadBalancerDnsName }; } -} \ No newline at end of file +} diff --git a/packages/@aws-cdk/aws-route53-targets/test/cloudfront-target.test.ts b/packages/@aws-cdk/aws-route53-targets/test/cloudfront-target.test.ts index d62ed2a4e1eb5..735dd145a869d 100644 --- a/packages/@aws-cdk/aws-route53-targets/test/cloudfront-target.test.ts +++ b/packages/@aws-cdk/aws-route53-targets/test/cloudfront-target.test.ts @@ -24,10 +24,10 @@ test('use CloudFront as record target', () => { // WHEN const zone = new route53.PublicHostedZone(stack, 'HostedZone', { zoneName: 'test.public' }); - new route53.AliasRecord(zone, 'Alias', { + new route53.ARecord(zone, 'Alias', { zone, recordName: '_foo', - target: new targets.CloudFrontTarget(distribution) + target: route53.RecordTarget.fromAliasRecordTarget(new targets.CloudFrontTarget(distribution)) }); // THEN @@ -37,4 +37,4 @@ test('use CloudFront as record target', () => { HostedZoneId: "Z2FDTNDATAQYW2" }, }); -}); \ No newline at end of file +}); diff --git a/packages/@aws-cdk/aws-route53-targets/test/integ.alb-alias-target.ts b/packages/@aws-cdk/aws-route53-targets/test/integ.alb-alias-target.ts index 37245fdaeba80..1ec8314b12c40 100644 --- a/packages/@aws-cdk/aws-route53-targets/test/integ.alb-alias-target.ts +++ b/packages/@aws-cdk/aws-route53-targets/test/integ.alb-alias-target.ts @@ -19,10 +19,10 @@ const lb = new elbv2.ApplicationLoadBalancer(stack, 'LB', { const zone = new route53.PublicHostedZone(stack, 'HostedZone', { zoneName: 'test.public' }); -new route53.AliasRecord(zone, 'Alias', { +new route53.ARecord(zone, 'Alias', { zone, recordName: '_foo', - target: new targets.LoadBalancerTarget(lb) + target: route53.RecordTarget.fromAliasRecordTarget(new targets.LoadBalancerTarget(lb)) }); app.run(); diff --git a/packages/@aws-cdk/aws-route53-targets/test/integ.cloudfront-alias-target.ts b/packages/@aws-cdk/aws-route53-targets/test/integ.cloudfront-alias-target.ts index e384dd9558cff..8f3695b85ee8c 100644 --- a/packages/@aws-cdk/aws-route53-targets/test/integ.cloudfront-alias-target.ts +++ b/packages/@aws-cdk/aws-route53-targets/test/integ.cloudfront-alias-target.ts @@ -25,10 +25,10 @@ const distribution = new cloudfront.CloudFrontWebDistribution(stack, 'MyDistribu ] }); -new route53.AliasRecord(zone, 'Alias', { +new route53.ARecord(zone, 'Alias', { zone, recordName: '_foo', - target: new targets.CloudFrontTarget(distribution) + target: route53.RecordTarget.fromAliasRecordTarget(new targets.CloudFrontTarget(distribution)) }); app.run(); diff --git a/packages/@aws-cdk/aws-route53-targets/test/load-balancer-target.test.ts b/packages/@aws-cdk/aws-route53-targets/test/load-balancer-target.test.ts index 623cb7bbe5023..df0f3879f4a9b 100644 --- a/packages/@aws-cdk/aws-route53-targets/test/load-balancer-target.test.ts +++ b/packages/@aws-cdk/aws-route53-targets/test/load-balancer-target.test.ts @@ -19,10 +19,10 @@ test('use ALB as record target', () => { const zone = new route53.PublicHostedZone(stack, 'HostedZone', { zoneName: 'test.public' }); // WHEN - new route53.AliasRecord(zone, 'Alias', { + new route53.ARecord(zone, 'Alias', { zone, recordName: '_foo', - target: new targets.LoadBalancerTarget(lb) + target: route53.RecordTarget.fromAliasRecordTarget(new targets.LoadBalancerTarget(lb)) }); // THEN diff --git a/packages/@aws-cdk/aws-route53/README.md b/packages/@aws-cdk/aws-route53/README.md index 44489dc7da90e..11709035d0ceb 100644 --- a/packages/@aws-cdk/aws-route53/README.md +++ b/packages/@aws-cdk/aws-route53/README.md @@ -48,7 +48,28 @@ new route53.TxtRecord(this, 'TXTRecord', { }); ``` -Constructs are available for A, AAAA, CAA, CNAME, MX, SRV and TXT records. +To add a A record to your zone: +```ts +import route53 = require('@aws-cdk/aws-route53'); + +new route53.AaaaRecord(this, 'AaaaRecord', { + zone: myZone, + target: route53.RecordTarget.fromIpAddresses('1.2.3.4', '5.6.7.8') +}) +``` + +To add a AAAA record pointing to a CloudFront distribution: +```ts +import route53 = require('@aws-cdk/aws-route53'); +import targets = require('@aws-cdk/aws-route53-targets'); + +new route53.AaaaRecord(this, 'Alias', { + zone: myZone, + target: route53.RecordTarget.fromAliasRecordTarget(new targets.CloudFrontTarget(distribution)) +}) +``` + +Constructs are available for A, AAAA, CAA, CNAME, MX, NS, SRV and TXT records. Use the `CaaAmazonRecord` construct to easily restrict certificate authorities allowed to issue certificates for a domain to Amazon only. diff --git a/packages/@aws-cdk/aws-route53/lib/alias-record-target.ts b/packages/@aws-cdk/aws-route53/lib/alias-record-target.ts index 66f17e95d7b86..219565d8aa9d6 100644 --- a/packages/@aws-cdk/aws-route53/lib/alias-record-target.ts +++ b/packages/@aws-cdk/aws-route53/lib/alias-record-target.ts @@ -1,4 +1,4 @@ -import { IAliasRecord } from "./records/alias"; +import { IRecordSet } from "./record-set"; /** * Classes that are valid alias record targets, like CloudFront distributions and load @@ -8,7 +8,7 @@ export interface IAliasRecordTarget { /** * Return hosted zone ID and DNS name, usable for Route53 alias targets */ - bind(record: IAliasRecord): AliasRecordTargetProps; + bind(record: IRecordSet): AliasRecordTargetProps; } /** @@ -24,4 +24,4 @@ export interface AliasRecordTargetProps { * DNS name of the target */ readonly dnsName: string; -} \ No newline at end of file +} diff --git a/packages/@aws-cdk/aws-route53/lib/hosted-zone.ts b/packages/@aws-cdk/aws-route53/lib/hosted-zone.ts index e84429c582a54..e89494dc02e70 100644 --- a/packages/@aws-cdk/aws-route53/lib/hosted-zone.ts +++ b/packages/@aws-cdk/aws-route53/lib/hosted-zone.ts @@ -1,7 +1,7 @@ import ec2 = require('@aws-cdk/aws-ec2'); import { Construct, Resource, Token } from '@aws-cdk/cdk'; import { HostedZoneAttributes, IHostedZone } from './hosted-zone-ref'; -import { CaaAmazonRecord, ZoneDelegationRecord } from './records'; +import { CaaAmazonRecord, ZoneDelegationRecord } from './record-set'; import { CfnHostedZone } from './route53.generated'; import { validateZoneName } from './util'; @@ -160,7 +160,7 @@ export class PublicHostedZone extends HostedZone implements IPublicHostedZone { public addDelegation(delegate: IPublicHostedZone, opts: ZoneDelegationOptions = {}): void { new ZoneDelegationRecord(this, `${this.zoneName} -> ${delegate.zoneName}`, { zone: this, - delegatedZoneName: delegate.zoneName, + recordName: delegate.zoneName, nameServers: delegate.hostedZoneNameServers!, // PublicHostedZones always have name servers! comment: opts.comment, ttl: opts.ttl, diff --git a/packages/@aws-cdk/aws-route53/lib/index.ts b/packages/@aws-cdk/aws-route53/lib/index.ts index 1abc508d84901..9a6ed0a853679 100644 --- a/packages/@aws-cdk/aws-route53/lib/index.ts +++ b/packages/@aws-cdk/aws-route53/lib/index.ts @@ -1,7 +1,7 @@ export * from './hosted-zone'; export * from './hosted-zone-provider'; export * from './hosted-zone-ref'; -export * from './records'; +export * from './record-set'; export * from './alias-record-target'; // AWS::Route53 CloudFormation Resources: diff --git a/packages/@aws-cdk/aws-route53/lib/records/basic.ts b/packages/@aws-cdk/aws-route53/lib/record-set.ts similarity index 58% rename from packages/@aws-cdk/aws-route53/lib/records/basic.ts rename to packages/@aws-cdk/aws-route53/lib/record-set.ts index c9fd9622b139d..c38a9ad54cabf 100644 --- a/packages/@aws-cdk/aws-route53/lib/records/basic.ts +++ b/packages/@aws-cdk/aws-route53/lib/record-set.ts @@ -1,7 +1,18 @@ -import { Construct, Resource } from '@aws-cdk/cdk'; -import { IHostedZone } from '../hosted-zone-ref'; -import { CfnRecordSet } from '../route53.generated'; -import { determineFullyQualifiedDomainName } from './_util'; +import { Construct, IResource, Resource, Token } from '@aws-cdk/cdk'; +import { IAliasRecordTarget } from './alias-record-target'; +import { IHostedZone } from './hosted-zone-ref'; +import { CfnRecordSet } from './route53.generated'; +import { determineFullyQualifiedDomainName } from './util'; + +/** + * A record set + */ +export interface IRecordSet extends IResource { + /** + * The domain name of the record + */ + readonly domainName: string; +} /** * The record type. @@ -22,9 +33,9 @@ export enum RecordType { } /** - * Propreties for a record. + * Options for a RecordSet. */ -export interface RecordProps { +export interface RecordSetOptions { /** * The hosted zone in which to define the new record. */ @@ -43,54 +54,90 @@ export interface RecordProps { * @default 1800 seconds */ readonly ttl?: number; + + /** + * A comment to add on the record. + * + * @default no comment + */ + readonly comment?: string; } /** - * Construction properties for a BasicRecord. + * Construction properties for a RecordSet. */ -export interface BasicRecordProps extends RecordProps { +export interface RecordSetProps extends RecordSetOptions { /** * The record type. */ readonly recordType: RecordType; /** - * The values for this record. + * The values for this record. Either `recordValues` or `target` must be + * specified. + * + * @default no values + */ + readonly recordValues?: string[]; + + /** + * The target for an alias record. Either `target` or `recordValues` must be + * specified. + * + * @default no target */ - readonly recordValues: string[]; + readonly target?: IAliasRecordTarget; } /** - * A basic record - * - * @resource AWS::Route53::RecordSet - * - * @see https://docs.aws.amazon.com/Route53/latest/DeveloperGuide/resource-record-sets-values-basic.html + * A record set. */ -export class BasicRecord extends Resource { - constructor(scope: Construct, id: string, props: BasicRecordProps) { +export class RecordSet extends Resource implements IRecordSet { + public readonly domainName: string; + + constructor(scope: Construct, id: string, props: RecordSetProps) { super(scope, id); - const ttl = props.ttl === undefined ? 1800 : props.ttl; + if (!(props.recordValues || props.target) || (props.recordValues && props.target)) { + throw new Error('Either `recordValues` or `target` must be specified.'); + } + + const ttl = props.target ? undefined : (props.ttl || 1800).toString(); - new CfnRecordSet(this, 'Resource', { + const recordSet = new CfnRecordSet(this, 'Resource', { hostedZoneId: props.zone.hostedZoneId, name: determineFullyQualifiedDomainName(props.recordName || props.zone.zoneName, props.zone), type: props.recordType, resourceRecords: props.recordValues, - ttl: ttl.toString() + aliasTarget: props.target && props.target.bind(this), + ttl, + comment: props.comment }); + + this.domainName = recordSet.ref; + } +} + +export class RecordTarget { + public static fromIpAddresses(...ipAddresses: string[]) { + return new RecordTarget(ipAddresses); + } + + public static fromAliasRecordTarget(aliasTarget: IAliasRecordTarget) { + return new RecordTarget(undefined, aliasTarget); + } + private constructor(public readonly values?: string[], public readonly aliasTarget?: IAliasRecordTarget) { } } /** * Construction properties for a ARecord. */ -export interface ARecordProps extends RecordProps { +export interface ARecordProps extends RecordSetOptions { /** - * The IP addresses. + * The target. */ - readonly ipAddresses: string[]; + readonly target: RecordTarget; } /** @@ -98,12 +145,13 @@ export interface ARecordProps extends RecordProps { * * @resource AWS::Route53::RecordSet */ -export class ARecord extends BasicRecord { +export class ARecord extends RecordSet { constructor(scope: Construct, id: string, props: ARecordProps) { super(scope, id, { ...props, recordType: RecordType.A, - recordValues: props.ipAddresses, + recordValues: props.target.values, + target: props.target.aliasTarget, }); } } @@ -111,19 +159,20 @@ export class ARecord extends BasicRecord { /** * Construction properties for a AaaaRecord. */ -export interface AaaaRecordProps extends RecordProps, ARecordProps {} +export interface AaaaRecordProps extends RecordSetOptions, ARecordProps {} /** * A DNS AAAA record * * @resource AWS::Route53::RecordSet */ -export class AaaaRecord extends BasicRecord { +export class AaaaRecord extends RecordSet { constructor(scope: Construct, id: string, props: AaaaRecordProps) { super(scope, id, { ...props, recordType: RecordType.AAAA, - recordValues: props.ipAddresses, + recordValues: props.target.values, + target: props.target.aliasTarget, }); } } @@ -131,7 +180,7 @@ export class AaaaRecord extends BasicRecord { /** * Construction properties for a CnameRecord. */ -export interface CnameRecordProps extends RecordProps { +export interface CnameRecordProps extends RecordSetOptions { /** * The domain name. */ @@ -143,7 +192,7 @@ export interface CnameRecordProps extends RecordProps { * * @resource AWS::Route53::RecordSet */ -export class CnameRecord extends BasicRecord { +export class CnameRecord extends RecordSet { constructor(scope: Construct, id: string, props: CnameRecordProps) { super(scope, id, { ...props, @@ -156,7 +205,7 @@ export class CnameRecord extends BasicRecord { /** * Construction properties for a TxtRecord. */ -export interface TxtRecordProps extends RecordProps { +export interface TxtRecordProps extends RecordSetOptions { /** * The text values. */ @@ -168,7 +217,7 @@ export interface TxtRecordProps extends RecordProps { * * @resource AWS::Route53::RecordSet */ -export class TxtRecord extends BasicRecord { +export class TxtRecord extends RecordSet { constructor(scope: Construct, id: string, props: TxtRecordProps) { super(scope, id, { ...props, @@ -205,7 +254,7 @@ export interface SrvRecordValue { /** * Construction properties for a SrvRecord. */ -export interface SrvRecordProps extends RecordProps { +export interface SrvRecordProps extends RecordSetOptions { /** * The values. */ @@ -217,7 +266,7 @@ export interface SrvRecordProps extends RecordProps { * * @resource AWS::Route53::RecordSet */ -export class SrvRecord extends BasicRecord { +export class SrvRecord extends RecordSet { constructor(scope: Construct, id: string, props: SrvRecordProps) { super(scope, id, { ...props, @@ -273,7 +322,7 @@ export interface CaaRecordValue { /** * Construction properties for a CaaRecord. */ -export interface CaaRecordProps extends RecordProps { +export interface CaaRecordProps extends RecordSetOptions { /** * The values. */ @@ -285,7 +334,7 @@ export interface CaaRecordProps extends RecordProps { * * @resource AWS::Route53::RecordSet */ -export class CaaRecord extends BasicRecord { +export class CaaRecord extends RecordSet { constructor(scope: Construct, id: string, props: CaaRecordProps) { super(scope, id, { ...props, @@ -298,7 +347,7 @@ export class CaaRecord extends BasicRecord { /** * Construction properties for a CaaAmazonRecord. */ -export interface CaaAmazonRecordProps extends RecordProps {} +export interface CaaAmazonRecordProps extends RecordSetOptions {} /** * A DNS Amazon CAA record. @@ -342,7 +391,7 @@ export interface MxRecordValue { /** * Construction properties for a MxRecord. */ -export interface MxRecordProps extends RecordProps { +export interface MxRecordProps extends RecordSetOptions { /** * The values. */ @@ -354,7 +403,7 @@ export interface MxRecordProps extends RecordProps { * * @resource AWS::Route53::RecordSet */ -export class MxRecord extends BasicRecord { +export class MxRecord extends RecordSet { constructor(scope: Construct, id: string, props: MxRecordProps) { super(scope, id, { ...props, @@ -363,3 +412,29 @@ export class MxRecord extends BasicRecord { }); } } + +/** + * Construction properties for a ZoneDelegationRecord + */ +export interface ZoneDelegationRecordProps extends RecordSetOptions { + /** + * The name servers to report in the delegation records. + */ + readonly nameServers: string[]; +} + +/** + * A record to delegate further lookups to a different set of name servers. + */ +export class ZoneDelegationRecord extends RecordSet { + constructor(scope: Construct, id: string, props: ZoneDelegationRecordProps) { + super(scope, id, { + ...props, + recordType: RecordType.NS, + recordValues: Token.isToken(props.nameServers) + ? props.nameServers // Can't map a string-array token! + : props.nameServers.map(ns => (Token.isToken(ns) || ns.endsWith('.')) ? ns : `${ns}.`), + ttl: props.ttl || 172_800 + }); + } +} diff --git a/packages/@aws-cdk/aws-route53/lib/records/_util.ts b/packages/@aws-cdk/aws-route53/lib/records/_util.ts deleted file mode 100644 index ba503721f258c..0000000000000 --- a/packages/@aws-cdk/aws-route53/lib/records/_util.ts +++ /dev/null @@ -1,29 +0,0 @@ -import { IHostedZone } from '../hosted-zone-ref'; - -/** - * Route53 requires the record names are specified as fully qualified names, but this - * forces lots of redundant work on the user (repeating the zone name over and over). - * This function allows the user to be lazier and offers a nicer experience, by - * qualifying relative names appropriately: - * - * @param providedName the user-specified name of the record. - * @param zoneName the fully-qualified name of the zone the record will be created in. - * - * @returns
    - *
  • If +providedName+ ends with a +.+, use it as-is
  • - *
  • If +providedName+ ends with or equals +zoneName+, append a trailing +.+
  • - *
  • Otherwise, append +.+, +zoneName+ and a trailing +.+
  • - *
- */ -export function determineFullyQualifiedDomainName(providedName: string, hostedZone: IHostedZone): string { - if (providedName.endsWith('.')) { - return providedName; - } - - const suffix = `.${hostedZone.zoneName}`; - if (providedName.endsWith(suffix) || providedName === hostedZone.zoneName) { - return `${providedName}.`; - } - - return `${providedName}${suffix}.`; -} diff --git a/packages/@aws-cdk/aws-route53/lib/records/alias.ts b/packages/@aws-cdk/aws-route53/lib/records/alias.ts deleted file mode 100644 index a933116e1fb14..0000000000000 --- a/packages/@aws-cdk/aws-route53/lib/records/alias.ts +++ /dev/null @@ -1,51 +0,0 @@ -import { Construct, IConstruct } from '@aws-cdk/cdk'; -import { IAliasRecordTarget } from '../alias-record-target'; -import { IHostedZone } from '../hosted-zone-ref'; -import { CfnRecordSet } from '../route53.generated'; -import { determineFullyQualifiedDomainName } from './_util'; - -export interface AliasRecordProps { - /** - * The zone in which this alias should be defined. - */ - readonly zone: IHostedZone; - /** - * Name for the record. This can be the FQDN for the record (foo.example.com) or - * a subdomain of the parent hosted zone (foo, with example.com as the hosted zone). - */ - readonly recordName: string; - /** - * Target for the alias record - */ - readonly target: IAliasRecordTarget; -} - -/** - * An alias record - */ -export interface IAliasRecord extends IConstruct { - /** - * The domain name of the record - */ - readonly domainName: string; -} - -/** - * Define a new Route53 alias record - */ -export class AliasRecord extends Construct implements IAliasRecord { - public readonly domainName: string; - - constructor(scope: Construct, id: string, props: AliasRecordProps) { - super(scope, id); - - const record = new CfnRecordSet(this, 'Resource', { - hostedZoneId: props.zone.hostedZoneId, - name: determineFullyQualifiedDomainName(props.recordName, props.zone), - type: 'A', // ipv4 - aliasTarget: props.target.bind(this) - }); - - this.domainName = record.ref; - } -} \ No newline at end of file diff --git a/packages/@aws-cdk/aws-route53/lib/records/index.ts b/packages/@aws-cdk/aws-route53/lib/records/index.ts deleted file mode 100644 index 3ecd42b96de83..0000000000000 --- a/packages/@aws-cdk/aws-route53/lib/records/index.ts +++ /dev/null @@ -1,3 +0,0 @@ -export * from './alias'; -export * from './basic'; -export * from './zone-delegation'; diff --git a/packages/@aws-cdk/aws-route53/lib/records/zone-delegation.ts b/packages/@aws-cdk/aws-route53/lib/records/zone-delegation.ts deleted file mode 100644 index 175e1c921aceb..0000000000000 --- a/packages/@aws-cdk/aws-route53/lib/records/zone-delegation.ts +++ /dev/null @@ -1,44 +0,0 @@ -import cdk = require('@aws-cdk/cdk'); -import { ZoneDelegationOptions } from '../hosted-zone'; -import { IHostedZone } from '../hosted-zone-ref'; -import { CfnRecordSet } from '../route53.generated'; -import { determineFullyQualifiedDomainName } from './_util'; - -export interface ZoneDelegationRecordProps extends ZoneDelegationOptions { - /** - * The zone in which this delegate is defined. - */ - readonly zone: IHostedZone; - /** - * The name of the zone that delegation is made to. - */ - readonly delegatedZoneName: string; - - /** - * The name servers to report in the delegation records. - */ - readonly nameServers: string[]; -} - -/** - * A record to delegate further lookups to a different set of name servers - */ -export class ZoneDelegationRecord extends cdk.Construct { - constructor(scope: cdk.Construct, id: string, props: ZoneDelegationRecordProps) { - super(scope, id); - - const ttl = props.ttl === undefined ? 172_800 : props.ttl; - const resourceRecords = cdk.Token.isToken(props.nameServers) - ? props.nameServers // Can't map a string-array token! - : props.nameServers.map(ns => (cdk.Token.isToken(ns) || ns.endsWith('.')) ? ns : `${ns}.`); - - new CfnRecordSet(this, 'Resource', { - hostedZoneId: props.zone.hostedZoneId, - name: determineFullyQualifiedDomainName(props.delegatedZoneName, props.zone), - type: 'NS', - ttl: ttl.toString(), - comment: props.comment, - resourceRecords, - }); - } -} diff --git a/packages/@aws-cdk/aws-route53/lib/util.ts b/packages/@aws-cdk/aws-route53/lib/util.ts index 2483b74f3a4ec..358f39f2da7ce 100644 --- a/packages/@aws-cdk/aws-route53/lib/util.ts +++ b/packages/@aws-cdk/aws-route53/lib/util.ts @@ -1,3 +1,5 @@ +import { IHostedZone } from './hosted-zone-ref'; + /** * Validates a zone name is valid by Route53 specifc naming rules, * and that there is no trailing dot in the name. @@ -26,3 +28,31 @@ class ValidationError extends Error { super(message); } } + +/** + * Route53 requires the record names are specified as fully qualified names, but this + * forces lots of redundant work on the user (repeating the zone name over and over). + * This function allows the user to be lazier and offers a nicer experience, by + * qualifying relative names appropriately: + * + * @param providedName the user-specified name of the record. + * @param zoneName the fully-qualified name of the zone the record will be created in. + * + * @returns
    + *
  • If +providedName+ ends with a +.+, use it as-is
  • + *
  • If +providedName+ ends with or equals +zoneName+, append a trailing +.+
  • + *
  • Otherwise, append +.+, +zoneName+ and a trailing +.+
  • + *
+ */ +export function determineFullyQualifiedDomainName(providedName: string, hostedZone: IHostedZone): string { + if (providedName.endsWith('.')) { + return providedName; + } + + const suffix = `.${hostedZone.zoneName}`; + if (providedName.endsWith(suffix) || providedName === hostedZone.zoneName) { + return `${providedName}.`; + } + + return `${providedName}${suffix}.`; +} diff --git a/packages/@aws-cdk/aws-route53/test/integ.route53.ts b/packages/@aws-cdk/aws-route53/test/integ.route53.ts index cefbc108d3bb0..3b74f25b76d3f 100644 --- a/packages/@aws-cdk/aws-route53/test/integ.route53.ts +++ b/packages/@aws-cdk/aws-route53/test/integ.route53.ts @@ -1,6 +1,6 @@ import ec2 = require('@aws-cdk/aws-ec2'); import cdk = require('@aws-cdk/cdk'); -import { ARecord, CaaAmazonRecord, CnameRecord, PrivateHostedZone, PublicHostedZone, TxtRecord } from '../lib'; +import { ARecord, CaaAmazonRecord, CnameRecord, PrivateHostedZone, PublicHostedZone, RecordTarget, TxtRecord } from '../lib'; const app = new cdk.App(); @@ -39,10 +39,7 @@ new CnameRecord(stack, 'CNAME', { new ARecord(stack, 'A', { zone: privateZone, recordName: 'test', - ipAddresses: [ - '1.2.3.4', - '5.6.7.8' - ] + target: RecordTarget.fromIpAddresses('1.2.3.4', '5.6.7.8') }); new CaaAmazonRecord(stack, 'CaaAmazon', { diff --git a/packages/@aws-cdk/aws-route53/test/test.alias-record.ts b/packages/@aws-cdk/aws-route53/test/test.alias-record.ts deleted file mode 100644 index 412f0f271d17d..0000000000000 --- a/packages/@aws-cdk/aws-route53/test/test.alias-record.ts +++ /dev/null @@ -1,79 +0,0 @@ -import { expect, haveResource } from '@aws-cdk/assert'; -import { Stack } from '@aws-cdk/cdk'; -import { Test } from 'nodeunit'; -import { AliasRecord, IAliasRecordTarget, PublicHostedZone } from '../lib'; - -export = { - 'test alias record'(test: Test) { - // GIVEN - const stack = new Stack(); - const zone = new PublicHostedZone(stack, 'HostedZone', { zoneName: 'test.public' }); - - const target: IAliasRecordTarget = { - bind: () => { - return { - hostedZoneId: 'Z2P70J7EXAMPLE', - dnsName: 'foo.example.com' - }; - } - }; - - // WHEN - new AliasRecord(zone, 'Alias', { - zone, - recordName: '_foo', - target - }); - - // THEN - stack contains a record set - expect(stack).to(haveResource('AWS::Route53::RecordSet', { - Name: '_foo.test.public.', - HostedZoneId: { - Ref: 'HostedZoneDB99F866' - }, - Type: 'A', - AliasTarget: { - HostedZoneId: 'Z2P70J7EXAMPLE', - DNSName: 'foo.example.com', - } - })); - - test.done(); - }, - 'test alias record on zone root'(test: Test) { - // GIVEN - const stack = new Stack(); - const zone = new PublicHostedZone(stack, 'HostedZone', { zoneName: 'test.public' }); - - const target: IAliasRecordTarget = { - bind: () => { - return { - hostedZoneId: 'Z2P70J7EXAMPLE', - dnsName: 'foo.example.com' - }; - } - }; - - // WHEN - new AliasRecord(zone, 'Alias', { - zone, - recordName: 'test.public', - target - }); - - // THEN - stack contains a record set - expect(stack).to(haveResource('AWS::Route53::RecordSet', { - Name: 'test.public.', - HostedZoneId: { - Ref: 'HostedZoneDB99F866' - }, - Type: 'A', - AliasTarget: { - HostedZoneId: 'Z2P70J7EXAMPLE', - DNSName: 'foo.example.com', - } - })); - - test.done(); - } -}; diff --git a/packages/@aws-cdk/aws-route53/test/test.basic-record.ts b/packages/@aws-cdk/aws-route53/test/test.record-set.ts similarity index 71% rename from packages/@aws-cdk/aws-route53/test/test.basic-record.ts rename to packages/@aws-cdk/aws-route53/test/test.record-set.ts index 8d339013b61ec..7545228a2ba1e 100644 --- a/packages/@aws-cdk/aws-route53/test/test.basic-record.ts +++ b/packages/@aws-cdk/aws-route53/test/test.record-set.ts @@ -8,12 +8,12 @@ export = { // GIVEN const stack = new Stack(); - // WHEN const zone = new route53.HostedZone(stack, 'HostedZone', { zoneName: 'myzone' }); - new route53.BasicRecord(stack, 'Basic', { + // WHEN + new route53.RecordSet(stack, 'Basic', { zone, recordName: 'www', recordType: route53.RecordType.CNAME, @@ -39,12 +39,12 @@ export = { // GIVEN const stack = new Stack(); - // WHEN const zone = new route53.HostedZone(stack, 'HostedZone', { zoneName: 'myzone' }); - new route53.BasicRecord(stack, 'Basic', { + // WHEN + new route53.RecordSet(stack, 'Basic', { zone, recordName: 'aa', recordType: route53.RecordType.CNAME, @@ -71,12 +71,12 @@ export = { // GIVEN const stack = new Stack(); - // WHEN const zone = new route53.HostedZone(stack, 'HostedZone', { zoneName: 'myzone' }); - new route53.BasicRecord(stack, 'Basic', { + // WHEN + new route53.RecordSet(stack, 'Basic', { zone, recordType: route53.RecordType.A, recordValues: ['1.2.3.4'], @@ -96,22 +96,19 @@ export = { test.done(); }, - 'A record'(test: Test) { + 'A record with ip addresses'(test: Test) { // GIVEN const stack = new Stack(); - // WHEN const zone = new route53.HostedZone(stack, 'HostedZone', { zoneName: 'myzone' }); + // WHEN new route53.ARecord(stack, 'A', { zone, recordName: 'www', - ipAddresses: [ - '1.2.3.4', - '5.6.7.8' - ], + target: route53.RecordTarget.fromIpAddresses('1.2.3.4', '5.6.7.8'), }); // THEN @@ -130,21 +127,59 @@ export = { test.done(); }, - 'AAAA record'(test: Test) { + 'A record with alias'(test: Test) { // GIVEN const stack = new Stack(); + const zone = new route53.HostedZone(stack, 'HostedZone', { + zoneName: 'myzone' + }); + + const target: route53.IAliasRecordTarget = { + bind: () => { + return { + hostedZoneId: 'Z2P70J7EXAMPLE', + dnsName: 'foo.example.com' + }; + } + }; + // WHEN + new route53.ARecord(zone, 'Alias', { + zone, + recordName: '_foo', + target: route53.RecordTarget.fromAliasRecordTarget(target) + }); + + // THEN + expect(stack).to(haveResource('AWS::Route53::RecordSet', { + Name: '_foo.myzone.', + HostedZoneId: { + Ref: 'HostedZoneDB99F866' + }, + Type: 'A', + AliasTarget: { + HostedZoneId: 'Z2P70J7EXAMPLE', + DNSName: 'foo.example.com', + } + })); + + test.done(); + }, + + 'AAAA record with ip addresses'(test: Test) { + // GIVEN + const stack = new Stack(); + const zone = new route53.HostedZone(stack, 'HostedZone', { zoneName: 'myzone' }); + // WHEN new route53.AaaaRecord(stack, 'AAAA', { zone, recordName: 'www', - ipAddresses: [ - '2001:0db8:85a3:0000:0000:8a2e:0370:7334' - ], + target: route53.RecordTarget.fromIpAddresses('2001:0db8:85a3:0000:0000:8a2e:0370:7334'), }); // THEN @@ -162,15 +197,53 @@ export = { test.done(); }, - 'CNAME record'(test: Test) { + 'AAAA record with alias on zone root'(test: Test) { // GIVEN const stack = new Stack(); + const zone = new route53.HostedZone(stack, 'HostedZone', { + zoneName: 'myzone' + }); + + const target: route53.IAliasRecordTarget = { + bind: () => { + return { + hostedZoneId: 'Z2P70J7EXAMPLE', + dnsName: 'foo.example.com' + }; + } + }; // WHEN + new route53.AaaaRecord(zone, 'Alias', { + zone, + target: route53.RecordTarget.fromAliasRecordTarget(target) + }); + + // THEN + expect(stack).to(haveResource('AWS::Route53::RecordSet', { + Name: 'myzone.', + HostedZoneId: { + Ref: 'HostedZoneDB99F866' + }, + Type: 'AAAA', + AliasTarget: { + HostedZoneId: 'Z2P70J7EXAMPLE', + DNSName: 'foo.example.com', + } + })); + + test.done(); + }, + + 'CNAME record'(test: Test) { + // GIVEN + const stack = new Stack(); + const zone = new route53.HostedZone(stack, 'HostedZone', { zoneName: 'myzone' }); + // WHEN new route53.CnameRecord(stack, 'CNAME', { zone, recordName: 'www', @@ -196,11 +269,11 @@ export = { // GIVEN const stack = new Stack(); - // WHEN const zone = new route53.HostedZone(stack, 'HostedZone', { zoneName: 'myzone' }); + // WHEN new route53.TxtRecord(stack, 'TXT', { zone, recordName: 'www', @@ -226,11 +299,11 @@ export = { // GIVEN const stack = new Stack(); - // WHEN const zone = new route53.HostedZone(stack, 'HostedZone', { zoneName: 'myzone' }); + // WHEN new route53.SrvRecord(stack, 'SRV', { zone, recordName: 'www', @@ -261,11 +334,11 @@ export = { // GIVEN const stack = new Stack(); - // WHEN const zone = new route53.HostedZone(stack, 'HostedZone', { zoneName: 'myzone' }); + // WHEN new route53.CaaRecord(stack, 'CAA', { zone, recordName: 'www', @@ -295,11 +368,11 @@ export = { // GIVEN const stack = new Stack(); - // WHEN const zone = new route53.HostedZone(stack, 'HostedZone', { zoneName: 'myzone' }); + // WHEN new route53.CaaAmazonRecord(stack, 'CAAAmazon', { zone, recordName: 'www', // should have no effect @@ -324,11 +397,11 @@ export = { // GIVEN const stack = new Stack(); - // WHEN const zone = new route53.HostedZone(stack, 'HostedZone', { zoneName: 'myzone' }); + // WHEN new route53.MxRecord(stack, 'MX', { zone, recordName: 'mail', @@ -351,5 +424,35 @@ export = { TTL: "1800" })); test.done(); + }, + + 'Zone delegation record'(test: Test) { + // GIVEN + const stack = new Stack(); + + const zone = new route53.HostedZone(stack, 'HostedZone', { + zoneName: 'myzone' + }); + + // WHEN + new route53.ZoneDelegationRecord(stack, 'NS', { + zone, + recordName: 'foo', + nameServers: ['ns-1777.awsdns-30.co.uk'] + }); + + // THEN + expect(stack).to(haveResource('AWS::Route53::RecordSet', { + Name: "foo.myzone.", + Type: "NS", + HostedZoneId: { + Ref: "HostedZoneDB99F866" + }, + ResourceRecords: [ + "ns-1777.awsdns-30.co.uk." + ], + TTL: "172800" + })); + test.done(); } }; diff --git a/packages/@aws-cdk/aws-route53/test/test.zone-delegation-record.ts b/packages/@aws-cdk/aws-route53/test/test.zone-delegation-record.ts deleted file mode 100644 index 03d05b2161c74..0000000000000 --- a/packages/@aws-cdk/aws-route53/test/test.zone-delegation-record.ts +++ /dev/null @@ -1,54 +0,0 @@ -import { exactlyMatchTemplate, expect } from '@aws-cdk/assert'; -import { App, Stack } from '@aws-cdk/cdk'; -import { Test } from 'nodeunit'; -import { PublicHostedZone, ZoneDelegationRecord } from '../lib'; - -export = { - 'Zone Delegation records': { - NS(test: Test) { - const app = new TestApp(); - const zone = new PublicHostedZone(app.stack, 'HostedZone', { zoneName: 'test.public' }); - new ZoneDelegationRecord(zone, 'NS', { - zone, - delegatedZoneName: 'foo', - nameServers: ['ns-1777.awsdns-30.co.uk'] - }); - expect(app.stack).to(exactlyMatchTemplate({ - Resources: { - HostedZoneDB99F866: { - Type: 'AWS::Route53::HostedZone', - Properties: { - Name: 'test.public.' - } - }, - HostedZoneNS1BB87CC3: { - Type: 'AWS::Route53::RecordSet', - Properties: { - HostedZoneId: { - Ref: 'HostedZoneDB99F866' - }, - Name: 'foo.test.public.', - ResourceRecords: ['ns-1777.awsdns-30.co.uk.'], - Type: 'NS', - TTL: '172800' - } - } - } - })); - test.done(); - } - } -}; - -class TestApp { - public readonly stack: Stack; - private readonly app = new App(); - - constructor() { - const account = '123456789012'; - const region = 'bermuda-triangle'; - this.app.node.setContext(`availability-zones:${account}:${region}`, - [`${region}-1a`]); - this.stack = new Stack(this.app, 'MyStack', { env: { account, region } }); - } -} From 06589ac2fb017905794aecb0595d9999cfa75833 Mon Sep 17 00:00:00 2001 From: Jonathan Goldwasser Date: Wed, 5 Jun 2019 18:31:55 +0200 Subject: [PATCH 4/6] JSDoc and fromAlias --- .../lib/fargate/load-balanced-fargate-service.ts | 2 +- .../test/cloudfront-target.test.ts | 2 +- .../test/integ.alb-alias-target.ts | 2 +- .../test/integ.cloudfront-alias-target.ts | 2 +- .../test/load-balancer-target.test.ts | 2 +- packages/@aws-cdk/aws-route53/README.md | 2 +- packages/@aws-cdk/aws-route53/lib/record-set.ts | 12 +++++++++++- .../@aws-cdk/aws-route53/test/test.record-set.ts | 4 ++-- 8 files changed, 19 insertions(+), 9 deletions(-) diff --git a/packages/@aws-cdk/aws-ecs-patterns/lib/fargate/load-balanced-fargate-service.ts b/packages/@aws-cdk/aws-ecs-patterns/lib/fargate/load-balanced-fargate-service.ts index ff799522ca8d6..d9cd708b7ac4d 100644 --- a/packages/@aws-cdk/aws-ecs-patterns/lib/fargate/load-balanced-fargate-service.ts +++ b/packages/@aws-cdk/aws-ecs-patterns/lib/fargate/load-balanced-fargate-service.ts @@ -123,7 +123,7 @@ export class LoadBalancedFargateService extends LoadBalancedServiceBase { new ARecord(this, "DNS", { zone: props.domainZone, recordName: props.domainName, - target: RecordTarget.fromAliasRecordTarget(new targets.LoadBalancerTarget(this.loadBalancer)), + target: RecordTarget.fromAlias(new targets.LoadBalancerTarget(this.loadBalancer)), }); } } diff --git a/packages/@aws-cdk/aws-route53-targets/test/cloudfront-target.test.ts b/packages/@aws-cdk/aws-route53-targets/test/cloudfront-target.test.ts index 735dd145a869d..424f73944f65a 100644 --- a/packages/@aws-cdk/aws-route53-targets/test/cloudfront-target.test.ts +++ b/packages/@aws-cdk/aws-route53-targets/test/cloudfront-target.test.ts @@ -27,7 +27,7 @@ test('use CloudFront as record target', () => { new route53.ARecord(zone, 'Alias', { zone, recordName: '_foo', - target: route53.RecordTarget.fromAliasRecordTarget(new targets.CloudFrontTarget(distribution)) + target: route53.RecordTarget.fromAlias(new targets.CloudFrontTarget(distribution)) }); // THEN diff --git a/packages/@aws-cdk/aws-route53-targets/test/integ.alb-alias-target.ts b/packages/@aws-cdk/aws-route53-targets/test/integ.alb-alias-target.ts index 1ec8314b12c40..21aac1716fbfd 100644 --- a/packages/@aws-cdk/aws-route53-targets/test/integ.alb-alias-target.ts +++ b/packages/@aws-cdk/aws-route53-targets/test/integ.alb-alias-target.ts @@ -22,7 +22,7 @@ const zone = new route53.PublicHostedZone(stack, 'HostedZone', { zoneName: 'test new route53.ARecord(zone, 'Alias', { zone, recordName: '_foo', - target: route53.RecordTarget.fromAliasRecordTarget(new targets.LoadBalancerTarget(lb)) + target: route53.RecordTarget.fromAlias(new targets.LoadBalancerTarget(lb)) }); app.run(); diff --git a/packages/@aws-cdk/aws-route53-targets/test/integ.cloudfront-alias-target.ts b/packages/@aws-cdk/aws-route53-targets/test/integ.cloudfront-alias-target.ts index 8f3695b85ee8c..61b89da2a27a8 100644 --- a/packages/@aws-cdk/aws-route53-targets/test/integ.cloudfront-alias-target.ts +++ b/packages/@aws-cdk/aws-route53-targets/test/integ.cloudfront-alias-target.ts @@ -28,7 +28,7 @@ const distribution = new cloudfront.CloudFrontWebDistribution(stack, 'MyDistribu new route53.ARecord(zone, 'Alias', { zone, recordName: '_foo', - target: route53.RecordTarget.fromAliasRecordTarget(new targets.CloudFrontTarget(distribution)) + target: route53.RecordTarget.fromAlias(new targets.CloudFrontTarget(distribution)) }); app.run(); diff --git a/packages/@aws-cdk/aws-route53-targets/test/load-balancer-target.test.ts b/packages/@aws-cdk/aws-route53-targets/test/load-balancer-target.test.ts index df0f3879f4a9b..35756f0bdaf0b 100644 --- a/packages/@aws-cdk/aws-route53-targets/test/load-balancer-target.test.ts +++ b/packages/@aws-cdk/aws-route53-targets/test/load-balancer-target.test.ts @@ -22,7 +22,7 @@ test('use ALB as record target', () => { new route53.ARecord(zone, 'Alias', { zone, recordName: '_foo', - target: route53.RecordTarget.fromAliasRecordTarget(new targets.LoadBalancerTarget(lb)) + target: route53.RecordTarget.fromAlias(new targets.LoadBalancerTarget(lb)) }); // THEN diff --git a/packages/@aws-cdk/aws-route53/README.md b/packages/@aws-cdk/aws-route53/README.md index 11709035d0ceb..3ecbfd351fa1f 100644 --- a/packages/@aws-cdk/aws-route53/README.md +++ b/packages/@aws-cdk/aws-route53/README.md @@ -65,7 +65,7 @@ import targets = require('@aws-cdk/aws-route53-targets'); new route53.AaaaRecord(this, 'Alias', { zone: myZone, - target: route53.RecordTarget.fromAliasRecordTarget(new targets.CloudFrontTarget(distribution)) + target: route53.RecordTarget.fromAlias(new targets.CloudFrontTarget(distribution)) }) ``` diff --git a/packages/@aws-cdk/aws-route53/lib/record-set.ts b/packages/@aws-cdk/aws-route53/lib/record-set.ts index c38a9ad54cabf..c2c71bfd924d1 100644 --- a/packages/@aws-cdk/aws-route53/lib/record-set.ts +++ b/packages/@aws-cdk/aws-route53/lib/record-set.ts @@ -118,14 +118,24 @@ export class RecordSet extends Resource implements IRecordSet { } } +/** + * Type union for A and AAAA record that accepts multiple types of target. + */ export class RecordTarget { + /** + * Use ip addresses as target. + */ public static fromIpAddresses(...ipAddresses: string[]) { return new RecordTarget(ipAddresses); } - public static fromAliasRecordTarget(aliasTarget: IAliasRecordTarget) { + /** + * Use an alias as target. + */ + public static fromAlias(aliasTarget: IAliasRecordTarget) { return new RecordTarget(undefined, aliasTarget); } + private constructor(public readonly values?: string[], public readonly aliasTarget?: IAliasRecordTarget) { } } diff --git a/packages/@aws-cdk/aws-route53/test/test.record-set.ts b/packages/@aws-cdk/aws-route53/test/test.record-set.ts index 7545228a2ba1e..c262cb9a6916b 100644 --- a/packages/@aws-cdk/aws-route53/test/test.record-set.ts +++ b/packages/@aws-cdk/aws-route53/test/test.record-set.ts @@ -148,7 +148,7 @@ export = { new route53.ARecord(zone, 'Alias', { zone, recordName: '_foo', - target: route53.RecordTarget.fromAliasRecordTarget(target) + target: route53.RecordTarget.fromAlias(target) }); // THEN @@ -216,7 +216,7 @@ export = { // WHEN new route53.AaaaRecord(zone, 'Alias', { zone, - target: route53.RecordTarget.fromAliasRecordTarget(target) + target: route53.RecordTarget.fromAlias(target) }); // THEN From 705bdc7b8a2b1a806b8623bbe73301af6f60c594 Mon Sep 17 00:00:00 2001 From: Jonathan Goldwasser Date: Thu, 6 Jun 2019 19:50:10 +0200 Subject: [PATCH 5/6] move type union to RecordSet --- .../fargate/load-balanced-fargate-service.ts | 4 +- .../test/cloudfront-target.test.ts | 2 +- .../test/integ.alb-alias-target.ts | 2 +- .../test/integ.cloudfront-alias-target.ts | 2 +- .../test/load-balancer-target.test.ts | 2 +- packages/@aws-cdk/aws-route53/README.md | 4 +- .../@aws-cdk/aws-route53/lib/record-set.ts | 89 +++++++++---------- .../aws-route53/test/integ.route53.ts | 4 +- .../aws-route53/test/test.record-set.ts | 10 +-- 9 files changed, 58 insertions(+), 61 deletions(-) diff --git a/packages/@aws-cdk/aws-ecs-patterns/lib/fargate/load-balanced-fargate-service.ts b/packages/@aws-cdk/aws-ecs-patterns/lib/fargate/load-balanced-fargate-service.ts index d9cd708b7ac4d..6129c57a2f7b9 100644 --- a/packages/@aws-cdk/aws-ecs-patterns/lib/fargate/load-balanced-fargate-service.ts +++ b/packages/@aws-cdk/aws-ecs-patterns/lib/fargate/load-balanced-fargate-service.ts @@ -1,5 +1,5 @@ import ecs = require('@aws-cdk/aws-ecs'); -import { ARecord, IHostedZone, RecordTarget } from '@aws-cdk/aws-route53'; +import { AddressRecordTarget, ARecord, IHostedZone } from '@aws-cdk/aws-route53'; import targets = require('@aws-cdk/aws-route53-targets'); import cdk = require('@aws-cdk/cdk'); import { LoadBalancedServiceBase, LoadBalancedServiceBaseProps } from '../base/load-balanced-service-base'; @@ -123,7 +123,7 @@ export class LoadBalancedFargateService extends LoadBalancedServiceBase { new ARecord(this, "DNS", { zone: props.domainZone, recordName: props.domainName, - target: RecordTarget.fromAlias(new targets.LoadBalancerTarget(this.loadBalancer)), + target: AddressRecordTarget.fromAlias(new targets.LoadBalancerTarget(this.loadBalancer)), }); } } diff --git a/packages/@aws-cdk/aws-route53-targets/test/cloudfront-target.test.ts b/packages/@aws-cdk/aws-route53-targets/test/cloudfront-target.test.ts index 424f73944f65a..cb7b33b1d4ced 100644 --- a/packages/@aws-cdk/aws-route53-targets/test/cloudfront-target.test.ts +++ b/packages/@aws-cdk/aws-route53-targets/test/cloudfront-target.test.ts @@ -27,7 +27,7 @@ test('use CloudFront as record target', () => { new route53.ARecord(zone, 'Alias', { zone, recordName: '_foo', - target: route53.RecordTarget.fromAlias(new targets.CloudFrontTarget(distribution)) + target: route53.AddressRecordTarget.fromAlias(new targets.CloudFrontTarget(distribution)) }); // THEN diff --git a/packages/@aws-cdk/aws-route53-targets/test/integ.alb-alias-target.ts b/packages/@aws-cdk/aws-route53-targets/test/integ.alb-alias-target.ts index 21aac1716fbfd..ceea3cfea60e9 100644 --- a/packages/@aws-cdk/aws-route53-targets/test/integ.alb-alias-target.ts +++ b/packages/@aws-cdk/aws-route53-targets/test/integ.alb-alias-target.ts @@ -22,7 +22,7 @@ const zone = new route53.PublicHostedZone(stack, 'HostedZone', { zoneName: 'test new route53.ARecord(zone, 'Alias', { zone, recordName: '_foo', - target: route53.RecordTarget.fromAlias(new targets.LoadBalancerTarget(lb)) + target: route53.AddressRecordTarget.fromAlias(new targets.LoadBalancerTarget(lb)) }); app.run(); diff --git a/packages/@aws-cdk/aws-route53-targets/test/integ.cloudfront-alias-target.ts b/packages/@aws-cdk/aws-route53-targets/test/integ.cloudfront-alias-target.ts index 61b89da2a27a8..0adb96e563ad4 100644 --- a/packages/@aws-cdk/aws-route53-targets/test/integ.cloudfront-alias-target.ts +++ b/packages/@aws-cdk/aws-route53-targets/test/integ.cloudfront-alias-target.ts @@ -28,7 +28,7 @@ const distribution = new cloudfront.CloudFrontWebDistribution(stack, 'MyDistribu new route53.ARecord(zone, 'Alias', { zone, recordName: '_foo', - target: route53.RecordTarget.fromAlias(new targets.CloudFrontTarget(distribution)) + target: route53.AddressRecordTarget.fromAlias(new targets.CloudFrontTarget(distribution)) }); app.run(); diff --git a/packages/@aws-cdk/aws-route53-targets/test/load-balancer-target.test.ts b/packages/@aws-cdk/aws-route53-targets/test/load-balancer-target.test.ts index 35756f0bdaf0b..c78062120398d 100644 --- a/packages/@aws-cdk/aws-route53-targets/test/load-balancer-target.test.ts +++ b/packages/@aws-cdk/aws-route53-targets/test/load-balancer-target.test.ts @@ -22,7 +22,7 @@ test('use ALB as record target', () => { new route53.ARecord(zone, 'Alias', { zone, recordName: '_foo', - target: route53.RecordTarget.fromAlias(new targets.LoadBalancerTarget(lb)) + target: route53.AddressRecordTarget.fromAlias(new targets.LoadBalancerTarget(lb)) }); // THEN diff --git a/packages/@aws-cdk/aws-route53/README.md b/packages/@aws-cdk/aws-route53/README.md index 3ecbfd351fa1f..39077177f5cb6 100644 --- a/packages/@aws-cdk/aws-route53/README.md +++ b/packages/@aws-cdk/aws-route53/README.md @@ -54,7 +54,7 @@ import route53 = require('@aws-cdk/aws-route53'); new route53.AaaaRecord(this, 'AaaaRecord', { zone: myZone, - target: route53.RecordTarget.fromIpAddresses('1.2.3.4', '5.6.7.8') + target: route53.AddressRecordTarget.fromIpAddresses('1.2.3.4', '5.6.7.8') }) ``` @@ -65,7 +65,7 @@ import targets = require('@aws-cdk/aws-route53-targets'); new route53.AaaaRecord(this, 'Alias', { zone: myZone, - target: route53.RecordTarget.fromAlias(new targets.CloudFrontTarget(distribution)) + target: route53.AddressRecordTarget.fromAlias(new targets.CloudFrontTarget(distribution)) }) ``` diff --git a/packages/@aws-cdk/aws-route53/lib/record-set.ts b/packages/@aws-cdk/aws-route53/lib/record-set.ts index c2c71bfd924d1..929ef4ec28192 100644 --- a/packages/@aws-cdk/aws-route53/lib/record-set.ts +++ b/packages/@aws-cdk/aws-route53/lib/record-set.ts @@ -63,6 +63,28 @@ export interface RecordSetOptions { readonly comment?: string; } +/** + * Type union for a record that accepts multiple types of target. + */ +export class RecordTarget { + /** + * Use string values as target. + */ + public static fromValues(...values: string[]) { + return new RecordTarget(values); + } + + /** + * Use an alias as target. + */ + public static fromAlias(aliasTarget: IAliasRecordTarget) { + return new RecordTarget(undefined, aliasTarget); + } + + protected constructor(public readonly values?: string[], public readonly aliasTarget?: IAliasRecordTarget) { + } +} + /** * Construction properties for a RecordSet. */ @@ -73,20 +95,10 @@ export interface RecordSetProps extends RecordSetOptions { readonly recordType: RecordType; /** - * The values for this record. Either `recordValues` or `target` must be - * specified. - * - * @default no values - */ - readonly recordValues?: string[]; - - /** - * The target for an alias record. Either `target` or `recordValues` must be - * specified. - * - * @default no target + * The target for this record, either `RecordTarget.fromValues()` or + * `RecordTarget.fromAlias()`. */ - readonly target?: IAliasRecordTarget; + readonly target: RecordTarget; } /** @@ -98,18 +110,14 @@ export class RecordSet extends Resource implements IRecordSet { constructor(scope: Construct, id: string, props: RecordSetProps) { super(scope, id); - if (!(props.recordValues || props.target) || (props.recordValues && props.target)) { - throw new Error('Either `recordValues` or `target` must be specified.'); - } - - const ttl = props.target ? undefined : (props.ttl || 1800).toString(); + const ttl = props.target.aliasTarget ? undefined : (props.ttl || 1800).toString(); const recordSet = new CfnRecordSet(this, 'Resource', { hostedZoneId: props.zone.hostedZoneId, name: determineFullyQualifiedDomainName(props.recordName || props.zone.zoneName, props.zone), type: props.recordType, - resourceRecords: props.recordValues, - aliasTarget: props.target && props.target.bind(this), + resourceRecords: props.target.values, + aliasTarget: props.target.aliasTarget && props.target.aliasTarget.bind(this), ttl, comment: props.comment }); @@ -119,24 +127,14 @@ export class RecordSet extends Resource implements IRecordSet { } /** - * Type union for A and AAAA record that accepts multiple types of target. + * */ -export class RecordTarget { +export class AddressRecordTarget extends RecordTarget { /** - * Use ip addresses as target. + * Use ip adresses as target. */ public static fromIpAddresses(...ipAddresses: string[]) { - return new RecordTarget(ipAddresses); - } - - /** - * Use an alias as target. - */ - public static fromAlias(aliasTarget: IAliasRecordTarget) { - return new RecordTarget(undefined, aliasTarget); - } - - private constructor(public readonly values?: string[], public readonly aliasTarget?: IAliasRecordTarget) { + return RecordTarget.fromValues(...ipAddresses); } } @@ -147,7 +145,7 @@ export interface ARecordProps extends RecordSetOptions { /** * The target. */ - readonly target: RecordTarget; + readonly target: AddressRecordTarget; } /** @@ -160,8 +158,7 @@ export class ARecord extends RecordSet { super(scope, id, { ...props, recordType: RecordType.A, - recordValues: props.target.values, - target: props.target.aliasTarget, + target: props.target, }); } } @@ -181,8 +178,7 @@ export class AaaaRecord extends RecordSet { super(scope, id, { ...props, recordType: RecordType.AAAA, - recordValues: props.target.values, - target: props.target.aliasTarget, + target: props.target, }); } } @@ -207,7 +203,7 @@ export class CnameRecord extends RecordSet { super(scope, id, { ...props, recordType: RecordType.CNAME, - recordValues: [props.domainName] + target: RecordTarget.fromValues(props.domainName) }); } } @@ -232,7 +228,7 @@ export class TxtRecord extends RecordSet { super(scope, id, { ...props, recordType: RecordType.TXT, - recordValues: props.values && props.values.map(v => JSON.stringify(v)), + target: RecordTarget.fromValues(...props.values.map(v => JSON.stringify(v))), }); } } @@ -281,7 +277,7 @@ export class SrvRecord extends RecordSet { super(scope, id, { ...props, recordType: RecordType.SRV, - recordValues: props.values.map(v => `${v.priority} ${v.weight} ${v.port} ${v.hostName}`), + target: RecordTarget.fromValues(...props.values.map(v => `${v.priority} ${v.weight} ${v.port} ${v.hostName}`)), }); } } @@ -349,7 +345,7 @@ export class CaaRecord extends RecordSet { super(scope, id, { ...props, recordType: RecordType.CAA, - recordValues: props.values.map(v => `${v.flag} ${v.tag} "${v.value}"`), + target: RecordTarget.fromValues(...props.values.map(v => `${v.flag} ${v.tag} "${v.value}"`)), }); } } @@ -418,7 +414,7 @@ export class MxRecord extends RecordSet { super(scope, id, { ...props, recordType: RecordType.MX, - recordValues: props.values.map(v => `${v.priority} ${v.hostName}`) + target: RecordTarget.fromValues(...props.values.map(v => `${v.priority} ${v.hostName}`)) }); } } @@ -441,9 +437,10 @@ export class ZoneDelegationRecord extends RecordSet { super(scope, id, { ...props, recordType: RecordType.NS, - recordValues: Token.isToken(props.nameServers) + target: RecordTarget.fromValues(...Token.isToken(props.nameServers) ? props.nameServers // Can't map a string-array token! - : props.nameServers.map(ns => (Token.isToken(ns) || ns.endsWith('.')) ? ns : `${ns}.`), + : props.nameServers.map(ns => (Token.isToken(ns) || ns.endsWith('.')) ? ns : `${ns}.`) + ), ttl: props.ttl || 172_800 }); } diff --git a/packages/@aws-cdk/aws-route53/test/integ.route53.ts b/packages/@aws-cdk/aws-route53/test/integ.route53.ts index 3b74f25b76d3f..435e660c80219 100644 --- a/packages/@aws-cdk/aws-route53/test/integ.route53.ts +++ b/packages/@aws-cdk/aws-route53/test/integ.route53.ts @@ -1,6 +1,6 @@ import ec2 = require('@aws-cdk/aws-ec2'); import cdk = require('@aws-cdk/cdk'); -import { ARecord, CaaAmazonRecord, CnameRecord, PrivateHostedZone, PublicHostedZone, RecordTarget, TxtRecord } from '../lib'; +import { AddressRecordTarget, ARecord, CaaAmazonRecord, CnameRecord, PrivateHostedZone, PublicHostedZone, TxtRecord } from '../lib'; const app = new cdk.App(); @@ -39,7 +39,7 @@ new CnameRecord(stack, 'CNAME', { new ARecord(stack, 'A', { zone: privateZone, recordName: 'test', - target: RecordTarget.fromIpAddresses('1.2.3.4', '5.6.7.8') + target: AddressRecordTarget.fromIpAddresses('1.2.3.4', '5.6.7.8') }); new CaaAmazonRecord(stack, 'CaaAmazon', { diff --git a/packages/@aws-cdk/aws-route53/test/test.record-set.ts b/packages/@aws-cdk/aws-route53/test/test.record-set.ts index c262cb9a6916b..8dca376245ab5 100644 --- a/packages/@aws-cdk/aws-route53/test/test.record-set.ts +++ b/packages/@aws-cdk/aws-route53/test/test.record-set.ts @@ -17,7 +17,7 @@ export = { zone, recordName: 'www', recordType: route53.RecordType.CNAME, - recordValues: ['zzz'] + target: route53.RecordTarget.fromValues('zzz') }); // THEN @@ -48,7 +48,7 @@ export = { zone, recordName: 'aa', recordType: route53.RecordType.CNAME, - recordValues: ['bbb'], + target: route53.RecordTarget.fromValues('bbb'), ttl: 6077 }); @@ -79,7 +79,7 @@ export = { new route53.RecordSet(stack, 'Basic', { zone, recordType: route53.RecordType.A, - recordValues: ['1.2.3.4'], + target: route53.RecordTarget.fromValues('1.2.3.4'), }); // THEN @@ -108,7 +108,7 @@ export = { new route53.ARecord(stack, 'A', { zone, recordName: 'www', - target: route53.RecordTarget.fromIpAddresses('1.2.3.4', '5.6.7.8'), + target: route53.AddressRecordTarget.fromIpAddresses('1.2.3.4', '5.6.7.8'), }); // THEN @@ -179,7 +179,7 @@ export = { new route53.AaaaRecord(stack, 'AAAA', { zone, recordName: 'www', - target: route53.RecordTarget.fromIpAddresses('2001:0db8:85a3:0000:0000:8a2e:0370:7334'), + target: route53.AddressRecordTarget.fromIpAddresses('2001:0db8:85a3:0000:0000:8a2e:0370:7334'), }); // THEN From 49020c8ffaf83e85f0c98b2b98e4ffe12566d250 Mon Sep 17 00:00:00 2001 From: Jonathan Goldwasser Date: Thu, 6 Jun 2019 23:33:19 +0200 Subject: [PATCH 6/6] README --- packages/@aws-cdk/aws-route53/README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/@aws-cdk/aws-route53/README.md b/packages/@aws-cdk/aws-route53/README.md index 39077177f5cb6..6d181e441799d 100644 --- a/packages/@aws-cdk/aws-route53/README.md +++ b/packages/@aws-cdk/aws-route53/README.md @@ -52,7 +52,7 @@ To add a A record to your zone: ```ts import route53 = require('@aws-cdk/aws-route53'); -new route53.AaaaRecord(this, 'AaaaRecord', { +new route53.ARecord(this, 'ARecord', { zone: myZone, target: route53.AddressRecordTarget.fromIpAddresses('1.2.3.4', '5.6.7.8') })