2021-10-04 13:47:40 +00:00
|
|
|
import json
|
|
|
|
from copy import deepcopy
|
2023-11-30 15:55:51 +00:00
|
|
|
|
|
|
|
import boto3
|
|
|
|
|
2021-10-04 13:47:40 +00:00
|
|
|
from moto import mock_cloudformation, mock_ec2, mock_route53
|
2023-11-30 15:55:51 +00:00
|
|
|
from tests.test_cloudformation.fixtures import (
|
|
|
|
route53_ec2_instance_with_public_ip,
|
|
|
|
route53_health_check,
|
|
|
|
route53_roundrobin,
|
|
|
|
)
|
2021-10-04 13:47:40 +00:00
|
|
|
|
|
|
|
template_hosted_zone = {
|
|
|
|
"AWSTemplateFormatVersion": "2010-09-09",
|
|
|
|
"Description": "Stack 1",
|
|
|
|
"Parameters": {},
|
|
|
|
"Resources": {
|
|
|
|
"Bar": {
|
|
|
|
"Type": "AWS::Route53::HostedZone",
|
|
|
|
"Properties": {"Name": "foo.bar.baz"},
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
template_record_set = {
|
|
|
|
"AWSTemplateFormatVersion": "2010-09-09",
|
|
|
|
"Description": "Stack 2",
|
|
|
|
"Parameters": {"ZoneId": {"Type": "String"}},
|
|
|
|
"Resources": {
|
|
|
|
"Foo": {
|
|
|
|
"Properties": {
|
|
|
|
"HostedZoneId": {"Ref": "ZoneId"},
|
|
|
|
"RecordSets": [
|
|
|
|
{
|
|
|
|
"Name": "test.vpc.internal",
|
|
|
|
"Type": "A",
|
|
|
|
"SetIdentifier": "test1",
|
|
|
|
"Weight": 50,
|
|
|
|
}
|
|
|
|
],
|
|
|
|
},
|
|
|
|
"Type": "AWS::Route53::RecordSetGroup",
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@mock_cloudformation
|
|
|
|
@mock_route53
|
|
|
|
def test_create_stack_hosted_zone_by_id():
|
|
|
|
cf_conn = boto3.client("cloudformation", region_name="us-east-1")
|
|
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
|
|
|
|
|
|
# when creating a hosted zone via CF
|
|
|
|
cf_conn.create_stack(
|
|
|
|
StackName="test_stack1", TemplateBody=json.dumps(template_hosted_zone)
|
|
|
|
)
|
|
|
|
|
|
|
|
# then a hosted zone should exist
|
|
|
|
zone = conn.list_hosted_zones()["HostedZones"][0]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert zone["Name"] == "foo.bar.baz"
|
|
|
|
assert zone["ResourceRecordSetCount"] == 2
|
2021-10-04 13:47:40 +00:00
|
|
|
|
|
|
|
# when adding a record set to this zone
|
|
|
|
cf_conn.create_stack(
|
|
|
|
StackName="test_stack2",
|
|
|
|
TemplateBody=json.dumps(template_record_set),
|
|
|
|
Parameters=[{"ParameterKey": "ZoneId", "ParameterValue": zone["Id"]}],
|
|
|
|
)
|
|
|
|
|
|
|
|
# then the hosted zone should have a record
|
|
|
|
updated_zone = conn.list_hosted_zones()["HostedZones"][0]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert updated_zone["Id"] == zone["Id"]
|
|
|
|
assert updated_zone["Name"] == "foo.bar.baz"
|
|
|
|
assert updated_zone["ResourceRecordSetCount"] == 3
|
2021-10-04 13:47:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_cloudformation
|
|
|
|
@mock_route53
|
|
|
|
def test_route53_roundrobin():
|
|
|
|
cf = boto3.client("cloudformation", region_name="us-west-1")
|
|
|
|
route53 = boto3.client("route53", region_name="us-west-1")
|
|
|
|
|
|
|
|
template_json = json.dumps(route53_roundrobin.template)
|
|
|
|
cf.create_stack(StackName="test_stack", TemplateBody=template_json)
|
|
|
|
|
|
|
|
zones = route53.list_hosted_zones()["HostedZones"]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert len(zones) == 1
|
2021-10-04 13:47:40 +00:00
|
|
|
zone_id = zones[0]["Id"].split("/")[2]
|
|
|
|
|
|
|
|
rrsets = route53.list_resource_record_sets(HostedZoneId=zone_id)[
|
|
|
|
"ResourceRecordSets"
|
|
|
|
]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert len(rrsets) == 4
|
2022-12-14 23:58:55 +00:00
|
|
|
record_set1 = rrsets[2]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert record_set1["Name"] == "test_stack.us-west-1.my_zone."
|
|
|
|
assert record_set1["SetIdentifier"] == "test_stack AWS"
|
|
|
|
assert record_set1["Type"] == "CNAME"
|
|
|
|
assert record_set1["TTL"] == 900
|
|
|
|
assert record_set1["Weight"] == 3
|
|
|
|
assert record_set1["ResourceRecords"][0]["Value"] == "aws.amazon.com"
|
2021-10-04 13:47:40 +00:00
|
|
|
|
2022-12-14 23:58:55 +00:00
|
|
|
record_set2 = rrsets[3]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert record_set2["Name"] == "test_stack.us-west-1.my_zone."
|
|
|
|
assert record_set2["SetIdentifier"] == "test_stack Amazon"
|
|
|
|
assert record_set2["Type"] == "CNAME"
|
|
|
|
assert record_set2["TTL"] == 900
|
|
|
|
assert record_set2["Weight"] == 1
|
|
|
|
assert record_set2["ResourceRecords"][0]["Value"] == "www.amazon.com"
|
2021-10-04 13:47:40 +00:00
|
|
|
|
|
|
|
stack = cf.describe_stacks(StackName="test_stack")["Stacks"][0]
|
|
|
|
output = stack["Outputs"][0]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert output["OutputKey"] == "DomainName"
|
|
|
|
assert output["OutputValue"] == f"arn:aws:route53:::hostedzone/{zone_id}"
|
2021-10-04 13:47:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_cloudformation
|
|
|
|
@mock_ec2
|
|
|
|
@mock_route53
|
|
|
|
def test_route53_ec2_instance_with_public_ip():
|
|
|
|
route53 = boto3.client("route53", region_name="us-west-1")
|
|
|
|
ec2 = boto3.client("ec2", region_name="us-west-1")
|
|
|
|
|
|
|
|
template_json = json.dumps(route53_ec2_instance_with_public_ip.template)
|
|
|
|
cf = boto3.client("cloudformation", region_name="us-west-1")
|
|
|
|
cf.create_stack(StackName="test_stack", TemplateBody=template_json)
|
|
|
|
|
|
|
|
instance_id = ec2.describe_instances()["Reservations"][0]["Instances"][0][
|
|
|
|
"InstanceId"
|
|
|
|
]
|
|
|
|
|
|
|
|
zones = route53.list_hosted_zones()["HostedZones"]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert len(zones) == 1
|
2021-10-04 13:47:40 +00:00
|
|
|
zone_id = zones[0]["Id"].split("/")[2]
|
|
|
|
|
|
|
|
rrsets = route53.list_resource_record_sets(HostedZoneId=zone_id)[
|
|
|
|
"ResourceRecordSets"
|
|
|
|
]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert len(rrsets) == 3
|
2021-10-04 13:47:40 +00:00
|
|
|
|
2022-12-14 23:58:55 +00:00
|
|
|
record_set = rrsets[2]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert record_set["Name"] == f"{instance_id}.us-west-1.my_zone."
|
|
|
|
assert record_set["Type"] == "A"
|
|
|
|
assert record_set["TTL"] == 900
|
|
|
|
assert record_set["ResourceRecords"][0]["Value"] == "10.0.0.25"
|
|
|
|
assert "SetIdentifier" not in record_set
|
|
|
|
assert "Weight" not in record_set
|
2021-10-04 13:47:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_cloudformation
|
|
|
|
@mock_route53
|
|
|
|
def test_route53_associate_health_check():
|
|
|
|
route53 = boto3.client("route53", region_name="us-west-1")
|
|
|
|
|
|
|
|
template_json = json.dumps(route53_health_check.template)
|
|
|
|
cf = boto3.client("cloudformation", region_name="us-west-1")
|
|
|
|
cf.create_stack(StackName="test_stack", TemplateBody=template_json)
|
|
|
|
|
|
|
|
checks = route53.list_health_checks()["HealthChecks"]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert len(checks) == 1
|
2021-10-04 13:47:40 +00:00
|
|
|
check = checks[0]
|
|
|
|
health_check_id = check["Id"]
|
|
|
|
config = check["HealthCheckConfig"]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert config["FailureThreshold"] == 3
|
|
|
|
assert config["IPAddress"] == "10.0.0.4"
|
|
|
|
assert config["Port"] == 80
|
|
|
|
assert config["RequestInterval"] == 10
|
|
|
|
assert config["ResourcePath"] == "/"
|
|
|
|
assert config["Type"] == "HTTP"
|
2021-10-04 13:47:40 +00:00
|
|
|
|
|
|
|
zones = route53.list_hosted_zones()["HostedZones"]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert len(zones) == 1
|
2021-10-04 13:47:40 +00:00
|
|
|
zone_id = zones[0]["Id"].split("/")[2]
|
|
|
|
|
|
|
|
rrsets = route53.list_resource_record_sets(HostedZoneId=zone_id)[
|
|
|
|
"ResourceRecordSets"
|
|
|
|
]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert len(rrsets) == 3
|
2021-10-04 13:47:40 +00:00
|
|
|
record_set = rrsets[0]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert record_set["HealthCheckId"] == health_check_id
|
2021-10-04 13:47:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_cloudformation
|
|
|
|
@mock_route53
|
|
|
|
def test_route53_with_update():
|
|
|
|
route53 = boto3.client("route53", region_name="us-west-1")
|
|
|
|
|
|
|
|
template_json = json.dumps(route53_health_check.template)
|
|
|
|
cf = boto3.client("cloudformation", region_name="us-west-1")
|
|
|
|
cf.create_stack(StackName="test_stack", TemplateBody=template_json)
|
|
|
|
|
|
|
|
zones = route53.list_hosted_zones()["HostedZones"]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert len(zones) == 1
|
2021-10-04 13:47:40 +00:00
|
|
|
zone_id = zones[0]["Id"]
|
|
|
|
zone_id = zone_id.split("/")
|
|
|
|
zone_id = zone_id[2]
|
|
|
|
|
|
|
|
rrsets = route53.list_resource_record_sets(HostedZoneId=zone_id)[
|
|
|
|
"ResourceRecordSets"
|
|
|
|
]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert len(rrsets) == 3
|
2021-10-04 13:47:40 +00:00
|
|
|
|
|
|
|
record_set = rrsets[0]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert record_set["ResourceRecords"][0]["Value"] == "my.example.com"
|
2021-10-04 13:47:40 +00:00
|
|
|
|
2022-06-10 19:33:17 +00:00
|
|
|
# # given
|
2021-10-04 13:47:40 +00:00
|
|
|
template = deepcopy(route53_health_check.template)
|
|
|
|
template["Resources"]["myDNSRecord"]["Properties"]["ResourceRecords"] = [
|
|
|
|
"my_other.example.com"
|
|
|
|
]
|
|
|
|
template_json = json.dumps(template)
|
|
|
|
|
2022-06-10 19:33:17 +00:00
|
|
|
# # when
|
2021-10-04 13:47:40 +00:00
|
|
|
cf.update_stack(StackName="test_stack", TemplateBody=template_json)
|
|
|
|
|
2022-06-10 19:33:17 +00:00
|
|
|
# # then
|
2021-10-04 13:47:40 +00:00
|
|
|
zones = route53.list_hosted_zones()["HostedZones"]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert len(zones) == 1
|
2021-10-04 13:47:40 +00:00
|
|
|
zone_id = zones[0]["Id"].split("/")[2]
|
|
|
|
|
|
|
|
rrsets = route53.list_resource_record_sets(HostedZoneId=zone_id)[
|
|
|
|
"ResourceRecordSets"
|
|
|
|
]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert len(rrsets) == 3
|
2021-10-04 13:47:40 +00:00
|
|
|
|
|
|
|
record_set = rrsets[0]
|
2023-07-08 21:53:09 +00:00
|
|
|
assert record_set["ResourceRecords"][0]["Value"] == "my_other.example.com"
|
2022-08-23 13:45:01 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_cloudformation
|
|
|
|
@mock_route53
|
|
|
|
def test_delete_route53_recordset():
|
|
|
|
cf = boto3.client("cloudformation", region_name="us-west-1")
|
|
|
|
|
|
|
|
# given a stack with a record set
|
|
|
|
stack_name = "test_stack_recordset_delete"
|
|
|
|
template_json = json.dumps(route53_health_check.template)
|
|
|
|
|
|
|
|
cf.create_stack(StackName=stack_name, TemplateBody=template_json)
|
|
|
|
|
|
|
|
# when the stack is deleted
|
|
|
|
cf.delete_stack(StackName=stack_name)
|
|
|
|
|
|
|
|
# then it should not error
|