1512 lines
52 KiB
Python
1512 lines
52 KiB
Python
import boto3
|
|
from botocore.exceptions import ClientError
|
|
|
|
import sure # noqa # pylint: disable=unused-import
|
|
|
|
import botocore
|
|
import pytest
|
|
|
|
from moto import mock_ec2, mock_route53
|
|
|
|
|
|
@mock_route53
|
|
def test_create_hosted_zone():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
response = conn.create_hosted_zone(
|
|
Name="testdns.aws.com.", CallerReference=str(hash("foo"))
|
|
)
|
|
firstzone = response["HostedZone"]
|
|
firstzone.should.have.key("Id").match(r"/hostedzone/[A-Z0-9]+")
|
|
firstzone.should.have.key("Name").equal("testdns.aws.com.")
|
|
firstzone.should.have.key("Config").equal({"PrivateZone": False})
|
|
firstzone.should.have.key("ResourceRecordSetCount").equal(2)
|
|
|
|
delegation = response["DelegationSet"]
|
|
delegation.should.have.key("NameServers").length_of(4)
|
|
delegation["NameServers"].should.contain("ns-2048.awsdns-64.com")
|
|
delegation["NameServers"].should.contain("ns-2049.awsdns-65.net")
|
|
delegation["NameServers"].should.contain("ns-2050.awsdns-66.org")
|
|
delegation["NameServers"].should.contain("ns-2051.awsdns-67.co.uk")
|
|
|
|
|
|
@mock_route53
|
|
def test_list_hosted_zones():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
|
|
res = conn.list_hosted_zones()["HostedZones"]
|
|
res.should.have.length_of(0)
|
|
|
|
zone1 = conn.create_hosted_zone(
|
|
Name="testdns1.aws.com.", CallerReference=str(hash("foo"))
|
|
)["HostedZone"]
|
|
zone2 = conn.create_hosted_zone(
|
|
Name="testdns2.aws.com.", CallerReference=str(hash("foo"))
|
|
)["HostedZone"]
|
|
|
|
res = conn.list_hosted_zones()["HostedZones"]
|
|
res.should.have.length_of(2)
|
|
|
|
res.should.contain(zone1)
|
|
res.should.contain(zone2)
|
|
|
|
|
|
@mock_route53
|
|
def test_delete_hosted_zone():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
|
|
zone1 = conn.create_hosted_zone(
|
|
Name="testdns1.aws.com.", CallerReference=str(hash("foo"))
|
|
)["HostedZone"]
|
|
conn.create_hosted_zone(Name="testdns2.aws.com.", CallerReference=str(hash("foo")))
|
|
|
|
conn.delete_hosted_zone(Id=zone1["Id"])
|
|
|
|
res = conn.list_hosted_zones()["HostedZones"]
|
|
res.should.have.length_of(1)
|
|
|
|
|
|
@mock_route53
|
|
def test_delete_hosted_zone_with_change_sets():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
|
|
zone_id = conn.create_hosted_zone(
|
|
Name="testdns.aws.com.", CallerReference=str(hash("foo"))
|
|
)["HostedZone"]["Id"]
|
|
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=zone_id,
|
|
ChangeBatch={
|
|
"Changes": [
|
|
{
|
|
"Action": "CREATE",
|
|
"ResourceRecordSet": {
|
|
"Name": "foo.bar.testdns.aws.com",
|
|
"Type": "A",
|
|
"ResourceRecords": [{"Value": "1.2.3.4"}],
|
|
},
|
|
}
|
|
]
|
|
},
|
|
)
|
|
|
|
with pytest.raises(ClientError) as exc:
|
|
conn.delete_hosted_zone(Id=zone_id)
|
|
err = exc.value.response["Error"]
|
|
err["Code"].should.equal("HostedZoneNotEmpty")
|
|
err["Message"].should.equal(
|
|
"The hosted zone contains resource records that are not SOA or NS records."
|
|
)
|
|
|
|
|
|
@mock_route53
|
|
def test_get_hosted_zone_count_no_zones():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
zone_count = conn.get_hosted_zone_count()
|
|
zone_count.should.have.key("HostedZoneCount")
|
|
isinstance(zone_count["HostedZoneCount"], int).should.be.true
|
|
zone_count["HostedZoneCount"].should.be.equal(0)
|
|
|
|
|
|
@mock_route53
|
|
def test_get_hosted_zone_count_one_zone():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
zone = "a"
|
|
zone_name = f"test.{zone}.com."
|
|
conn.create_hosted_zone(
|
|
Name=zone_name,
|
|
CallerReference=str(hash("foo")),
|
|
HostedZoneConfig=dict(PrivateZone=False, Comment=f"test {zone} com"),
|
|
)
|
|
zone_count = conn.get_hosted_zone_count()
|
|
zone_count.should.have.key("HostedZoneCount")
|
|
isinstance(zone_count["HostedZoneCount"], int).should.be.true
|
|
zone_count["HostedZoneCount"].should.be.equal(1)
|
|
|
|
|
|
@mock_route53
|
|
def test_get_hosted_zone_count_many_zones():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
zones = {}
|
|
zone_indexes = []
|
|
for char in range(ord("a"), ord("d") + 1):
|
|
for char2 in range(ord("a"), ord("z") + 1):
|
|
zone_indexes.append(f"{chr(char)}{chr(char2)}")
|
|
|
|
# Create 100-ish zones and make sure we get 100 back. This works
|
|
# for 702 zones {a..zz}, but seemed a needless waste of
|
|
# time/resources.
|
|
for zone in zone_indexes:
|
|
zone_name = f"test.{zone}.com."
|
|
zones[zone] = conn.create_hosted_zone(
|
|
Name=zone_name,
|
|
CallerReference=str(hash("foo")),
|
|
HostedZoneConfig=dict(PrivateZone=False, Comment=f"test {zone} com"),
|
|
)
|
|
zone_count = conn.get_hosted_zone_count()
|
|
zone_count.should.have.key("HostedZoneCount")
|
|
isinstance(zone_count["HostedZoneCount"], int).should.be.true
|
|
zone_count["HostedZoneCount"].shouldnt.be.equal(0)
|
|
zone_count["HostedZoneCount"].should.be.equal(len(zone_indexes))
|
|
|
|
|
|
@mock_route53
|
|
def test_get_unknown_hosted_zone():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
|
|
with pytest.raises(ClientError) as ex:
|
|
conn.get_hosted_zone(Id="unknown")
|
|
|
|
err = ex.value.response["Error"]
|
|
err["Code"].should.equal("NoSuchHostedZone")
|
|
err["Message"].should.equal("No hosted zone found with ID: unknown")
|
|
|
|
|
|
@mock_route53
|
|
def test_update_hosted_zone_comment():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
response = conn.create_hosted_zone(
|
|
Name="testdns.aws.com.", CallerReference=str(hash("foo"))
|
|
)
|
|
zone_id = response["HostedZone"]["Id"].split("/")[-1]
|
|
|
|
conn.update_hosted_zone_comment(Id=zone_id, Comment="yolo")
|
|
|
|
resp = conn.get_hosted_zone(Id=zone_id)["HostedZone"]
|
|
resp["Config"].should.have.key("Comment").equals("yolo")
|
|
|
|
|
|
@mock_route53
|
|
def test_list_resource_record_set_unknown_zone():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
|
|
with pytest.raises(ClientError) as ex:
|
|
conn.list_resource_record_sets(HostedZoneId="abcd")
|
|
|
|
err = ex.value.response["Error"]
|
|
err["Code"].should.equal("NoSuchHostedZone")
|
|
err["Message"].should.equal("No hosted zone found with ID: abcd")
|
|
|
|
|
|
@mock_route53
|
|
def test_list_resource_record_set_unknown_type():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
zone = conn.create_hosted_zone(
|
|
Name="testdns1.aws.com.", CallerReference=str(hash("foo"))
|
|
)["HostedZone"]
|
|
|
|
with pytest.raises(ClientError) as ex:
|
|
conn.list_resource_record_sets(HostedZoneId=zone["Id"], StartRecordType="A")
|
|
|
|
err = ex.value.response["Error"]
|
|
err["Code"].should.equal("400")
|
|
err["Message"].should.equal("Bad Request")
|
|
|
|
|
|
@mock_route53
|
|
def test_use_health_check_in_resource_record_set():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
|
|
check = conn.create_health_check(
|
|
CallerReference="?",
|
|
HealthCheckConfig={
|
|
"IPAddress": "10.0.0.25",
|
|
"Port": 80,
|
|
"Type": "HTTP",
|
|
"ResourcePath": "/",
|
|
"RequestInterval": 10,
|
|
"FailureThreshold": 2,
|
|
},
|
|
)["HealthCheck"]
|
|
check_id = check["Id"]
|
|
|
|
zone = conn.create_hosted_zone(
|
|
Name="testdns.aws.com", CallerReference=str(hash("foo"))
|
|
)
|
|
zone_id = zone["HostedZone"]["Id"]
|
|
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=zone_id,
|
|
ChangeBatch={
|
|
"Changes": [
|
|
{
|
|
"Action": "CREATE",
|
|
"ResourceRecordSet": {
|
|
"Name": "foo.bar.testdns.aws.com",
|
|
"Type": "A",
|
|
"HealthCheckId": check_id,
|
|
"ResourceRecords": [{"Value": "1.2.3.4"}],
|
|
},
|
|
}
|
|
]
|
|
},
|
|
)
|
|
|
|
record_sets = conn.list_resource_record_sets(HostedZoneId=zone_id)[
|
|
"ResourceRecordSets"
|
|
]
|
|
record_sets[2]["Name"].should.equal("foo.bar.testdns.aws.com.")
|
|
record_sets[2]["HealthCheckId"].should.equal(check_id)
|
|
|
|
|
|
@mock_route53
|
|
def test_hosted_zone_comment_preserved():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
|
|
firstzone = conn.create_hosted_zone(
|
|
Name="testdns.aws.com",
|
|
CallerReference=str(hash("foo")),
|
|
HostedZoneConfig={"Comment": "test comment"},
|
|
)
|
|
zone_id = firstzone["HostedZone"]["Id"]
|
|
|
|
hosted_zone = conn.get_hosted_zone(Id=zone_id)
|
|
hosted_zone["HostedZone"]["Config"]["Comment"].should.equal("test comment")
|
|
|
|
hosted_zones = conn.list_hosted_zones()
|
|
hosted_zones["HostedZones"][0]["Config"]["Comment"].should.equal("test comment")
|
|
|
|
|
|
@mock_route53
|
|
def test_deleting_weighted_route():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
|
|
zone = conn.create_hosted_zone(
|
|
Name="testdns.aws.com", CallerReference=str(hash("foo"))
|
|
)
|
|
zone_id = zone["HostedZone"]["Id"]
|
|
|
|
for identifier in ["success-test-foo", "success-test-bar"]:
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=zone_id,
|
|
ChangeBatch={
|
|
"Changes": [
|
|
{
|
|
"Action": "CREATE",
|
|
"ResourceRecordSet": {
|
|
"Name": "cname.testdns.aws.com",
|
|
"Type": "CNAME",
|
|
"SetIdentifier": identifier,
|
|
"Weight": 50,
|
|
},
|
|
}
|
|
]
|
|
},
|
|
)
|
|
|
|
cnames = conn.list_resource_record_sets(
|
|
HostedZoneId=zone_id, StartRecordName="cname", StartRecordType="CNAME"
|
|
)["ResourceRecordSets"]
|
|
cnames.should.have.length_of(4)
|
|
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=zone_id,
|
|
ChangeBatch={
|
|
"Changes": [
|
|
{
|
|
"Action": "DELETE",
|
|
"ResourceRecordSet": {
|
|
"Name": "cname.testdns.aws.com",
|
|
"Type": "CNAME",
|
|
"SetIdentifier": "success-test-foo",
|
|
"Weight": 50,
|
|
},
|
|
}
|
|
]
|
|
},
|
|
)
|
|
|
|
cnames = conn.list_resource_record_sets(
|
|
HostedZoneId=zone_id, StartRecordName="cname", StartRecordType="CNAME"
|
|
)["ResourceRecordSets"]
|
|
cnames.should.have.length_of(3)
|
|
cnames[-1]["Name"].should.equal("cname.testdns.aws.com.")
|
|
cnames[-1]["SetIdentifier"].should.equal("success-test-bar")
|
|
|
|
|
|
@mock_route53
|
|
def test_deleting_latency_route():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
|
|
zone = conn.create_hosted_zone(
|
|
Name="testdns.aws.com", CallerReference=str(hash("foo"))
|
|
)
|
|
zone_id = zone["HostedZone"]["Id"]
|
|
|
|
for _id, region in [
|
|
("success-test-foo", "us-west-2"),
|
|
("success-test-bar", "us-west-1"),
|
|
]:
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=zone_id,
|
|
ChangeBatch={
|
|
"Changes": [
|
|
{
|
|
"Action": "CREATE",
|
|
"ResourceRecordSet": {
|
|
"Name": "cname.testdns.aws.com",
|
|
"Type": "CNAME",
|
|
"SetIdentifier": _id,
|
|
"Region": region,
|
|
"ResourceRecords": [{"Value": "example.com"}],
|
|
},
|
|
}
|
|
]
|
|
},
|
|
)
|
|
|
|
cnames = conn.list_resource_record_sets(
|
|
HostedZoneId=zone_id, StartRecordName="cname", StartRecordType="CNAME"
|
|
)["ResourceRecordSets"]
|
|
cnames.should.have.length_of(4)
|
|
foo_cname = [
|
|
cname
|
|
for cname in cnames
|
|
if cname.get("SetIdentifier") and cname["SetIdentifier"] == "success-test-foo"
|
|
][0]
|
|
foo_cname["Region"].should.equal("us-west-2")
|
|
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=zone_id,
|
|
ChangeBatch={
|
|
"Changes": [
|
|
{
|
|
"Action": "DELETE",
|
|
"ResourceRecordSet": {
|
|
"Name": "cname.testdns.aws.com",
|
|
"Type": "CNAME",
|
|
"SetIdentifier": "success-test-foo",
|
|
"Region": "us-west-2",
|
|
"ResourceRecords": [{"Value": "example.com"}],
|
|
},
|
|
}
|
|
]
|
|
},
|
|
)
|
|
cnames = conn.list_resource_record_sets(
|
|
HostedZoneId=zone_id, StartRecordName="cname", StartRecordType="CNAME"
|
|
)["ResourceRecordSets"]
|
|
cnames.should.have.length_of(3)
|
|
cnames[-1]["SetIdentifier"].should.equal("success-test-bar")
|
|
cnames[-1]["Region"].should.equal("us-west-1")
|
|
|
|
|
|
@mock_route53
|
|
def test_list_or_change_tags_for_resource_request():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
health_check = conn.create_health_check(
|
|
CallerReference="foobar",
|
|
HealthCheckConfig={
|
|
"IPAddress": "192.0.2.44",
|
|
"Port": 123,
|
|
"Type": "HTTP",
|
|
"ResourcePath": "/",
|
|
"RequestInterval": 30,
|
|
"FailureThreshold": 123,
|
|
"HealthThreshold": 123,
|
|
},
|
|
)
|
|
healthcheck_id = health_check["HealthCheck"]["Id"]
|
|
|
|
# confirm this works for resources with zero tags
|
|
response = conn.list_tags_for_resource(
|
|
ResourceType="healthcheck", ResourceId=healthcheck_id
|
|
)
|
|
response["ResourceTagSet"]["Tags"].should.equal([])
|
|
|
|
tag1 = {"Key": "Deploy", "Value": "True"}
|
|
tag2 = {"Key": "Name", "Value": "UnitTest"}
|
|
|
|
# Test adding a tag for a resource id
|
|
conn.change_tags_for_resource(
|
|
ResourceType="healthcheck", ResourceId=healthcheck_id, AddTags=[tag1, tag2]
|
|
)
|
|
|
|
# Check to make sure that the response has the 'ResourceTagSet' key
|
|
response = conn.list_tags_for_resource(
|
|
ResourceType="healthcheck", ResourceId=healthcheck_id
|
|
)
|
|
response.should.contain("ResourceTagSet")
|
|
|
|
# Validate that each key was added
|
|
response["ResourceTagSet"]["Tags"].should.contain(tag1)
|
|
response["ResourceTagSet"]["Tags"].should.contain(tag2)
|
|
|
|
len(response["ResourceTagSet"]["Tags"]).should.equal(2)
|
|
|
|
# Try to remove the tags
|
|
conn.change_tags_for_resource(
|
|
ResourceType="healthcheck",
|
|
ResourceId=healthcheck_id,
|
|
RemoveTagKeys=[tag1["Key"]],
|
|
)
|
|
|
|
# Check to make sure that the response has the 'ResourceTagSet' key
|
|
response = conn.list_tags_for_resource(
|
|
ResourceType="healthcheck", ResourceId=healthcheck_id
|
|
)
|
|
response.should.contain("ResourceTagSet")
|
|
response["ResourceTagSet"]["Tags"].should_not.contain(tag1)
|
|
response["ResourceTagSet"]["Tags"].should.contain(tag2)
|
|
|
|
# Remove the second tag
|
|
conn.change_tags_for_resource(
|
|
ResourceType="healthcheck",
|
|
ResourceId=healthcheck_id,
|
|
RemoveTagKeys=[tag2["Key"]],
|
|
)
|
|
|
|
response = conn.list_tags_for_resource(
|
|
ResourceType="healthcheck", ResourceId=healthcheck_id
|
|
)
|
|
response["ResourceTagSet"]["Tags"].should_not.contain(tag2)
|
|
|
|
# Re-add the tags
|
|
conn.change_tags_for_resource(
|
|
ResourceType="healthcheck", ResourceId=healthcheck_id, AddTags=[tag1, tag2]
|
|
)
|
|
|
|
# Remove both
|
|
conn.change_tags_for_resource(
|
|
ResourceType="healthcheck",
|
|
ResourceId=healthcheck_id,
|
|
RemoveTagKeys=[tag1["Key"], tag2["Key"]],
|
|
)
|
|
|
|
response = conn.list_tags_for_resource(
|
|
ResourceType="healthcheck", ResourceId=healthcheck_id
|
|
)
|
|
response["ResourceTagSet"]["Tags"].should.equal([])
|
|
|
|
|
|
@mock_ec2
|
|
@mock_route53
|
|
def test_list_hosted_zones_by_name():
|
|
|
|
# Create mock VPC so we can get a VPC ID
|
|
ec2c = boto3.client("ec2", region_name="us-east-1")
|
|
vpc_id = ec2c.create_vpc(CidrBlock="10.1.0.0/16").get("Vpc").get("VpcId")
|
|
region = "us-east-1"
|
|
|
|
conn = boto3.client("route53", region_name=region)
|
|
zone_b = conn.create_hosted_zone(
|
|
Name="test.b.com.",
|
|
CallerReference=str(hash("foo")),
|
|
HostedZoneConfig=dict(PrivateZone=True, Comment="test com"),
|
|
VPC={"VPCRegion": region, "VPCId": vpc_id},
|
|
)
|
|
|
|
zone_b = conn.list_hosted_zones_by_name(DNSName="test.b.com.")
|
|
len(zone_b["HostedZones"]).should.equal(1)
|
|
zone_b["HostedZones"][0]["Name"].should.equal("test.b.com.")
|
|
zone_b["HostedZones"][0].should.have.key("Config")
|
|
zone_b["HostedZones"][0]["Config"].should.have.key("PrivateZone")
|
|
zone_b["HostedZones"][0]["Config"]["PrivateZone"].should.be.equal(True)
|
|
|
|
# We declared this a a private hosted zone above, so let's make
|
|
# sure it really is!
|
|
zone_b_id = zone_b["HostedZones"][0]["Id"].split("/")[-1]
|
|
b_hosted_zone = conn.get_hosted_zone(Id=zone_b_id)
|
|
|
|
# Pull the HostedZone block out and test it.
|
|
b_hosted_zone.should.have.key("HostedZone")
|
|
b_hz = b_hosted_zone["HostedZone"]
|
|
b_hz.should.have.key("Config")
|
|
b_hz["Config"].should.have.key("PrivateZone")
|
|
b_hz["Config"]["PrivateZone"].should.be.equal(True)
|
|
|
|
# Check for the VPCs block since this *should* be a VPC-Private Zone
|
|
b_hosted_zone.should.have.key("VPCs")
|
|
b_hosted_zone["VPCs"].should.have.length_of(1)
|
|
b_hz_vpcs = b_hosted_zone["VPCs"][0]
|
|
b_hz_vpcs.should.have.key("VPCId")
|
|
b_hz_vpcs.should.have.key("VPCRegion")
|
|
b_hz_vpcs["VPCId"].should_not.equal("")
|
|
b_hz_vpcs["VPCRegion"].should_not.equal("")
|
|
b_hz_vpcs["VPCId"].should.be.equal(vpc_id)
|
|
b_hz_vpcs["VPCRegion"].should.be.equal(region)
|
|
|
|
# Now create other zones and test them.
|
|
conn.create_hosted_zone(
|
|
Name="test.a.org.",
|
|
CallerReference=str(hash("bar")),
|
|
HostedZoneConfig=dict(PrivateZone=False, Comment="test org"),
|
|
)
|
|
conn.create_hosted_zone(
|
|
Name="test.a.org.",
|
|
CallerReference=str(hash("bar")),
|
|
HostedZoneConfig=dict(PrivateZone=False, Comment="test org 2"),
|
|
)
|
|
|
|
# Now makes sure the other zones we created above are NOT private...
|
|
zones = conn.list_hosted_zones_by_name(DNSName="test.a.org.")
|
|
len(zones["HostedZones"]).should.equal(2)
|
|
zones["HostedZones"][0]["Name"].should.equal("test.a.org.")
|
|
zones["HostedZones"][0].should.have.key("Config")
|
|
zones["HostedZones"][0]["Config"].should.have.key("PrivateZone")
|
|
zones["HostedZones"][0]["Config"]["PrivateZone"].should.be.equal(False)
|
|
|
|
zones["HostedZones"][1]["Name"].should.equal("test.a.org.")
|
|
zones["HostedZones"][1].should.have.key("Config")
|
|
zones["HostedZones"][1]["Config"].should.have.key("PrivateZone")
|
|
zones["HostedZones"][1]["Config"]["PrivateZone"].should.be.equal(False)
|
|
|
|
# test sort order
|
|
zones = conn.list_hosted_zones_by_name()
|
|
len(zones["HostedZones"]).should.equal(3)
|
|
zones["HostedZones"][0]["Name"].should.equal("test.b.com.")
|
|
zones["HostedZones"][1]["Name"].should.equal("test.a.org.")
|
|
zones["HostedZones"][2]["Name"].should.equal("test.a.org.")
|
|
|
|
|
|
@mock_route53
|
|
def test_list_hosted_zones_by_dns_name():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
conn.create_hosted_zone(
|
|
Name="test.b.com.",
|
|
CallerReference=str(hash("foo")),
|
|
HostedZoneConfig=dict(PrivateZone=False, Comment="test com"),
|
|
)
|
|
conn.create_hosted_zone(
|
|
Name="test.a.org.",
|
|
CallerReference=str(hash("bar")),
|
|
HostedZoneConfig=dict(PrivateZone=False, Comment="test org"),
|
|
)
|
|
conn.create_hosted_zone(
|
|
Name="test.a.org.",
|
|
CallerReference=str(hash("bar")),
|
|
HostedZoneConfig=dict(PrivateZone=False, Comment="test org 2"),
|
|
)
|
|
conn.create_hosted_zone(
|
|
Name="my.test.net.",
|
|
CallerReference=str(hash("baz")),
|
|
HostedZoneConfig=dict(PrivateZone=False, Comment="test net"),
|
|
)
|
|
|
|
# test lookup
|
|
zones = conn.list_hosted_zones_by_name(DNSName="test.b.com.")
|
|
len(zones["HostedZones"]).should.equal(1)
|
|
zones["DNSName"].should.equal("test.b.com.")
|
|
zones = conn.list_hosted_zones_by_name(DNSName="test.a.org.")
|
|
len(zones["HostedZones"]).should.equal(2)
|
|
zones["DNSName"].should.equal("test.a.org.")
|
|
zones["DNSName"].should.equal("test.a.org.")
|
|
zones = conn.list_hosted_zones_by_name(DNSName="my.test.net.")
|
|
len(zones["HostedZones"]).should.equal(1)
|
|
zones["DNSName"].should.equal("my.test.net.")
|
|
zones = conn.list_hosted_zones_by_name(DNSName="my.test.net")
|
|
len(zones["HostedZones"]).should.equal(1)
|
|
zones["DNSName"].should.equal("my.test.net.")
|
|
|
|
# test sort order
|
|
zones = conn.list_hosted_zones_by_name()
|
|
len(zones["HostedZones"]).should.equal(4)
|
|
zones["HostedZones"][0]["Name"].should.equal("test.b.com.")
|
|
zones["HostedZones"][1]["Name"].should.equal("my.test.net.")
|
|
zones["HostedZones"][2]["Name"].should.equal("test.a.org.")
|
|
zones["HostedZones"][3]["Name"].should.equal("test.a.org.")
|
|
|
|
|
|
@mock_route53
|
|
def test_change_resource_record_sets_crud_valid():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
conn.create_hosted_zone(
|
|
Name="db.",
|
|
CallerReference=str(hash("foo")),
|
|
HostedZoneConfig=dict(PrivateZone=False, Comment="db"),
|
|
)
|
|
|
|
zones = conn.list_hosted_zones_by_name(DNSName="db.")
|
|
len(zones["HostedZones"]).should.equal(1)
|
|
zones["HostedZones"][0]["Name"].should.equal("db.")
|
|
hosted_zone_id = zones["HostedZones"][0]["Id"]
|
|
|
|
# Create A Record.
|
|
a_record_endpoint_payload = {
|
|
"Comment": "Create A record prod.redis.db",
|
|
"Changes": [
|
|
{
|
|
"Action": "CREATE",
|
|
"ResourceRecordSet": {
|
|
"Name": "prod.redis.db.",
|
|
"Type": "A",
|
|
"TTL": 10,
|
|
"ResourceRecords": [{"Value": "127.0.0.1"}],
|
|
},
|
|
}
|
|
],
|
|
}
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id, ChangeBatch=a_record_endpoint_payload
|
|
)
|
|
|
|
response = conn.list_resource_record_sets(HostedZoneId=hosted_zone_id)
|
|
len(response["ResourceRecordSets"]).should.equal(3)
|
|
a_record_detail = response["ResourceRecordSets"][2]
|
|
a_record_detail["Name"].should.equal("prod.redis.db.")
|
|
a_record_detail["Type"].should.equal("A")
|
|
a_record_detail["TTL"].should.equal(10)
|
|
a_record_detail["ResourceRecords"].should.equal([{"Value": "127.0.0.1"}])
|
|
|
|
# Update A Record.
|
|
cname_record_endpoint_payload = {
|
|
"Comment": "Update A record prod.redis.db",
|
|
"Changes": [
|
|
{
|
|
"Action": "UPSERT",
|
|
"ResourceRecordSet": {
|
|
"Name": "prod.redis.db.",
|
|
"Type": "A",
|
|
"TTL": 60,
|
|
"ResourceRecords": [{"Value": "192.168.1.1"}],
|
|
},
|
|
}
|
|
],
|
|
}
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id, ChangeBatch=cname_record_endpoint_payload
|
|
)
|
|
|
|
response = conn.list_resource_record_sets(HostedZoneId=hosted_zone_id)
|
|
len(response["ResourceRecordSets"]).should.equal(3)
|
|
cname_record_detail = response["ResourceRecordSets"][2]
|
|
cname_record_detail["Name"].should.equal("prod.redis.db.")
|
|
cname_record_detail["Type"].should.equal("A")
|
|
cname_record_detail["TTL"].should.equal(60)
|
|
cname_record_detail["ResourceRecords"].should.equal([{"Value": "192.168.1.1"}])
|
|
|
|
# Update to add Alias.
|
|
cname_alias_record_endpoint_payload = {
|
|
"Comment": "Update to Alias prod.redis.db",
|
|
"Changes": [
|
|
{
|
|
"Action": "UPSERT",
|
|
"ResourceRecordSet": {
|
|
"Name": "prod.redis.db.",
|
|
"Type": "A",
|
|
"TTL": 60,
|
|
"AliasTarget": {
|
|
"HostedZoneId": hosted_zone_id,
|
|
"DNSName": "prod.redis.alias.",
|
|
"EvaluateTargetHealth": False,
|
|
},
|
|
},
|
|
}
|
|
],
|
|
}
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id, ChangeBatch=cname_alias_record_endpoint_payload
|
|
)
|
|
|
|
response = conn.list_resource_record_sets(HostedZoneId=hosted_zone_id)
|
|
cname_alias_record_detail = response["ResourceRecordSets"][2]
|
|
cname_alias_record_detail["Name"].should.equal("prod.redis.db.")
|
|
cname_alias_record_detail["Type"].should.equal("A")
|
|
cname_alias_record_detail["TTL"].should.equal(60)
|
|
cname_alias_record_detail["AliasTarget"].should.equal(
|
|
{
|
|
"HostedZoneId": hosted_zone_id,
|
|
"DNSName": "prod.redis.alias.",
|
|
"EvaluateTargetHealth": False,
|
|
}
|
|
)
|
|
cname_alias_record_detail.should_not.contain("ResourceRecords")
|
|
|
|
# Delete record.
|
|
delete_payload = {
|
|
"Comment": "delete prod.redis.db",
|
|
"Changes": [
|
|
{
|
|
"Action": "DELETE",
|
|
"ResourceRecordSet": {
|
|
"Name": "prod.redis.db.",
|
|
"Type": "A",
|
|
"TTL": 60,
|
|
"ResourceRecords": [{"Value": "192.168.1.1"}],
|
|
},
|
|
}
|
|
],
|
|
}
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id, ChangeBatch=delete_payload
|
|
)
|
|
response = conn.list_resource_record_sets(HostedZoneId=hosted_zone_id)
|
|
len(response["ResourceRecordSets"]).should.equal(2)
|
|
|
|
|
|
@mock_route53
|
|
def test_change_resource_record_sets_crud_valid_with_special_xml_chars():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
conn.create_hosted_zone(
|
|
Name="db.",
|
|
CallerReference=str(hash("foo")),
|
|
HostedZoneConfig=dict(PrivateZone=False, Comment="db"),
|
|
)
|
|
|
|
zones = conn.list_hosted_zones_by_name(DNSName="db.")
|
|
len(zones["HostedZones"]).should.equal(1)
|
|
zones["HostedZones"][0]["Name"].should.equal("db.")
|
|
hosted_zone_id = zones["HostedZones"][0]["Id"]
|
|
|
|
# Create TXT Record.
|
|
txt_record_endpoint_payload = {
|
|
"Comment": "Create TXT record prod.redis.db",
|
|
"Changes": [
|
|
{
|
|
"Action": "CREATE",
|
|
"ResourceRecordSet": {
|
|
"Name": "prod.redis.db.",
|
|
"Type": "TXT",
|
|
"TTL": 10,
|
|
"ResourceRecords": [{"Value": "SomeInitialValue"}],
|
|
},
|
|
}
|
|
],
|
|
}
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id, ChangeBatch=txt_record_endpoint_payload
|
|
)
|
|
|
|
response = conn.list_resource_record_sets(HostedZoneId=hosted_zone_id)
|
|
len(response["ResourceRecordSets"]).should.equal(3)
|
|
a_record_detail = response["ResourceRecordSets"][2]
|
|
a_record_detail["Name"].should.equal("prod.redis.db.")
|
|
a_record_detail["Type"].should.equal("TXT")
|
|
a_record_detail["TTL"].should.equal(10)
|
|
a_record_detail["ResourceRecords"].should.equal([{"Value": "SomeInitialValue"}])
|
|
|
|
# Update TXT Record with XML Special Character &.
|
|
txt_record_with_special_char_endpoint_payload = {
|
|
"Comment": "Update TXT record prod.redis.db",
|
|
"Changes": [
|
|
{
|
|
"Action": "UPSERT",
|
|
"ResourceRecordSet": {
|
|
"Name": "prod.redis.db.",
|
|
"Type": "TXT",
|
|
"TTL": 60,
|
|
"ResourceRecords": [{"Value": "SomeInitialValue&NewValue"}],
|
|
},
|
|
}
|
|
],
|
|
}
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id,
|
|
ChangeBatch=txt_record_with_special_char_endpoint_payload,
|
|
)
|
|
|
|
response = conn.list_resource_record_sets(HostedZoneId=hosted_zone_id)
|
|
len(response["ResourceRecordSets"]).should.equal(3)
|
|
cname_record_detail = response["ResourceRecordSets"][2]
|
|
cname_record_detail["Name"].should.equal("prod.redis.db.")
|
|
cname_record_detail["Type"].should.equal("TXT")
|
|
cname_record_detail["TTL"].should.equal(60)
|
|
cname_record_detail["ResourceRecords"].should.equal(
|
|
[{"Value": "SomeInitialValue&NewValue"}]
|
|
)
|
|
|
|
# Delete record.
|
|
delete_payload = {
|
|
"Comment": "delete prod.redis.db",
|
|
"Changes": [
|
|
{
|
|
"Action": "DELETE",
|
|
"ResourceRecordSet": {
|
|
"Name": "prod.redis.db.",
|
|
"Type": "TXT",
|
|
"TTL": 10,
|
|
"ResourceRecords": [{"Value": "SomeInitialValue"}],
|
|
},
|
|
}
|
|
],
|
|
}
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id, ChangeBatch=delete_payload
|
|
)
|
|
response = conn.list_resource_record_sets(HostedZoneId=hosted_zone_id)
|
|
len(response["ResourceRecordSets"]).should.equal(2)
|
|
|
|
|
|
@mock_route53
|
|
def test_change_resource_record_set__delete_should_match_create():
|
|
# To delete a resource record set, you must specify all the same values that you specified when you created it.
|
|
client = boto3.client("route53", region_name="us-east-1")
|
|
name = "example.com"
|
|
hosted_zone_id = client.create_hosted_zone(Name=name, CallerReference=name)[
|
|
"HostedZone"
|
|
]["Id"]
|
|
|
|
create_call = client.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id,
|
|
ChangeBatch={
|
|
"Changes": [
|
|
{
|
|
"Action": "CREATE",
|
|
"ResourceRecordSet": {
|
|
"Name": name,
|
|
"Type": "A",
|
|
"TTL": 300,
|
|
"ResourceRecords": [{"Value": "192.168.0.1"}],
|
|
},
|
|
}
|
|
]
|
|
},
|
|
)
|
|
waiter = client.get_waiter("resource_record_sets_changed")
|
|
waiter.wait(Id=create_call["ChangeInfo"]["Id"])
|
|
|
|
with pytest.raises(ClientError) as exc:
|
|
client.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id,
|
|
ChangeBatch={
|
|
"Changes": [
|
|
{
|
|
"Action": "DELETE",
|
|
"ResourceRecordSet": {
|
|
"Name": name,
|
|
"Type": "A"
|
|
# Missing TTL and ResourceRecords
|
|
},
|
|
}
|
|
]
|
|
},
|
|
)
|
|
err = exc.value.response["Error"]
|
|
err["Code"].should.equal("InvalidInput")
|
|
err["Message"].should.equal(
|
|
"Invalid request: Expected exactly one of [AliasTarget, all of [TTL, and ResourceRecords], or TrafficPolicyInstanceId], but found none in Change with [Action=DELETE, Name=example.com, Type=A, SetIdentifier=null]"
|
|
)
|
|
|
|
|
|
@mock_route53
|
|
def test_change_weighted_resource_record_sets():
|
|
conn = boto3.client("route53", region_name="us-east-2")
|
|
conn.create_hosted_zone(
|
|
Name="test.vpc.internal.", CallerReference=str(hash("test"))
|
|
)
|
|
|
|
zones = conn.list_hosted_zones_by_name(DNSName="test.vpc.internal.")
|
|
|
|
hosted_zone_id = zones["HostedZones"][0]["Id"]
|
|
|
|
# Create 2 weighted records
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id,
|
|
ChangeBatch={
|
|
"Changes": [
|
|
{
|
|
"Action": "CREATE",
|
|
"ResourceRecordSet": {
|
|
"Name": "test.vpc.internal",
|
|
"Type": "A",
|
|
"SetIdentifier": "test1",
|
|
"Weight": 50,
|
|
"AliasTarget": {
|
|
"HostedZoneId": "Z3AADJGX6KTTL2",
|
|
"DNSName": "internal-test1lb-447688172.us-east-2.elb.amazonaws.com.",
|
|
"EvaluateTargetHealth": True,
|
|
},
|
|
},
|
|
},
|
|
{
|
|
"Action": "CREATE",
|
|
"ResourceRecordSet": {
|
|
"Name": "test.vpc.internal",
|
|
"Type": "A",
|
|
"SetIdentifier": "test2",
|
|
"Weight": 50,
|
|
"AliasTarget": {
|
|
"HostedZoneId": "Z3AADJGX6KTTL2",
|
|
"DNSName": "internal-testlb2-1116641781.us-east-2.elb.amazonaws.com.",
|
|
"EvaluateTargetHealth": True,
|
|
},
|
|
},
|
|
},
|
|
]
|
|
},
|
|
)
|
|
|
|
rr_sets = conn.list_resource_record_sets(HostedZoneId=hosted_zone_id)[
|
|
"ResourceRecordSets"
|
|
]
|
|
record = [r for r in rr_sets if r["Type"] == "A"][0]
|
|
# Update the first record to have a weight of 90
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id,
|
|
ChangeBatch={
|
|
"Changes": [
|
|
{
|
|
"Action": "UPSERT",
|
|
"ResourceRecordSet": {
|
|
"Name": record["Name"],
|
|
"Type": record["Type"],
|
|
"SetIdentifier": record["SetIdentifier"],
|
|
"Weight": 90,
|
|
"AliasTarget": {
|
|
"HostedZoneId": record["AliasTarget"]["HostedZoneId"],
|
|
"DNSName": record["AliasTarget"]["DNSName"],
|
|
"EvaluateTargetHealth": record["AliasTarget"][
|
|
"EvaluateTargetHealth"
|
|
],
|
|
},
|
|
},
|
|
}
|
|
]
|
|
},
|
|
)
|
|
|
|
record = [r for r in rr_sets if r["Type"] == "A"][1]
|
|
# Update the second record to have a weight of 10
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id,
|
|
ChangeBatch={
|
|
"Changes": [
|
|
{
|
|
"Action": "UPSERT",
|
|
"ResourceRecordSet": {
|
|
"Name": record["Name"],
|
|
"Type": record["Type"],
|
|
"SetIdentifier": record["SetIdentifier"],
|
|
"Weight": 10,
|
|
"AliasTarget": {
|
|
"HostedZoneId": record["AliasTarget"]["HostedZoneId"],
|
|
"DNSName": record["AliasTarget"]["DNSName"],
|
|
"EvaluateTargetHealth": record["AliasTarget"][
|
|
"EvaluateTargetHealth"
|
|
],
|
|
},
|
|
},
|
|
}
|
|
]
|
|
},
|
|
)
|
|
|
|
response = conn.list_resource_record_sets(HostedZoneId=hosted_zone_id)
|
|
for record in response["ResourceRecordSets"]:
|
|
if record.get("SetIdentifier"):
|
|
if record["SetIdentifier"] == "test1":
|
|
record["Weight"].should.equal(90)
|
|
if record["SetIdentifier"] == "test2":
|
|
record["Weight"].should.equal(10)
|
|
|
|
|
|
@mock_route53
|
|
def test_failover_record_sets():
|
|
conn = boto3.client("route53", region_name="us-east-2")
|
|
conn.create_hosted_zone(Name="test.zone.", CallerReference=str(hash("test")))
|
|
zones = conn.list_hosted_zones_by_name(DNSName="test.zone.")
|
|
hosted_zone_id = zones["HostedZones"][0]["Id"]
|
|
|
|
# Create geolocation record
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id,
|
|
ChangeBatch={
|
|
"Changes": [
|
|
{
|
|
"Action": "CREATE",
|
|
"ResourceRecordSet": {
|
|
"Name": "failover.test.zone.",
|
|
"Type": "A",
|
|
"TTL": 10,
|
|
"ResourceRecords": [{"Value": "127.0.0.1"}],
|
|
"Failover": "PRIMARY",
|
|
},
|
|
}
|
|
]
|
|
},
|
|
)
|
|
|
|
response = conn.list_resource_record_sets(HostedZoneId=hosted_zone_id)
|
|
record = response["ResourceRecordSets"][2]
|
|
record["Failover"].should.equal("PRIMARY")
|
|
|
|
|
|
@mock_route53
|
|
def test_geolocation_record_sets():
|
|
conn = boto3.client("route53", region_name="us-east-2")
|
|
conn.create_hosted_zone(Name="test.zone.", CallerReference=str(hash("test")))
|
|
zones = conn.list_hosted_zones_by_name(DNSName="test.zone.")
|
|
hosted_zone_id = zones["HostedZones"][0]["Id"]
|
|
|
|
# Create geolocation record
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id,
|
|
ChangeBatch={
|
|
"Changes": [
|
|
{
|
|
"Action": "CREATE",
|
|
"ResourceRecordSet": {
|
|
"Name": "georecord1.test.zone.",
|
|
"Type": "A",
|
|
"TTL": 10,
|
|
"ResourceRecords": [{"Value": "127.0.0.1"}],
|
|
"GeoLocation": {"ContinentCode": "EU"},
|
|
},
|
|
},
|
|
{
|
|
"Action": "CREATE",
|
|
"ResourceRecordSet": {
|
|
"Name": "georecord2.test.zone.",
|
|
"Type": "A",
|
|
"TTL": 10,
|
|
"ResourceRecords": [{"Value": "127.0.0.2"}],
|
|
"GeoLocation": {"CountryCode": "US", "SubdivisionCode": "NY"},
|
|
},
|
|
},
|
|
]
|
|
},
|
|
)
|
|
|
|
response = conn.list_resource_record_sets(HostedZoneId=hosted_zone_id)
|
|
rrs = response["ResourceRecordSets"]
|
|
rrs[2]["GeoLocation"].should.equal({"ContinentCode": "EU"})
|
|
rrs[3]["GeoLocation"].should.equal({"CountryCode": "US", "SubdivisionCode": "NY"})
|
|
|
|
|
|
@mock_route53
|
|
def test_change_resource_record_invalid():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
conn.create_hosted_zone(
|
|
Name="db.",
|
|
CallerReference=str(hash("foo")),
|
|
HostedZoneConfig=dict(PrivateZone=False, Comment="db"),
|
|
)
|
|
|
|
zones = conn.list_hosted_zones_by_name(DNSName="db.")
|
|
len(zones["HostedZones"]).should.equal(1)
|
|
zones["HostedZones"][0]["Name"].should.equal("db.")
|
|
hosted_zone_id = zones["HostedZones"][0]["Id"]
|
|
|
|
invalid_a_record_payload = {
|
|
"Comment": "this should fail",
|
|
"Changes": [
|
|
{
|
|
"Action": "CREATE",
|
|
"ResourceRecordSet": {
|
|
"Name": "prod.scooby.doo",
|
|
"Type": "A",
|
|
"TTL": 10,
|
|
"ResourceRecords": [{"Value": "127.0.0.1"}],
|
|
},
|
|
}
|
|
],
|
|
}
|
|
|
|
with pytest.raises(botocore.exceptions.ClientError):
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id, ChangeBatch=invalid_a_record_payload
|
|
)
|
|
|
|
response = conn.list_resource_record_sets(HostedZoneId=hosted_zone_id)
|
|
len(response["ResourceRecordSets"]).should.equal(2)
|
|
|
|
invalid_cname_record_payload = {
|
|
"Comment": "this should also fail",
|
|
"Changes": [
|
|
{
|
|
"Action": "UPSERT",
|
|
"ResourceRecordSet": {
|
|
"Name": "prod.scooby.doo",
|
|
"Type": "CNAME",
|
|
"TTL": 10,
|
|
"ResourceRecords": [{"Value": "127.0.0.1"}],
|
|
},
|
|
}
|
|
],
|
|
}
|
|
|
|
with pytest.raises(botocore.exceptions.ClientError):
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id, ChangeBatch=invalid_cname_record_payload
|
|
)
|
|
|
|
response = conn.list_resource_record_sets(HostedZoneId=hosted_zone_id)
|
|
len(response["ResourceRecordSets"]).should.equal(2)
|
|
|
|
|
|
@mock_route53
|
|
def test_change_resource_record_invalid_action_value():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
conn.create_hosted_zone(
|
|
Name="db.",
|
|
CallerReference=str(hash("foo")),
|
|
HostedZoneConfig=dict(PrivateZone=False, Comment="db"),
|
|
)
|
|
|
|
zones = conn.list_hosted_zones_by_name(DNSName="db.")
|
|
len(zones["HostedZones"]).should.equal(1)
|
|
zones["HostedZones"][0]["Name"].should.equal("db.")
|
|
hosted_zone_id = zones["HostedZones"][0]["Id"]
|
|
|
|
invalid_a_record_payload = {
|
|
"Comment": "this should fail",
|
|
"Changes": [
|
|
{
|
|
"Action": "INVALID_ACTION",
|
|
"ResourceRecordSet": {
|
|
"Name": "prod.scooby.doo",
|
|
"Type": "A",
|
|
"TTL": 10,
|
|
"ResourceRecords": [{"Value": "127.0.0.1"}],
|
|
},
|
|
}
|
|
],
|
|
}
|
|
|
|
with pytest.raises(botocore.exceptions.ClientError) as exc:
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id, ChangeBatch=invalid_a_record_payload
|
|
)
|
|
|
|
err = exc.value.response["Error"]
|
|
err["Code"].should.equal("InvalidInput")
|
|
err["Message"].should.equal(
|
|
"Invalid XML ; cvc-enumeration-valid: Value 'INVALID_ACTION' is not facet-valid"
|
|
" with respect to enumeration '[CREATE, DELETE, UPSERT]'. It must be a value from the enumeration."
|
|
)
|
|
|
|
response = conn.list_resource_record_sets(HostedZoneId=hosted_zone_id)
|
|
len(response["ResourceRecordSets"]).should.equal(2)
|
|
|
|
|
|
@mock_route53
|
|
def test_change_resource_record_set_create__should_fail_when_record_already_exists():
|
|
ZONE = "cname.local"
|
|
FQDN = f"test.{ZONE}"
|
|
FQDN_TARGET = "develop.domain.com"
|
|
|
|
client = boto3.client("route53", region_name="us-east-1")
|
|
zone_id = client.create_hosted_zone(
|
|
Name=ZONE, CallerReference="ref", DelegationSetId="string"
|
|
)["HostedZone"]["Id"]
|
|
changes = {
|
|
"Changes": [
|
|
{
|
|
"Action": "CREATE",
|
|
"ResourceRecordSet": {
|
|
"Name": FQDN,
|
|
"Type": "CNAME",
|
|
"TTL": 600,
|
|
"ResourceRecords": [{"Value": FQDN_TARGET}],
|
|
},
|
|
}
|
|
]
|
|
}
|
|
client.change_resource_record_sets(HostedZoneId=zone_id, ChangeBatch=changes)
|
|
|
|
with pytest.raises(ClientError) as exc:
|
|
client.change_resource_record_sets(HostedZoneId=zone_id, ChangeBatch=changes)
|
|
|
|
err = exc.value.response["Error"]
|
|
err["Code"].should.equal("InvalidChangeBatch")
|
|
err["Message"].should.equal(
|
|
"Tried to create resource record set [name='test.cname.local.', type='CNAME'] but it already exists"
|
|
)
|
|
|
|
|
|
@mock_route53
|
|
def test_change_resource_record_set__should_create_record_when_using_upsert():
|
|
route53_client = boto3.client("route53", region_name="us-east-1")
|
|
|
|
hosted_zone = route53_client.create_hosted_zone(
|
|
Name="example.com", CallerReference="irrelevant"
|
|
)["HostedZone"]
|
|
|
|
resource_record = {
|
|
"Name": "test.example.com.",
|
|
"Type": "CNAME",
|
|
"TTL": 60,
|
|
"ResourceRecords": [{"Value": "www.test.example.com"}],
|
|
}
|
|
|
|
route53_client.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone["Id"],
|
|
ChangeBatch={
|
|
"Changes": [{"Action": "UPSERT", "ResourceRecordSet": resource_record}],
|
|
},
|
|
)
|
|
|
|
response = route53_client.list_resource_record_sets(HostedZoneId=hosted_zone["Id"])
|
|
|
|
# The 1st and 2nd records are NS and SOA records, respectively.
|
|
len(response["ResourceRecordSets"]).should.equal(3)
|
|
response["ResourceRecordSets"][2].should.equal(resource_record)
|
|
|
|
# a subsequest UPSERT with the same ChangeBatch should succeed as well
|
|
route53_client.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone["Id"],
|
|
ChangeBatch={
|
|
"Changes": [{"Action": "UPSERT", "ResourceRecordSet": resource_record}],
|
|
},
|
|
)
|
|
response = route53_client.list_resource_record_sets(HostedZoneId=hosted_zone["Id"])
|
|
|
|
# The 1st and 2nd records are NS and SOA records, respectively.
|
|
len(response["ResourceRecordSets"]).should.equal(3)
|
|
response["ResourceRecordSets"][2].should.equal(resource_record)
|
|
|
|
|
|
@mock_route53
|
|
def test_list_resource_record_sets_name_type_filters():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
create_hosted_zone_response = conn.create_hosted_zone(
|
|
Name="db.",
|
|
CallerReference=str(hash("foo")),
|
|
HostedZoneConfig=dict(PrivateZone=False, Comment="db"),
|
|
)
|
|
hosted_zone_id = create_hosted_zone_response["HostedZone"]["Id"]
|
|
|
|
def create_resource_record_set(rec_type, rec_name):
|
|
payload = {
|
|
"Comment": f"create {rec_type} record {rec_name}",
|
|
"Changes": [
|
|
{
|
|
"Action": "CREATE",
|
|
"ResourceRecordSet": {
|
|
"Name": rec_name,
|
|
"Type": rec_type,
|
|
"TTL": 10,
|
|
"ResourceRecords": [{"Value": "127.0.0.1"}],
|
|
},
|
|
}
|
|
],
|
|
}
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id, ChangeBatch=payload
|
|
)
|
|
|
|
# record_type, record_name
|
|
all_records = [
|
|
("A", "a.a.db."),
|
|
("A", "a.b.db."),
|
|
("A", "b.b.db."),
|
|
("CNAME", "b.b.db."),
|
|
("CNAME", "b.c.db."),
|
|
("CNAME", "c.c.db."),
|
|
]
|
|
for record_type, record_name in all_records:
|
|
create_resource_record_set(record_type, record_name)
|
|
|
|
start_with = 2
|
|
response = conn.list_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id,
|
|
StartRecordType=all_records[start_with][0],
|
|
StartRecordName=all_records[start_with][1],
|
|
)
|
|
|
|
response["IsTruncated"].should.equal(False)
|
|
|
|
returned_records = [
|
|
(record["Type"], record["Name"]) for record in response["ResourceRecordSets"]
|
|
]
|
|
len(returned_records).should.equal(len(all_records) - start_with)
|
|
for desired_record in all_records[start_with:]:
|
|
returned_records.should.contain(desired_record)
|
|
|
|
|
|
@mock_route53
|
|
def test_get_change():
|
|
conn = boto3.client("route53", region_name="us-east-2")
|
|
|
|
change_id = "123456"
|
|
response = conn.get_change(Id=change_id)
|
|
|
|
response["ChangeInfo"]["Id"].should.equal(change_id)
|
|
response["ChangeInfo"]["Status"].should.equal("INSYNC")
|
|
|
|
|
|
@mock_route53
|
|
def test_change_resource_record_sets_records_limit():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
conn.create_hosted_zone(
|
|
Name="db.",
|
|
CallerReference=str(hash("foo")),
|
|
HostedZoneConfig=dict(PrivateZone=False, Comment="db"),
|
|
)
|
|
|
|
zones = conn.list_hosted_zones_by_name(DNSName="db.")
|
|
len(zones["HostedZones"]).should.equal(1)
|
|
zones["HostedZones"][0]["Name"].should.equal("db.")
|
|
hosted_zone_id = zones["HostedZones"][0]["Id"]
|
|
|
|
# Changes creating exactly 1,000 resource records.
|
|
changes = []
|
|
for ci in range(4):
|
|
resourcerecords = []
|
|
for rri in range(250):
|
|
resourcerecords.append({"Value": f"127.0.0.{rri}"})
|
|
changes.append(
|
|
{
|
|
"Action": "CREATE",
|
|
"ResourceRecordSet": {
|
|
"Name": f"foo{ci}.db.",
|
|
"Type": "A",
|
|
"TTL": 10,
|
|
"ResourceRecords": resourcerecords,
|
|
},
|
|
}
|
|
)
|
|
create_1000_resource_records_payload = {
|
|
"Comment": "Create four records with 250 resource records each",
|
|
"Changes": changes,
|
|
}
|
|
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id, ChangeBatch=create_1000_resource_records_payload
|
|
)
|
|
|
|
# Changes creating over 1,000 resource records.
|
|
too_many_changes = create_1000_resource_records_payload["Changes"].copy()
|
|
too_many_changes.append(
|
|
{
|
|
"Action": "CREATE",
|
|
"ResourceRecordSet": {
|
|
"Name": "toomany.db.",
|
|
"Type": "A",
|
|
"TTL": 10,
|
|
"ResourceRecords": [{"Value": "127.0.0.1"}],
|
|
},
|
|
}
|
|
)
|
|
|
|
create_1001_resource_records_payload = {
|
|
"Comment": "Create four records with 250 resource records each, plus one more",
|
|
"Changes": too_many_changes,
|
|
}
|
|
with pytest.raises(ClientError) as exc:
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id,
|
|
ChangeBatch=create_1001_resource_records_payload,
|
|
)
|
|
err = exc.value.response["Error"]
|
|
err["Code"].should.equal("InvalidChangeBatch")
|
|
|
|
# Changes upserting exactly 500 resource records.
|
|
changes = []
|
|
for ci in range(2):
|
|
resourcerecords = []
|
|
for rri in range(250):
|
|
resourcerecords.append({"Value": f"127.0.0.{rri}"})
|
|
changes.append(
|
|
{
|
|
"Action": "UPSERT",
|
|
"ResourceRecordSet": {
|
|
"Name": f"foo{ci}.db.",
|
|
"Type": "A",
|
|
"TTL": 10,
|
|
"ResourceRecords": resourcerecords,
|
|
},
|
|
}
|
|
)
|
|
upsert_500_resource_records_payload = {
|
|
"Comment": "Upsert two records with 250 resource records each",
|
|
"Changes": changes,
|
|
}
|
|
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id, ChangeBatch=upsert_500_resource_records_payload
|
|
)
|
|
|
|
# Changes upserting over 1,000 resource records.
|
|
too_many_changes = upsert_500_resource_records_payload["Changes"].copy()
|
|
too_many_changes.append(
|
|
{
|
|
"Action": "UPSERT",
|
|
"ResourceRecordSet": {
|
|
"Name": "toomany.db.",
|
|
"Type": "A",
|
|
"TTL": 10,
|
|
"ResourceRecords": [{"Value": "127.0.0.1"}],
|
|
},
|
|
}
|
|
)
|
|
|
|
upsert_501_resource_records_payload = {
|
|
"Comment": "Upsert two records with 250 resource records each, plus one more",
|
|
"Changes": too_many_changes,
|
|
}
|
|
|
|
with pytest.raises(ClientError) as exc:
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id, ChangeBatch=upsert_501_resource_records_payload
|
|
)
|
|
err = exc.value.response["Error"]
|
|
err["Code"].should.equal("InvalidChangeBatch")
|
|
err["Message"].should.equal("Number of records limit of 1000 exceeded.")
|
|
|
|
|
|
@mock_route53
|
|
def test_list_resource_recordset_pagination():
|
|
conn = boto3.client("route53", region_name="us-east-1")
|
|
conn.create_hosted_zone(
|
|
Name="db.",
|
|
CallerReference=str(hash("foo")),
|
|
HostedZoneConfig=dict(PrivateZone=True, Comment="db"),
|
|
)
|
|
|
|
zones = conn.list_hosted_zones_by_name(DNSName="db.")
|
|
len(zones["HostedZones"]).should.equal(1)
|
|
zones["HostedZones"][0]["Name"].should.equal("db.")
|
|
hosted_zone_id = zones["HostedZones"][0]["Id"]
|
|
|
|
# Create A Record.
|
|
a_record_endpoint_payload = {
|
|
"Comment": "Create 500 A records",
|
|
"Changes": [
|
|
{
|
|
"Action": "CREATE",
|
|
"ResourceRecordSet": {
|
|
"Name": f"env{idx}.redis.db.",
|
|
"Type": "A",
|
|
"TTL": 10,
|
|
"ResourceRecords": [{"Value": "127.0.0.1"}],
|
|
},
|
|
}
|
|
for idx in range(500)
|
|
],
|
|
}
|
|
conn.change_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id, ChangeBatch=a_record_endpoint_payload
|
|
)
|
|
|
|
response = conn.list_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id, MaxItems="100"
|
|
)
|
|
response.should.have.key("ResourceRecordSets").length_of(100)
|
|
response.should.have.key("IsTruncated").equals(True)
|
|
response.should.have.key("MaxItems").equals("100")
|
|
response.should.have.key("NextRecordName").equals("env187.redis.db.")
|
|
response.should.have.key("NextRecordType").equals("A")
|
|
|
|
response = conn.list_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id,
|
|
StartRecordName=response["NextRecordName"],
|
|
StartRecordType=response["NextRecordType"],
|
|
)
|
|
response.should.have.key("ResourceRecordSets").length_of(300)
|
|
response.should.have.key("IsTruncated").equals(True)
|
|
response.should.have.key("MaxItems").equals("300")
|
|
response.should.have.key("NextRecordName").equals("env457.redis.db.")
|
|
response.should.have.key("NextRecordType").equals("A")
|
|
|
|
response = conn.list_resource_record_sets(
|
|
HostedZoneId=hosted_zone_id,
|
|
StartRecordName=response["NextRecordName"],
|
|
StartRecordType=response["NextRecordType"],
|
|
)
|
|
response.should.have.key("ResourceRecordSets").length_of(102)
|
|
response.should.have.key("IsTruncated").equals(False)
|
|
response.should.have.key("MaxItems").equals("300")
|
|
response.shouldnt.have.key("NextRecordName")
|
|
response.shouldnt.have.key("NextRecordType")
|
|
|
|
|
|
@mock_route53
|
|
def test_get_dns_sec():
|
|
client = boto3.client("route53", region_name="us-east-1")
|
|
|
|
hosted_zone_id = client.create_hosted_zone(
|
|
Name="testdns.aws.com.", CallerReference=str(hash("foo"))
|
|
)["HostedZone"]["Id"]
|
|
dns_sec = client.get_dnssec(HostedZoneId=hosted_zone_id)
|
|
dns_sec.should.have.key("Status").equals({"ServeSignature": "NOT_SIGNING"})
|