From 284641e8cf620e77d81688d12df4190380d67753 Mon Sep 17 00:00:00 2001 From: Marco Rucci Date: Sun, 6 Dec 2015 20:58:54 +0100 Subject: [PATCH] Add support for default VPCs and subnets. The first VPC created will be elected as default. All subnets of the default VPC are considered default for their availability zone. --- moto/ec2/models.py | 15 ++++++++++----- moto/ec2/responses/subnets.py | 1 + moto/ec2/responses/vpcs.py | 1 + tests/test_ec2/test_subnets.py | 5 +++++ 4 files changed, 17 insertions(+), 5 deletions(-) diff --git a/moto/ec2/models.py b/moto/ec2/models.py index 798325881..f1ce042b8 100644 --- a/moto/ec2/models.py +++ b/moto/ec2/models.py @@ -1543,12 +1543,13 @@ class EBSBackend(object): class VPC(TaggedEC2Resource): - def __init__(self, ec2_backend, vpc_id, cidr_block): + def __init__(self, ec2_backend, vpc_id, cidr_block, is_default): self.ec2_backend = ec2_backend self.id = vpc_id self.cidr_block = cidr_block self.dhcp_options = None self.state = 'available' + self.is_default = is_default @classmethod def create_from_cloudformation_json(cls, resource_name, cloudformation_json, region_name): @@ -1590,7 +1591,7 @@ class VPCBackend(object): def create_vpc(self, cidr_block): vpc_id = random_vpc_id() - vpc = VPC(self, vpc_id, cidr_block) + vpc = VPC(self, vpc_id, cidr_block, len(self.vpcs) == 0) self.vpcs[vpc_id] = vpc # AWS creates a default main route table and security group. @@ -1734,12 +1735,13 @@ class VPCPeeringConnectionBackend(object): class Subnet(TaggedEC2Resource): - def __init__(self, ec2_backend, subnet_id, vpc_id, cidr_block, availability_zone): + def __init__(self, ec2_backend, subnet_id, vpc_id, cidr_block, availability_zone, defaultForAz): self.ec2_backend = ec2_backend self.id = subnet_id self.vpc_id = vpc_id self.cidr_block = cidr_block self._availability_zone = availability_zone + self.defaultForAz = defaultForAz @classmethod def create_from_cloudformation_json(cls, resource_name, cloudformation_json, region_name): @@ -1799,6 +1801,8 @@ class Subnet(TaggedEC2Resource): return self.id elif filter_name == 'availabilityZone': return self.availability_zone + elif filter_name == 'defaultForAz': + return self.defaultForAz filter_value = super(Subnet, self).get_filter_value(filter_name) @@ -1827,8 +1831,9 @@ class SubnetBackend(object): def create_subnet(self, vpc_id, cidr_block, availability_zone=None): subnet_id = random_subnet_id() - subnet = Subnet(self, subnet_id, vpc_id, cidr_block, availability_zone) - self.get_vpc(vpc_id) # Validate VPC exists + vpc = self.get_vpc(vpc_id) # Validate VPC exists + defaultForAz = "true" if vpc.is_default else "false" + subnet = Subnet(self, subnet_id, vpc_id, cidr_block, availability_zone, defaultForAz) # AWS associates a new subnet with the default Network ACL self.associate_default_network_acl_with_subnet(subnet_id) diff --git a/moto/ec2/responses/subnets.py b/moto/ec2/responses/subnets.py index a0798a615..d7579fa5d 100644 --- a/moto/ec2/responses/subnets.py +++ b/moto/ec2/responses/subnets.py @@ -73,6 +73,7 @@ DESCRIBE_SUBNETS_RESPONSE = """ {{ subnet.cidr_block }} 251 {{ subnet.availability_zone }} + {{ subnet.defaultForAz }} {% for tag in subnet.get_tags() %} diff --git a/moto/ec2/responses/vpcs.py b/moto/ec2/responses/vpcs.py index ddc148da3..b3ebbb740 100644 --- a/moto/ec2/responses/vpcs.py +++ b/moto/ec2/responses/vpcs.py @@ -57,6 +57,7 @@ DESCRIBE_VPCS_RESPONSE = """ {{ vpc.cidr_block }} dopt-7a8b9c2d default + {{ vpc.is_default }} {% for tag in vpc.get_tags() %} diff --git a/tests/test_ec2/test_subnets.py b/tests/test_ec2/test_subnets.py index 311e9b295..2c62bc6c7 100644 --- a/tests/test_ec2/test_subnets.py +++ b/tests/test_ec2/test_subnets.py @@ -115,6 +115,11 @@ def test_get_subnets_filtering(): subnets_by_az.should.have.length_of(1) set([subnet.id for subnet in subnets_by_az]).should.equal(set([subnetB1.id])) + # Filter by defaultForAz + subnets_by_az = conn.get_all_subnets(filters={'defaultForAz': "true"}) + subnets_by_az.should.have.length_of(1) + set([subnet.id for subnet in subnets_by_az]).should.equal(set([subnetA.id])) + # Unsupported filter conn.get_all_subnets.when.called_with(filters={'not-implemented-filter': 'foobar'}).should.throw(NotImplementedError)