2014-08-27 15:17:06 +00:00
|
|
|
from __future__ import unicode_literals
|
2019-10-31 15:44:26 +00:00
|
|
|
|
2014-08-25 22:09:38 +00:00
|
|
|
# Ensure 'assert_raises' context manager support for Python 2.6
|
2015-07-13 23:27:00 +00:00
|
|
|
import tests.backport_assert_raises # noqa
|
2014-08-25 22:09:38 +00:00
|
|
|
from nose.tools import assert_raises
|
|
|
|
|
2016-04-04 22:17:56 +00:00
|
|
|
import boto3
|
2013-02-22 04:13:01 +00:00
|
|
|
import boto
|
2015-04-08 19:05:20 +00:00
|
|
|
import boto.vpc
|
2013-03-06 03:53:53 +00:00
|
|
|
from boto.exception import EC2ResponseError
|
2019-05-25 17:35:07 +00:00
|
|
|
from botocore.exceptions import ParamValidationError, ClientError
|
2015-07-13 23:27:00 +00:00
|
|
|
import json
|
2013-08-03 21:21:25 +00:00
|
|
|
import sure # noqa
|
2019-11-08 16:40:17 +00:00
|
|
|
import random
|
2013-02-22 04:13:01 +00:00
|
|
|
|
2017-02-16 03:35:45 +00:00
|
|
|
from moto import mock_cloudformation_deprecated, mock_ec2, mock_ec2_deprecated
|
2013-02-22 04:13:01 +00:00
|
|
|
|
|
|
|
|
2017-02-16 03:35:45 +00:00
|
|
|
@mock_ec2_deprecated
|
2013-02-22 04:13:01 +00:00
|
|
|
def test_subnets():
|
2019-10-31 15:44:26 +00:00
|
|
|
ec2 = boto.connect_ec2("the_key", "the_secret")
|
|
|
|
conn = boto.connect_vpc("the_key", "the_secret")
|
2013-03-06 03:53:53 +00:00
|
|
|
vpc = conn.create_vpc("10.0.0.0/16")
|
|
|
|
subnet = conn.create_subnet(vpc.id, "10.0.0.0/18")
|
|
|
|
|
|
|
|
all_subnets = conn.get_all_subnets()
|
2017-01-12 01:37:57 +00:00
|
|
|
all_subnets.should.have.length_of(1 + len(ec2.get_all_zones()))
|
2013-03-06 03:53:53 +00:00
|
|
|
|
|
|
|
conn.delete_subnet(subnet.id)
|
|
|
|
|
|
|
|
all_subnets = conn.get_all_subnets()
|
2017-01-12 01:37:57 +00:00
|
|
|
all_subnets.should.have.length_of(0 + len(ec2.get_all_zones()))
|
2013-03-06 03:53:53 +00:00
|
|
|
|
2014-08-25 17:54:47 +00:00
|
|
|
with assert_raises(EC2ResponseError) as cm:
|
|
|
|
conn.delete_subnet(subnet.id)
|
2019-10-31 15:44:26 +00:00
|
|
|
cm.exception.code.should.equal("InvalidSubnetID.NotFound")
|
2014-08-25 17:54:47 +00:00
|
|
|
cm.exception.status.should.equal(400)
|
|
|
|
cm.exception.request_id.should_not.be.none
|
2014-05-11 23:00:28 +00:00
|
|
|
|
|
|
|
|
2017-02-16 03:35:45 +00:00
|
|
|
@mock_ec2_deprecated
|
2014-08-26 21:04:34 +00:00
|
|
|
def test_subnet_create_vpc_validation():
|
2019-10-31 15:44:26 +00:00
|
|
|
conn = boto.connect_vpc("the_key", "the_secret")
|
2014-08-26 21:04:34 +00:00
|
|
|
|
|
|
|
with assert_raises(EC2ResponseError) as cm:
|
|
|
|
conn.create_subnet("vpc-abcd1234", "10.0.0.0/18")
|
2019-10-31 15:44:26 +00:00
|
|
|
cm.exception.code.should.equal("InvalidVpcID.NotFound")
|
2014-08-26 21:04:34 +00:00
|
|
|
cm.exception.status.should.equal(400)
|
|
|
|
cm.exception.request_id.should_not.be.none
|
|
|
|
|
|
|
|
|
2017-02-16 03:35:45 +00:00
|
|
|
@mock_ec2_deprecated
|
2014-05-11 23:00:28 +00:00
|
|
|
def test_subnet_tagging():
|
2019-10-31 15:44:26 +00:00
|
|
|
conn = boto.connect_vpc("the_key", "the_secret")
|
2014-05-11 23:00:28 +00:00
|
|
|
vpc = conn.create_vpc("10.0.0.0/16")
|
|
|
|
subnet = conn.create_subnet(vpc.id, "10.0.0.0/18")
|
|
|
|
|
|
|
|
subnet.add_tag("a key", "some value")
|
|
|
|
|
|
|
|
tag = conn.get_all_tags()[0]
|
|
|
|
tag.name.should.equal("a key")
|
|
|
|
tag.value.should.equal("some value")
|
|
|
|
|
|
|
|
# Refresh the subnet
|
2017-01-12 01:37:57 +00:00
|
|
|
subnet = conn.get_all_subnets(subnet_ids=[subnet.id])[0]
|
2014-05-11 23:00:28 +00:00
|
|
|
subnet.tags.should.have.length_of(1)
|
|
|
|
subnet.tags["a key"].should.equal("some value")
|
2014-08-26 20:25:37 +00:00
|
|
|
|
|
|
|
|
2017-02-16 03:35:45 +00:00
|
|
|
@mock_ec2_deprecated
|
2015-04-08 20:47:43 +00:00
|
|
|
def test_subnet_should_have_proper_availability_zone_set():
|
2019-10-31 15:44:26 +00:00
|
|
|
conn = boto.vpc.connect_to_region("us-west-1")
|
2015-04-08 20:47:43 +00:00
|
|
|
vpcA = conn.create_vpc("10.0.0.0/16")
|
2019-10-31 15:44:26 +00:00
|
|
|
subnetA = conn.create_subnet(vpcA.id, "10.0.0.0/24", availability_zone="us-west-1b")
|
|
|
|
subnetA.availability_zone.should.equal("us-west-1b")
|
2015-04-08 20:47:43 +00:00
|
|
|
|
2016-05-12 20:36:09 +00:00
|
|
|
|
|
|
|
@mock_ec2
|
|
|
|
def test_default_subnet():
|
2019-10-31 15:44:26 +00:00
|
|
|
ec2 = boto3.resource("ec2", region_name="us-west-1")
|
2016-05-12 20:36:09 +00:00
|
|
|
|
2017-01-12 01:37:57 +00:00
|
|
|
default_vpc = list(ec2.vpcs.all())[0]
|
2019-10-31 15:44:26 +00:00
|
|
|
default_vpc.cidr_block.should.equal("172.31.0.0/16")
|
2016-05-12 20:36:09 +00:00
|
|
|
default_vpc.reload()
|
|
|
|
default_vpc.is_default.should.be.ok
|
|
|
|
|
2017-02-24 02:37:43 +00:00
|
|
|
subnet = ec2.create_subnet(
|
2019-10-31 15:44:26 +00:00
|
|
|
VpcId=default_vpc.id, CidrBlock="172.31.48.0/20", AvailabilityZone="us-west-1a"
|
|
|
|
)
|
2016-05-12 20:36:09 +00:00
|
|
|
subnet.reload()
|
2017-01-12 01:37:57 +00:00
|
|
|
subnet.map_public_ip_on_launch.shouldnt.be.ok
|
2016-05-12 20:36:09 +00:00
|
|
|
|
|
|
|
|
2017-02-16 03:35:45 +00:00
|
|
|
@mock_ec2_deprecated
|
2016-05-12 20:36:09 +00:00
|
|
|
def test_non_default_subnet():
|
2019-10-31 15:44:26 +00:00
|
|
|
vpc_cli = boto.vpc.connect_to_region("us-west-1")
|
2017-01-12 01:37:57 +00:00
|
|
|
|
|
|
|
# Create the non default VPC
|
|
|
|
vpc = vpc_cli.create_vpc("10.0.0.0/16")
|
|
|
|
vpc.is_default.shouldnt.be.ok
|
|
|
|
|
|
|
|
subnet = vpc_cli.create_subnet(vpc.id, "10.0.0.0/24")
|
|
|
|
subnet = vpc_cli.get_all_subnets(subnet_ids=[subnet.id])[0]
|
2019-10-31 15:44:26 +00:00
|
|
|
subnet.mapPublicIpOnLaunch.should.equal("false")
|
2017-01-12 01:37:57 +00:00
|
|
|
|
2016-05-12 20:36:09 +00:00
|
|
|
|
2017-01-12 01:37:57 +00:00
|
|
|
@mock_ec2
|
|
|
|
def test_boto3_non_default_subnet():
|
2019-10-31 15:44:26 +00:00
|
|
|
ec2 = boto3.resource("ec2", region_name="us-west-1")
|
2016-05-12 20:36:09 +00:00
|
|
|
|
|
|
|
# Create the non default VPC
|
2019-10-31 15:44:26 +00:00
|
|
|
vpc = ec2.create_vpc(CidrBlock="10.0.0.0/16")
|
2016-05-12 20:36:09 +00:00
|
|
|
vpc.reload()
|
|
|
|
vpc.is_default.shouldnt.be.ok
|
|
|
|
|
2017-02-24 02:37:43 +00:00
|
|
|
subnet = ec2.create_subnet(
|
2019-10-31 15:44:26 +00:00
|
|
|
VpcId=vpc.id, CidrBlock="10.0.0.0/24", AvailabilityZone="us-west-1a"
|
|
|
|
)
|
2016-05-12 20:36:09 +00:00
|
|
|
subnet.reload()
|
|
|
|
subnet.map_public_ip_on_launch.shouldnt.be.ok
|
|
|
|
|
|
|
|
|
2016-04-04 22:17:56 +00:00
|
|
|
@mock_ec2
|
2019-05-28 15:33:25 +00:00
|
|
|
def test_modify_subnet_attribute_public_ip_on_launch():
|
2019-10-31 15:44:26 +00:00
|
|
|
ec2 = boto3.resource("ec2", region_name="us-west-1")
|
|
|
|
client = boto3.client("ec2", region_name="us-west-1")
|
2016-05-12 20:36:09 +00:00
|
|
|
|
2017-01-12 01:37:57 +00:00
|
|
|
# Get the default VPC
|
|
|
|
vpc = list(ec2.vpcs.all())[0]
|
2016-05-12 20:36:09 +00:00
|
|
|
|
2017-02-24 02:37:43 +00:00
|
|
|
subnet = ec2.create_subnet(
|
2019-10-31 15:44:26 +00:00
|
|
|
VpcId=vpc.id, CidrBlock="172.31.48.0/20", AvailabilityZone="us-west-1a"
|
|
|
|
)
|
2016-04-04 22:17:56 +00:00
|
|
|
|
2016-05-12 20:36:09 +00:00
|
|
|
# 'map_public_ip_on_launch' is set when calling 'DescribeSubnets' action
|
|
|
|
subnet.reload()
|
|
|
|
|
|
|
|
# For non default subnet, attribute value should be 'False'
|
|
|
|
subnet.map_public_ip_on_launch.shouldnt.be.ok
|
2016-04-04 22:17:56 +00:00
|
|
|
|
2017-02-24 02:37:43 +00:00
|
|
|
client.modify_subnet_attribute(
|
2019-10-31 15:44:26 +00:00
|
|
|
SubnetId=subnet.id, MapPublicIpOnLaunch={"Value": False}
|
|
|
|
)
|
2016-04-04 22:17:56 +00:00
|
|
|
subnet.reload()
|
2017-01-12 01:37:57 +00:00
|
|
|
subnet.map_public_ip_on_launch.shouldnt.be.ok
|
2016-04-04 22:17:56 +00:00
|
|
|
|
2017-02-24 02:37:43 +00:00
|
|
|
client.modify_subnet_attribute(
|
2019-10-31 15:44:26 +00:00
|
|
|
SubnetId=subnet.id, MapPublicIpOnLaunch={"Value": True}
|
|
|
|
)
|
2017-01-12 01:37:57 +00:00
|
|
|
subnet.reload()
|
2016-05-12 20:36:09 +00:00
|
|
|
subnet.map_public_ip_on_launch.should.be.ok
|
2016-04-04 22:17:56 +00:00
|
|
|
|
2017-01-12 01:37:57 +00:00
|
|
|
|
2019-05-28 15:33:25 +00:00
|
|
|
@mock_ec2
|
|
|
|
def test_modify_subnet_attribute_assign_ipv6_address_on_creation():
|
2019-10-31 15:44:26 +00:00
|
|
|
ec2 = boto3.resource("ec2", region_name="us-west-1")
|
|
|
|
client = boto3.client("ec2", region_name="us-west-1")
|
2019-05-28 15:33:25 +00:00
|
|
|
|
|
|
|
# Get the default VPC
|
|
|
|
vpc = list(ec2.vpcs.all())[0]
|
|
|
|
|
|
|
|
subnet = ec2.create_subnet(
|
2019-10-31 15:44:26 +00:00
|
|
|
VpcId=vpc.id, CidrBlock="172.31.112.0/20", AvailabilityZone="us-west-1a"
|
|
|
|
)
|
2019-05-28 15:33:25 +00:00
|
|
|
|
|
|
|
# 'map_public_ip_on_launch' is set when calling 'DescribeSubnets' action
|
|
|
|
subnet.reload()
|
|
|
|
|
|
|
|
# For non default subnet, attribute value should be 'False'
|
|
|
|
subnet.assign_ipv6_address_on_creation.shouldnt.be.ok
|
|
|
|
|
|
|
|
client.modify_subnet_attribute(
|
2019-10-31 15:44:26 +00:00
|
|
|
SubnetId=subnet.id, AssignIpv6AddressOnCreation={"Value": False}
|
|
|
|
)
|
2019-05-28 15:33:25 +00:00
|
|
|
subnet.reload()
|
|
|
|
subnet.assign_ipv6_address_on_creation.shouldnt.be.ok
|
|
|
|
|
|
|
|
client.modify_subnet_attribute(
|
2019-10-31 15:44:26 +00:00
|
|
|
SubnetId=subnet.id, AssignIpv6AddressOnCreation={"Value": True}
|
|
|
|
)
|
2019-05-28 15:33:25 +00:00
|
|
|
subnet.reload()
|
|
|
|
subnet.assign_ipv6_address_on_creation.should.be.ok
|
|
|
|
|
|
|
|
|
2016-04-04 22:17:56 +00:00
|
|
|
@mock_ec2
|
|
|
|
def test_modify_subnet_attribute_validation():
|
2019-10-31 15:44:26 +00:00
|
|
|
ec2 = boto3.resource("ec2", region_name="us-west-1")
|
|
|
|
client = boto3.client("ec2", region_name="us-west-1")
|
|
|
|
vpc = ec2.create_vpc(CidrBlock="10.0.0.0/16")
|
2017-02-24 02:37:43 +00:00
|
|
|
subnet = ec2.create_subnet(
|
2019-10-31 15:44:26 +00:00
|
|
|
VpcId=vpc.id, CidrBlock="10.0.0.0/24", AvailabilityZone="us-west-1a"
|
|
|
|
)
|
2016-04-04 22:17:56 +00:00
|
|
|
|
|
|
|
with assert_raises(ParamValidationError):
|
2017-02-24 02:37:43 +00:00
|
|
|
client.modify_subnet_attribute(
|
2019-10-31 15:44:26 +00:00
|
|
|
SubnetId=subnet.id, MapPublicIpOnLaunch={"Value": "invalid"}
|
|
|
|
)
|
2015-04-08 20:47:43 +00:00
|
|
|
|
2017-01-12 01:37:57 +00:00
|
|
|
|
2017-09-18 18:08:39 +00:00
|
|
|
@mock_ec2_deprecated
|
|
|
|
def test_subnet_get_by_id():
|
2019-10-31 15:44:26 +00:00
|
|
|
ec2 = boto.ec2.connect_to_region("us-west-1")
|
|
|
|
conn = boto.vpc.connect_to_region("us-west-1")
|
2017-09-18 18:08:39 +00:00
|
|
|
vpcA = conn.create_vpc("10.0.0.0/16")
|
2019-10-31 15:44:26 +00:00
|
|
|
subnetA = conn.create_subnet(vpcA.id, "10.0.0.0/24", availability_zone="us-west-1a")
|
2017-09-18 18:08:39 +00:00
|
|
|
vpcB = conn.create_vpc("10.0.0.0/16")
|
|
|
|
subnetB1 = conn.create_subnet(
|
2019-10-31 15:44:26 +00:00
|
|
|
vpcB.id, "10.0.0.0/24", availability_zone="us-west-1a"
|
|
|
|
)
|
2017-09-18 18:08:39 +00:00
|
|
|
subnetB2 = conn.create_subnet(
|
2019-10-31 15:44:26 +00:00
|
|
|
vpcB.id, "10.0.1.0/24", availability_zone="us-west-1b"
|
|
|
|
)
|
2017-09-18 18:08:39 +00:00
|
|
|
|
|
|
|
subnets_by_id = conn.get_all_subnets(subnet_ids=[subnetA.id, subnetB1.id])
|
|
|
|
subnets_by_id.should.have.length_of(2)
|
|
|
|
subnets_by_id = tuple(map(lambda s: s.id, subnets_by_id))
|
|
|
|
subnetA.id.should.be.within(subnets_by_id)
|
|
|
|
subnetB1.id.should.be.within(subnets_by_id)
|
|
|
|
|
|
|
|
with assert_raises(EC2ResponseError) as cm:
|
2019-10-31 15:44:26 +00:00
|
|
|
conn.get_all_subnets(subnet_ids=["subnet-does_not_exist"])
|
|
|
|
cm.exception.code.should.equal("InvalidSubnetID.NotFound")
|
2017-09-18 18:08:39 +00:00
|
|
|
cm.exception.status.should.equal(400)
|
|
|
|
cm.exception.request_id.should_not.be.none
|
|
|
|
|
|
|
|
|
2017-02-16 03:35:45 +00:00
|
|
|
@mock_ec2_deprecated
|
2014-08-26 20:25:37 +00:00
|
|
|
def test_get_subnets_filtering():
|
2019-10-31 15:44:26 +00:00
|
|
|
ec2 = boto.ec2.connect_to_region("us-west-1")
|
|
|
|
conn = boto.vpc.connect_to_region("us-west-1")
|
2014-08-26 20:25:37 +00:00
|
|
|
vpcA = conn.create_vpc("10.0.0.0/16")
|
2019-10-31 15:44:26 +00:00
|
|
|
subnetA = conn.create_subnet(vpcA.id, "10.0.0.0/24", availability_zone="us-west-1a")
|
2014-08-26 20:25:37 +00:00
|
|
|
vpcB = conn.create_vpc("10.0.0.0/16")
|
2017-02-24 02:37:43 +00:00
|
|
|
subnetB1 = conn.create_subnet(
|
2019-10-31 15:44:26 +00:00
|
|
|
vpcB.id, "10.0.0.0/24", availability_zone="us-west-1a"
|
|
|
|
)
|
2017-02-24 02:37:43 +00:00
|
|
|
subnetB2 = conn.create_subnet(
|
2019-10-31 15:44:26 +00:00
|
|
|
vpcB.id, "10.0.1.0/24", availability_zone="us-west-1b"
|
|
|
|
)
|
2014-08-26 20:25:37 +00:00
|
|
|
|
|
|
|
all_subnets = conn.get_all_subnets()
|
2017-01-12 01:37:57 +00:00
|
|
|
all_subnets.should.have.length_of(3 + len(ec2.get_all_zones()))
|
2014-08-26 20:25:37 +00:00
|
|
|
|
|
|
|
# Filter by VPC ID
|
2019-10-31 15:44:26 +00:00
|
|
|
subnets_by_vpc = conn.get_all_subnets(filters={"vpc-id": vpcB.id})
|
2014-08-26 20:25:37 +00:00
|
|
|
subnets_by_vpc.should.have.length_of(2)
|
2017-02-24 02:37:43 +00:00
|
|
|
set([subnet.id for subnet in subnets_by_vpc]).should.equal(
|
2019-10-31 15:44:26 +00:00
|
|
|
set([subnetB1.id, subnetB2.id])
|
|
|
|
)
|
2014-08-26 20:25:37 +00:00
|
|
|
|
|
|
|
# Filter by CIDR variations
|
2019-10-31 15:44:26 +00:00
|
|
|
subnets_by_cidr1 = conn.get_all_subnets(filters={"cidr": "10.0.0.0/24"})
|
2014-08-26 20:25:37 +00:00
|
|
|
subnets_by_cidr1.should.have.length_of(2)
|
2019-10-31 15:44:26 +00:00
|
|
|
set([subnet.id for subnet in subnets_by_cidr1]).should.equal(
|
|
|
|
set([subnetA.id, subnetB1.id])
|
|
|
|
)
|
2014-08-26 20:25:37 +00:00
|
|
|
|
2019-10-31 15:44:26 +00:00
|
|
|
subnets_by_cidr2 = conn.get_all_subnets(filters={"cidr-block": "10.0.0.0/24"})
|
2014-08-26 20:25:37 +00:00
|
|
|
subnets_by_cidr2.should.have.length_of(2)
|
2019-10-31 15:44:26 +00:00
|
|
|
set([subnet.id for subnet in subnets_by_cidr2]).should.equal(
|
|
|
|
set([subnetA.id, subnetB1.id])
|
|
|
|
)
|
2014-08-26 20:25:37 +00:00
|
|
|
|
2019-10-31 15:44:26 +00:00
|
|
|
subnets_by_cidr3 = conn.get_all_subnets(filters={"cidrBlock": "10.0.0.0/24"})
|
2014-08-26 20:25:37 +00:00
|
|
|
subnets_by_cidr3.should.have.length_of(2)
|
2019-10-31 15:44:26 +00:00
|
|
|
set([subnet.id for subnet in subnets_by_cidr3]).should.equal(
|
|
|
|
set([subnetA.id, subnetB1.id])
|
|
|
|
)
|
2014-08-26 20:25:37 +00:00
|
|
|
|
|
|
|
# Filter by VPC ID and CIDR
|
2017-02-24 02:37:43 +00:00
|
|
|
subnets_by_vpc_and_cidr = conn.get_all_subnets(
|
2019-10-31 15:44:26 +00:00
|
|
|
filters={"vpc-id": vpcB.id, "cidr": "10.0.0.0/24"}
|
|
|
|
)
|
2014-08-26 20:25:37 +00:00
|
|
|
subnets_by_vpc_and_cidr.should.have.length_of(1)
|
2019-10-31 15:44:26 +00:00
|
|
|
set([subnet.id for subnet in subnets_by_vpc_and_cidr]).should.equal(
|
|
|
|
set([subnetB1.id])
|
|
|
|
)
|
2014-08-26 20:25:37 +00:00
|
|
|
|
|
|
|
# Filter by subnet ID
|
2019-10-31 15:44:26 +00:00
|
|
|
subnets_by_id = conn.get_all_subnets(filters={"subnet-id": subnetA.id})
|
2014-08-26 20:25:37 +00:00
|
|
|
subnets_by_id.should.have.length_of(1)
|
|
|
|
set([subnet.id for subnet in subnets_by_id]).should.equal(set([subnetA.id]))
|
|
|
|
|
2015-04-08 19:05:20 +00:00
|
|
|
# Filter by availabilityZone
|
2017-02-24 02:37:43 +00:00
|
|
|
subnets_by_az = conn.get_all_subnets(
|
2019-10-31 15:44:26 +00:00
|
|
|
filters={"availabilityZone": "us-west-1a", "vpc-id": vpcB.id}
|
|
|
|
)
|
2015-04-08 19:05:20 +00:00
|
|
|
subnets_by_az.should.have.length_of(1)
|
2019-10-31 15:44:26 +00:00
|
|
|
set([subnet.id for subnet in subnets_by_az]).should.equal(set([subnetB1.id]))
|
2015-04-08 19:05:20 +00:00
|
|
|
|
2015-12-06 19:58:54 +00:00
|
|
|
# Filter by defaultForAz
|
2017-01-12 01:37:57 +00:00
|
|
|
|
2019-10-31 15:44:26 +00:00
|
|
|
subnets_by_az = conn.get_all_subnets(filters={"defaultForAz": "true"})
|
2017-01-12 01:37:57 +00:00
|
|
|
subnets_by_az.should.have.length_of(len(conn.get_all_zones()))
|
2015-12-06 19:58:54 +00:00
|
|
|
|
2014-08-26 20:25:37 +00:00
|
|
|
# Unsupported filter
|
2017-02-24 02:37:43 +00:00
|
|
|
conn.get_all_subnets.when.called_with(
|
2019-10-31 15:44:26 +00:00
|
|
|
filters={"not-implemented-filter": "foobar"}
|
|
|
|
).should.throw(NotImplementedError)
|
2015-07-13 23:27:00 +00:00
|
|
|
|
|
|
|
|
2017-02-16 03:35:45 +00:00
|
|
|
@mock_ec2_deprecated
|
|
|
|
@mock_cloudformation_deprecated
|
2015-07-13 23:27:00 +00:00
|
|
|
def test_subnet_tags_through_cloudformation():
|
2019-10-31 15:44:26 +00:00
|
|
|
vpc_conn = boto.vpc.connect_to_region("us-west-1")
|
2015-07-13 23:27:00 +00:00
|
|
|
vpc = vpc_conn.create_vpc("10.0.0.0/16")
|
|
|
|
|
|
|
|
subnet_template = {
|
|
|
|
"AWSTemplateFormatVersion": "2010-09-09",
|
|
|
|
"Resources": {
|
|
|
|
"testSubnet": {
|
|
|
|
"Type": "AWS::EC2::Subnet",
|
|
|
|
"Properties": {
|
|
|
|
"VpcId": vpc.id,
|
|
|
|
"CidrBlock": "10.0.0.0/24",
|
|
|
|
"AvailabilityZone": "us-west-1b",
|
2019-10-31 15:44:26 +00:00
|
|
|
"Tags": [
|
|
|
|
{"Key": "foo", "Value": "bar"},
|
|
|
|
{"Key": "blah", "Value": "baz"},
|
|
|
|
],
|
|
|
|
},
|
2015-07-13 23:27:00 +00:00
|
|
|
}
|
2019-10-31 15:44:26 +00:00
|
|
|
},
|
2015-07-13 23:27:00 +00:00
|
|
|
}
|
|
|
|
cf_conn = boto.cloudformation.connect_to_region("us-west-1")
|
|
|
|
template_json = json.dumps(subnet_template)
|
2019-10-31 15:44:26 +00:00
|
|
|
cf_conn.create_stack("test_stack", template_body=template_json)
|
2015-07-13 23:27:00 +00:00
|
|
|
|
2019-10-31 15:44:26 +00:00
|
|
|
subnet = vpc_conn.get_all_subnets(filters={"cidrBlock": "10.0.0.0/24"})[0]
|
2015-07-13 23:27:00 +00:00
|
|
|
subnet.tags["foo"].should.equal("bar")
|
|
|
|
subnet.tags["blah"].should.equal("baz")
|
2019-05-25 17:35:07 +00:00
|
|
|
|
|
|
|
|
2019-05-28 15:33:25 +00:00
|
|
|
@mock_ec2
|
|
|
|
def test_create_subnet_response_fields():
|
2019-10-31 15:44:26 +00:00
|
|
|
ec2 = boto3.resource("ec2", region_name="us-west-1")
|
|
|
|
client = boto3.client("ec2", region_name="us-west-1")
|
2019-05-28 15:33:25 +00:00
|
|
|
|
2019-10-31 15:44:26 +00:00
|
|
|
vpc = ec2.create_vpc(CidrBlock="10.0.0.0/16")
|
2019-05-28 15:33:25 +00:00
|
|
|
subnet = client.create_subnet(
|
2019-10-31 15:44:26 +00:00
|
|
|
VpcId=vpc.id, CidrBlock="10.0.0.0/24", AvailabilityZone="us-west-1a"
|
|
|
|
)["Subnet"]
|
|
|
|
|
|
|
|
subnet.should.have.key("AvailabilityZone")
|
|
|
|
subnet.should.have.key("AvailabilityZoneId")
|
|
|
|
subnet.should.have.key("AvailableIpAddressCount")
|
|
|
|
subnet.should.have.key("CidrBlock")
|
|
|
|
subnet.should.have.key("State")
|
|
|
|
subnet.should.have.key("SubnetId")
|
|
|
|
subnet.should.have.key("VpcId")
|
|
|
|
subnet.shouldnt.have.key("Tags")
|
|
|
|
subnet.should.have.key("DefaultForAz").which.should.equal(False)
|
|
|
|
subnet.should.have.key("MapPublicIpOnLaunch").which.should.equal(False)
|
|
|
|
subnet.should.have.key("OwnerId")
|
|
|
|
subnet.should.have.key("AssignIpv6AddressOnCreation").which.should.equal(False)
|
|
|
|
|
|
|
|
subnet_arn = "arn:aws:ec2:{region}:{owner_id}:subnet/{subnet_id}".format(
|
|
|
|
region=subnet["AvailabilityZone"][0:-1],
|
|
|
|
owner_id=subnet["OwnerId"],
|
|
|
|
subnet_id=subnet["SubnetId"],
|
|
|
|
)
|
|
|
|
subnet.should.have.key("SubnetArn").which.should.equal(subnet_arn)
|
|
|
|
subnet.should.have.key("Ipv6CidrBlockAssociationSet").which.should.equal([])
|
2019-05-28 15:33:25 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_ec2
|
|
|
|
def test_describe_subnet_response_fields():
|
2019-10-31 15:44:26 +00:00
|
|
|
ec2 = boto3.resource("ec2", region_name="us-west-1")
|
|
|
|
client = boto3.client("ec2", region_name="us-west-1")
|
2019-05-28 15:33:25 +00:00
|
|
|
|
2019-10-31 15:44:26 +00:00
|
|
|
vpc = ec2.create_vpc(CidrBlock="10.0.0.0/16")
|
2019-05-28 15:33:25 +00:00
|
|
|
subnet_object = ec2.create_subnet(
|
2019-10-31 15:44:26 +00:00
|
|
|
VpcId=vpc.id, CidrBlock="10.0.0.0/24", AvailabilityZone="us-west-1a"
|
|
|
|
)
|
2019-05-28 15:33:25 +00:00
|
|
|
|
2019-10-31 15:44:26 +00:00
|
|
|
subnets = client.describe_subnets(SubnetIds=[subnet_object.id])["Subnets"]
|
2019-05-28 15:33:25 +00:00
|
|
|
subnets.should.have.length_of(1)
|
|
|
|
subnet = subnets[0]
|
|
|
|
|
2019-10-31 15:44:26 +00:00
|
|
|
subnet.should.have.key("AvailabilityZone")
|
|
|
|
subnet.should.have.key("AvailabilityZoneId")
|
|
|
|
subnet.should.have.key("AvailableIpAddressCount")
|
|
|
|
subnet.should.have.key("CidrBlock")
|
|
|
|
subnet.should.have.key("State")
|
|
|
|
subnet.should.have.key("SubnetId")
|
|
|
|
subnet.should.have.key("VpcId")
|
|
|
|
subnet.shouldnt.have.key("Tags")
|
|
|
|
subnet.should.have.key("DefaultForAz").which.should.equal(False)
|
|
|
|
subnet.should.have.key("MapPublicIpOnLaunch").which.should.equal(False)
|
|
|
|
subnet.should.have.key("OwnerId")
|
|
|
|
subnet.should.have.key("AssignIpv6AddressOnCreation").which.should.equal(False)
|
|
|
|
|
|
|
|
subnet_arn = "arn:aws:ec2:{region}:{owner_id}:subnet/{subnet_id}".format(
|
|
|
|
region=subnet["AvailabilityZone"][0:-1],
|
|
|
|
owner_id=subnet["OwnerId"],
|
|
|
|
subnet_id=subnet["SubnetId"],
|
|
|
|
)
|
|
|
|
subnet.should.have.key("SubnetArn").which.should.equal(subnet_arn)
|
|
|
|
subnet.should.have.key("Ipv6CidrBlockAssociationSet").which.should.equal([])
|
2019-05-28 15:33:25 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_ec2
|
|
|
|
def test_create_subnet_with_invalid_availability_zone():
|
2019-10-31 15:44:26 +00:00
|
|
|
ec2 = boto3.resource("ec2", region_name="us-west-1")
|
|
|
|
client = boto3.client("ec2", region_name="us-west-1")
|
2019-05-28 15:33:25 +00:00
|
|
|
|
2019-10-31 15:44:26 +00:00
|
|
|
vpc = ec2.create_vpc(CidrBlock="10.0.0.0/16")
|
2019-05-28 15:33:25 +00:00
|
|
|
|
2019-10-31 15:44:26 +00:00
|
|
|
subnet_availability_zone = "asfasfas"
|
2019-05-28 15:33:25 +00:00
|
|
|
with assert_raises(ClientError) as ex:
|
|
|
|
subnet = client.create_subnet(
|
2019-10-31 15:44:26 +00:00
|
|
|
VpcId=vpc.id,
|
|
|
|
CidrBlock="10.0.0.0/24",
|
|
|
|
AvailabilityZone=subnet_availability_zone,
|
|
|
|
)
|
2019-05-28 15:33:25 +00:00
|
|
|
assert str(ex.exception).startswith(
|
|
|
|
"An error occurred (InvalidParameterValue) when calling the CreateSubnet "
|
2019-10-31 15:44:26 +00:00
|
|
|
"operation: Value ({}) for parameter availabilityZone is invalid. Subnets can currently only be created in the following availability zones: ".format(
|
|
|
|
subnet_availability_zone
|
|
|
|
)
|
|
|
|
)
|
2019-05-28 15:33:25 +00:00
|
|
|
|
|
|
|
|
2019-05-25 17:35:07 +00:00
|
|
|
@mock_ec2
|
|
|
|
def test_create_subnet_with_invalid_cidr_range():
|
2019-10-31 15:44:26 +00:00
|
|
|
ec2 = boto3.resource("ec2", region_name="us-west-1")
|
2019-05-25 17:35:07 +00:00
|
|
|
|
2019-10-31 15:44:26 +00:00
|
|
|
vpc = ec2.create_vpc(CidrBlock="10.0.0.0/16")
|
2019-05-25 17:35:07 +00:00
|
|
|
vpc.reload()
|
|
|
|
vpc.is_default.shouldnt.be.ok
|
|
|
|
|
2019-10-31 15:44:26 +00:00
|
|
|
subnet_cidr_block = "10.1.0.0/20"
|
2019-05-25 17:35:07 +00:00
|
|
|
with assert_raises(ClientError) as ex:
|
|
|
|
subnet = ec2.create_subnet(VpcId=vpc.id, CidrBlock=subnet_cidr_block)
|
|
|
|
str(ex.exception).should.equal(
|
|
|
|
"An error occurred (InvalidSubnet.Range) when calling the CreateSubnet "
|
2019-10-31 15:44:26 +00:00
|
|
|
"operation: The CIDR '{}' is invalid.".format(subnet_cidr_block)
|
|
|
|
)
|
2019-05-25 17:35:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_ec2
|
|
|
|
def test_create_subnet_with_invalid_cidr_block_parameter():
|
2019-10-31 15:44:26 +00:00
|
|
|
ec2 = boto3.resource("ec2", region_name="us-west-1")
|
2019-05-25 17:35:07 +00:00
|
|
|
|
2019-10-31 15:44:26 +00:00
|
|
|
vpc = ec2.create_vpc(CidrBlock="10.0.0.0/16")
|
2019-05-25 17:35:07 +00:00
|
|
|
vpc.reload()
|
|
|
|
vpc.is_default.shouldnt.be.ok
|
|
|
|
|
2019-10-31 15:44:26 +00:00
|
|
|
subnet_cidr_block = "1000.1.0.0/20"
|
2019-05-25 17:35:07 +00:00
|
|
|
with assert_raises(ClientError) as ex:
|
|
|
|
subnet = ec2.create_subnet(VpcId=vpc.id, CidrBlock=subnet_cidr_block)
|
|
|
|
str(ex.exception).should.equal(
|
|
|
|
"An error occurred (InvalidParameterValue) when calling the CreateSubnet "
|
2019-10-31 15:44:26 +00:00
|
|
|
"operation: Value ({}) for parameter cidrBlock is invalid. This is not a valid CIDR block.".format(
|
|
|
|
subnet_cidr_block
|
|
|
|
)
|
|
|
|
)
|
2019-05-25 17:35:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_ec2
|
|
|
|
def test_create_subnets_with_overlapping_cidr_blocks():
|
2019-10-31 15:44:26 +00:00
|
|
|
ec2 = boto3.resource("ec2", region_name="us-west-1")
|
2019-05-25 17:35:07 +00:00
|
|
|
|
2019-10-31 15:44:26 +00:00
|
|
|
vpc = ec2.create_vpc(CidrBlock="10.0.0.0/16")
|
2019-05-25 17:35:07 +00:00
|
|
|
vpc.reload()
|
|
|
|
vpc.is_default.shouldnt.be.ok
|
|
|
|
|
2019-10-31 15:44:26 +00:00
|
|
|
subnet_cidr_block = "10.0.0.0/24"
|
2019-05-25 17:35:07 +00:00
|
|
|
with assert_raises(ClientError) as ex:
|
|
|
|
subnet1 = ec2.create_subnet(VpcId=vpc.id, CidrBlock=subnet_cidr_block)
|
|
|
|
subnet2 = ec2.create_subnet(VpcId=vpc.id, CidrBlock=subnet_cidr_block)
|
|
|
|
str(ex.exception).should.equal(
|
|
|
|
"An error occurred (InvalidSubnet.Conflict) when calling the CreateSubnet "
|
2019-10-31 15:44:26 +00:00
|
|
|
"operation: The CIDR '{}' conflicts with another subnet".format(
|
|
|
|
subnet_cidr_block
|
|
|
|
)
|
|
|
|
)
|
2019-11-08 16:40:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_ec2
|
|
|
|
def test_available_ip_addresses_in_subnet():
|
|
|
|
ec2 = boto3.resource("ec2", region_name="us-west-1")
|
|
|
|
client = boto3.client("ec2", region_name="us-west-1")
|
|
|
|
|
|
|
|
vpc = ec2.create_vpc(CidrBlock="10.0.0.0/16")
|
|
|
|
cidr_range_addresses = [
|
|
|
|
("10.0.0.0/16", 65531),
|
|
|
|
("10.0.0.0/17", 32763),
|
|
|
|
("10.0.0.0/18", 16379),
|
|
|
|
("10.0.0.0/19", 8187),
|
|
|
|
("10.0.0.0/20", 4091),
|
|
|
|
("10.0.0.0/21", 2043),
|
|
|
|
("10.0.0.0/22", 1019),
|
|
|
|
("10.0.0.0/23", 507),
|
|
|
|
("10.0.0.0/24", 251),
|
|
|
|
("10.0.0.0/25", 123),
|
|
|
|
("10.0.0.0/26", 59),
|
|
|
|
("10.0.0.0/27", 27),
|
|
|
|
("10.0.0.0/28", 11),
|
|
|
|
]
|
|
|
|
for (cidr, expected_count) in cidr_range_addresses:
|
|
|
|
validate_subnet_details(client, vpc, cidr, expected_count)
|
|
|
|
|
|
|
|
|
|
|
|
@mock_ec2
|
|
|
|
def test_available_ip_addresses_in_subnet_with_enis():
|
|
|
|
ec2 = boto3.resource("ec2", region_name="us-west-1")
|
|
|
|
client = boto3.client("ec2", region_name="us-west-1")
|
|
|
|
|
|
|
|
vpc = ec2.create_vpc(CidrBlock="10.0.0.0/16")
|
|
|
|
# Verify behaviour for various CIDR ranges (...)
|
|
|
|
# Don't try to assign ENIs to /27 and /28, as there are not a lot of IP addresses to go around
|
|
|
|
cidr_range_addresses = [
|
|
|
|
("10.0.0.0/16", 65531),
|
|
|
|
("10.0.0.0/17", 32763),
|
|
|
|
("10.0.0.0/18", 16379),
|
|
|
|
("10.0.0.0/19", 8187),
|
|
|
|
("10.0.0.0/20", 4091),
|
|
|
|
("10.0.0.0/21", 2043),
|
|
|
|
("10.0.0.0/22", 1019),
|
|
|
|
("10.0.0.0/23", 507),
|
|
|
|
("10.0.0.0/24", 251),
|
|
|
|
("10.0.0.0/25", 123),
|
|
|
|
("10.0.0.0/26", 59),
|
|
|
|
]
|
|
|
|
for (cidr, expected_count) in cidr_range_addresses:
|
|
|
|
validate_subnet_details_after_creating_eni(client, vpc, cidr, expected_count)
|
|
|
|
|
|
|
|
|
|
|
|
def validate_subnet_details(client, vpc, cidr, expected_ip_address_count):
|
|
|
|
subnet = client.create_subnet(
|
|
|
|
VpcId=vpc.id, CidrBlock=cidr, AvailabilityZone="us-west-1b"
|
|
|
|
)["Subnet"]
|
|
|
|
subnet["AvailableIpAddressCount"].should.equal(expected_ip_address_count)
|
|
|
|
client.delete_subnet(SubnetId=subnet["SubnetId"])
|
|
|
|
|
|
|
|
|
|
|
|
def validate_subnet_details_after_creating_eni(
|
|
|
|
client, vpc, cidr, expected_ip_address_count
|
|
|
|
):
|
|
|
|
subnet = client.create_subnet(
|
|
|
|
VpcId=vpc.id, CidrBlock=cidr, AvailabilityZone="us-west-1b"
|
|
|
|
)["Subnet"]
|
|
|
|
# Create a random number of Elastic Network Interfaces
|
|
|
|
nr_of_eni_to_create = random.randint(0, 5)
|
|
|
|
ip_addresses_assigned = 0
|
|
|
|
enis_created = []
|
|
|
|
for i in range(0, nr_of_eni_to_create):
|
|
|
|
# Create a random number of IP addresses per ENI
|
|
|
|
nr_of_ip_addresses = random.randint(1, 5)
|
|
|
|
if nr_of_ip_addresses == 1:
|
|
|
|
# Pick the first available IP address (First 4 are reserved by AWS)
|
|
|
|
private_address = "10.0.0." + str(ip_addresses_assigned + 4)
|
|
|
|
eni = client.create_network_interface(
|
|
|
|
SubnetId=subnet["SubnetId"], PrivateIpAddress=private_address
|
|
|
|
)["NetworkInterface"]
|
|
|
|
enis_created.append(eni)
|
|
|
|
ip_addresses_assigned = ip_addresses_assigned + 1
|
|
|
|
else:
|
|
|
|
# Assign a list of IP addresses
|
|
|
|
private_addresses = [
|
|
|
|
"10.0.0." + str(4 + ip_addresses_assigned + i)
|
|
|
|
for i in range(0, nr_of_ip_addresses)
|
|
|
|
]
|
|
|
|
eni = client.create_network_interface(
|
|
|
|
SubnetId=subnet["SubnetId"],
|
|
|
|
PrivateIpAddresses=[
|
|
|
|
{"PrivateIpAddress": address} for address in private_addresses
|
|
|
|
],
|
|
|
|
)["NetworkInterface"]
|
|
|
|
enis_created.append(eni)
|
|
|
|
ip_addresses_assigned = ip_addresses_assigned + nr_of_ip_addresses + 1 #
|
|
|
|
# Verify that the nr of available IP addresses takes these ENIs into account
|
|
|
|
updated_subnet = client.describe_subnets(SubnetIds=[subnet["SubnetId"]])["Subnets"][
|
|
|
|
0
|
|
|
|
]
|
|
|
|
private_addresses = [
|
|
|
|
eni["PrivateIpAddress"] for eni in enis_created if eni["PrivateIpAddress"]
|
|
|
|
]
|
|
|
|
for eni in enis_created:
|
|
|
|
private_addresses.extend(
|
|
|
|
[address["PrivateIpAddress"] for address in eni["PrivateIpAddresses"]]
|
|
|
|
)
|
|
|
|
error_msg = (
|
|
|
|
"Nr of IP addresses for Subnet with CIDR {0} is incorrect. Expected: {1}, Actual: {2}. "
|
|
|
|
"Addresses: {3}"
|
|
|
|
)
|
|
|
|
with sure.ensure(
|
|
|
|
error_msg,
|
|
|
|
cidr,
|
|
|
|
str(expected_ip_address_count),
|
|
|
|
updated_subnet["AvailableIpAddressCount"],
|
|
|
|
str(private_addresses),
|
|
|
|
):
|
|
|
|
updated_subnet["AvailableIpAddressCount"].should.equal(
|
|
|
|
expected_ip_address_count - ip_addresses_assigned
|
|
|
|
)
|
|
|
|
# Clean up, as we have to create a few more subnets that shouldn't interfere with each other
|
|
|
|
for eni in enis_created:
|
|
|
|
client.delete_network_interface(NetworkInterfaceId=eni["NetworkInterfaceId"])
|
|
|
|
client.delete_subnet(SubnetId=subnet["SubnetId"])
|