Add cluster subnet group CRUD.
This commit is contained in:
parent
72a7946126
commit
ef3e5448ea
@ -1576,6 +1576,12 @@ class Subnet(TaggedEC2Resource):
|
|||||||
)
|
)
|
||||||
return subnet
|
return subnet
|
||||||
|
|
||||||
|
@property
|
||||||
|
def availability_zone(self):
|
||||||
|
# This could probably be smarter, but there doesn't appear to be a
|
||||||
|
# way to pull AZs for a region in boto
|
||||||
|
return self.ec2_backend.region_name + "a"
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def physical_resource_id(self):
|
def physical_resource_id(self):
|
||||||
return self.id
|
return self.id
|
||||||
@ -2435,6 +2441,15 @@ class EC2Backend(BaseBackend, InstanceBackend, TagBackend, AmiBackend,
|
|||||||
ElasticAddressBackend, KeyPairBackend, DHCPOptionsSetBackend,
|
ElasticAddressBackend, KeyPairBackend, DHCPOptionsSetBackend,
|
||||||
NetworkAclBackend):
|
NetworkAclBackend):
|
||||||
|
|
||||||
|
def __init__(self, region_name):
|
||||||
|
super(EC2Backend, self).__init__()
|
||||||
|
self.region_name = region_name
|
||||||
|
|
||||||
|
def reset(self):
|
||||||
|
region_name = self.region_name
|
||||||
|
self.__dict__ = {}
|
||||||
|
self.__init__(region_name)
|
||||||
|
|
||||||
# Use this to generate a proper error template response when in a response handler.
|
# Use this to generate a proper error template response when in a response handler.
|
||||||
def raise_error(self, code, message):
|
def raise_error(self, code, message):
|
||||||
raise EC2ClientError(code, message)
|
raise EC2ClientError(code, message)
|
||||||
@ -2488,4 +2503,4 @@ class EC2Backend(BaseBackend, InstanceBackend, TagBackend, AmiBackend,
|
|||||||
|
|
||||||
ec2_backends = {}
|
ec2_backends = {}
|
||||||
for region in boto.ec2.regions():
|
for region in boto.ec2.regions():
|
||||||
ec2_backends[region.name] = EC2Backend()
|
ec2_backends[region.name] = EC2Backend(region.name)
|
||||||
|
@ -22,3 +22,17 @@ class ClusterNotFoundError(RedshiftClientError):
|
|||||||
super(ClusterNotFoundError, self).__init__(
|
super(ClusterNotFoundError, self).__init__(
|
||||||
'ClusterNotFound',
|
'ClusterNotFound',
|
||||||
"Cluster {0} not found.".format(cluster_identifier))
|
"Cluster {0} not found.".format(cluster_identifier))
|
||||||
|
|
||||||
|
|
||||||
|
class ClusterSubnetGroupNotFound(RedshiftClientError):
|
||||||
|
def __init__(self, subnet_identifier):
|
||||||
|
super(ClusterSubnetGroupNotFound, self).__init__(
|
||||||
|
'ClusterSubnetGroupNotFound',
|
||||||
|
"Subnet group {0} not found.".format(subnet_identifier))
|
||||||
|
|
||||||
|
|
||||||
|
class InvalidSubnetError(RedshiftClientError):
|
||||||
|
def __init__(self, subnet_identifier):
|
||||||
|
super(InvalidSubnetError, self).__init__(
|
||||||
|
'InvalidSubnet',
|
||||||
|
"Subnet {0} not found.".format(subnet_identifier))
|
||||||
|
@ -2,7 +2,8 @@ from __future__ import unicode_literals
|
|||||||
|
|
||||||
import boto.redshift
|
import boto.redshift
|
||||||
from moto.core import BaseBackend
|
from moto.core import BaseBackend
|
||||||
from .exceptions import ClusterNotFoundError
|
from moto.ec2 import ec2_backends
|
||||||
|
from .exceptions import ClusterNotFoundError, ClusterSubnetGroupNotFound, InvalidSubnetError
|
||||||
|
|
||||||
|
|
||||||
class Cluster(object):
|
class Cluster(object):
|
||||||
@ -69,10 +70,51 @@ class Cluster(object):
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class SubnetGroup(object):
|
||||||
|
|
||||||
|
def __init__(self, ec2_backend, cluster_subnet_group_name, description, subnet_ids):
|
||||||
|
self.ec2_backend = ec2_backend
|
||||||
|
self.cluster_subnet_group_name = cluster_subnet_group_name
|
||||||
|
self.description = description
|
||||||
|
self.subnet_ids = subnet_ids
|
||||||
|
if not self.subnets:
|
||||||
|
raise InvalidSubnetError(subnet_ids)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def subnets(self):
|
||||||
|
return self.ec2_backend.get_all_subnets(filters={'subnet-id': self.subnet_ids})
|
||||||
|
|
||||||
|
@property
|
||||||
|
def vpc_id(self):
|
||||||
|
return self.subnets[0].vpc_id
|
||||||
|
|
||||||
|
def to_json(self):
|
||||||
|
return {
|
||||||
|
"VpcId": self.vpc_id,
|
||||||
|
"Description": self.description,
|
||||||
|
"ClusterSubnetGroupName": self.cluster_subnet_group_name,
|
||||||
|
"SubnetGroupStatus": "Complete",
|
||||||
|
"Subnets": [{
|
||||||
|
"SubnetStatus": "Active",
|
||||||
|
"SubnetIdentifier": subnet.id,
|
||||||
|
"SubnetAvailabilityZone": {
|
||||||
|
"Name": subnet.availability_zone
|
||||||
|
},
|
||||||
|
} for subnet in self.subnets],
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
class RedshiftBackend(BaseBackend):
|
class RedshiftBackend(BaseBackend):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self, ec2_backend):
|
||||||
self.clusters = {}
|
self.clusters = {}
|
||||||
|
self.subnet_groups = {}
|
||||||
|
self.ec2_backend = ec2_backend
|
||||||
|
|
||||||
|
def reset(self):
|
||||||
|
ec2_backend = self.ec2_backend
|
||||||
|
self.__dict__ = {}
|
||||||
|
self.__init__(ec2_backend)
|
||||||
|
|
||||||
def create_cluster(self, **cluster_kwargs):
|
def create_cluster(self, **cluster_kwargs):
|
||||||
cluster_identifier = cluster_kwargs['cluster_identifier']
|
cluster_identifier = cluster_kwargs['cluster_identifier']
|
||||||
@ -110,7 +152,26 @@ class RedshiftBackend(BaseBackend):
|
|||||||
return self.clusters.pop(cluster_identifier)
|
return self.clusters.pop(cluster_identifier)
|
||||||
raise ClusterNotFoundError(cluster_identifier)
|
raise ClusterNotFoundError(cluster_identifier)
|
||||||
|
|
||||||
|
def create_cluster_subnet_group(self, cluster_subnet_group_name, description, subnet_ids):
|
||||||
|
subnet_group = SubnetGroup(self.ec2_backend, cluster_subnet_group_name, description, subnet_ids)
|
||||||
|
self.subnet_groups[cluster_subnet_group_name] = subnet_group
|
||||||
|
return subnet_group
|
||||||
|
|
||||||
|
def describe_cluster_subnet_groups(self, subnet_identifier):
|
||||||
|
subnet_groups = self.subnet_groups.values()
|
||||||
|
if subnet_identifier:
|
||||||
|
if subnet_identifier in self.subnet_groups:
|
||||||
|
return [self.subnet_groups[subnet_identifier]]
|
||||||
|
else:
|
||||||
|
raise ClusterSubnetGroupNotFound(subnet_identifier)
|
||||||
|
return subnet_groups
|
||||||
|
|
||||||
|
def delete_cluster_subnet_group(self, subnet_identifier):
|
||||||
|
if subnet_identifier in self.subnet_groups:
|
||||||
|
return self.subnet_groups.pop(subnet_identifier)
|
||||||
|
raise ClusterSubnetGroupNotFound(subnet_identifier)
|
||||||
|
|
||||||
|
|
||||||
redshift_backends = {}
|
redshift_backends = {}
|
||||||
for region in boto.redshift.regions():
|
for region in boto.redshift.regions():
|
||||||
redshift_backends[region.name] = RedshiftBackend()
|
redshift_backends[region.name] = RedshiftBackend(ec2_backends[region.name])
|
||||||
|
@ -111,3 +111,52 @@ class RedshiftResponse(BaseResponse):
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
def create_cluster_subnet_group(self):
|
||||||
|
cluster_subnet_group_name = self._get_param('ClusterSubnetGroupName')
|
||||||
|
description = self._get_param('Description')
|
||||||
|
subnet_ids = self._get_multi_param('SubnetIds.member')
|
||||||
|
|
||||||
|
subnet_group = self.redshift_backend.create_cluster_subnet_group(
|
||||||
|
cluster_subnet_group_name=cluster_subnet_group_name,
|
||||||
|
description=description,
|
||||||
|
subnet_ids=subnet_ids,
|
||||||
|
)
|
||||||
|
|
||||||
|
return json.dumps({
|
||||||
|
"CreateClusterSubnetGroupResponse": {
|
||||||
|
"CreateClusterSubnetGroupResult": {
|
||||||
|
"ClusterSubnetGroup": subnet_group.to_json(),
|
||||||
|
},
|
||||||
|
"ResponseMetadata": {
|
||||||
|
"RequestId": "384ac68d-3775-11df-8963-01868b7c937a",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
def describe_cluster_subnet_groups(self):
|
||||||
|
subnet_identifier = self._get_param("ClusterSubnetGroupName")
|
||||||
|
subnet_groups = self.redshift_backend.describe_cluster_subnet_groups(subnet_identifier)
|
||||||
|
|
||||||
|
return json.dumps({
|
||||||
|
"DescribeClusterSubnetGroupsResponse": {
|
||||||
|
"DescribeClusterSubnetGroupsResult": {
|
||||||
|
"ClusterSubnetGroups": [subnet_group.to_json() for subnet_group in subnet_groups]
|
||||||
|
},
|
||||||
|
"ResponseMetadata": {
|
||||||
|
"RequestId": "384ac68d-3775-11df-8963-01868b7c937a",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
def delete_cluster_subnet_group(self):
|
||||||
|
subnet_identifier = self._get_param("ClusterSubnetGroupName")
|
||||||
|
self.redshift_backend.delete_cluster_subnet_group(subnet_identifier)
|
||||||
|
|
||||||
|
return json.dumps({
|
||||||
|
"DeleteClusterSubnetGroupResponse": {
|
||||||
|
"ResponseMetadata": {
|
||||||
|
"RequestId": "384ac68d-3775-11df-8963-01868b7c937a",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
@ -1,10 +1,10 @@
|
|||||||
from __future__ import unicode_literals
|
from __future__ import unicode_literals
|
||||||
|
|
||||||
import boto
|
import boto
|
||||||
from boto.redshift.exceptions import ClusterNotFound
|
from boto.redshift.exceptions import ClusterNotFound, ClusterSubnetGroupNotFound
|
||||||
import sure # noqa
|
import sure # noqa
|
||||||
|
|
||||||
from moto import mock_redshift
|
from moto import mock_ec2, mock_redshift
|
||||||
|
|
||||||
|
|
||||||
@mock_redshift
|
@mock_redshift
|
||||||
@ -21,7 +21,6 @@ def test_create_cluster():
|
|||||||
cluster_type="multi-node",
|
cluster_type="multi-node",
|
||||||
# cluster_security_groups=None,
|
# cluster_security_groups=None,
|
||||||
# vpc_security_group_ids=None,
|
# vpc_security_group_ids=None,
|
||||||
# cluster_subnet_group_name=None,
|
|
||||||
availability_zone="us-east-1d",
|
availability_zone="us-east-1d",
|
||||||
preferred_maintenance_window="Mon:03:00-Mon:11:00",
|
preferred_maintenance_window="Mon:03:00-Mon:11:00",
|
||||||
# cluster_parameter_group_name=None,
|
# cluster_parameter_group_name=None,
|
||||||
@ -94,7 +93,7 @@ def test_default_cluster_attibutes():
|
|||||||
cluster['DBName'].should.equal("dev")
|
cluster['DBName'].should.equal("dev")
|
||||||
# cluster['ClusterSecurityGroups'].should.equal([])
|
# cluster['ClusterSecurityGroups'].should.equal([])
|
||||||
# cluster['VpcSecurityGroups'].should.equal([])
|
# cluster['VpcSecurityGroups'].should.equal([])
|
||||||
# cluster['ClusterSubnetGroupName'].should.equal(None)
|
cluster['ClusterSubnetGroupName'].should.equal(None)
|
||||||
assert "us-east-" in cluster['AvailabilityZone']
|
assert "us-east-" in cluster['AvailabilityZone']
|
||||||
cluster['PreferredMaintenanceWindow'].should.equal("Mon:03:00-Mon:03:30")
|
cluster['PreferredMaintenanceWindow'].should.equal("Mon:03:00-Mon:03:30")
|
||||||
# cluster['ClusterParameterGroups'].should.equal([])
|
# cluster['ClusterParameterGroups'].should.equal([])
|
||||||
@ -105,6 +104,32 @@ def test_default_cluster_attibutes():
|
|||||||
cluster['NumberOfNodes'].should.equal(1)
|
cluster['NumberOfNodes'].should.equal(1)
|
||||||
|
|
||||||
|
|
||||||
|
@mock_redshift
|
||||||
|
@mock_ec2
|
||||||
|
def test_create_cluster_in_subnet_group():
|
||||||
|
vpc_conn = boto.connect_vpc()
|
||||||
|
vpc = vpc_conn.create_vpc("10.0.0.0/16")
|
||||||
|
subnet = vpc_conn.create_subnet(vpc.id, "10.0.0.0/24")
|
||||||
|
redshift_conn = boto.connect_redshift()
|
||||||
|
redshift_conn.create_cluster_subnet_group(
|
||||||
|
"my_subnet_group",
|
||||||
|
"This is my subnet group",
|
||||||
|
subnet_ids=[subnet.id],
|
||||||
|
)
|
||||||
|
|
||||||
|
redshift_conn.create_cluster(
|
||||||
|
"my_cluster",
|
||||||
|
node_type="dw.hs1.xlarge",
|
||||||
|
master_username="username",
|
||||||
|
master_user_password="password",
|
||||||
|
cluster_subnet_group_name='my_subnet_group',
|
||||||
|
)
|
||||||
|
|
||||||
|
cluster_response = redshift_conn.describe_clusters("my_cluster")
|
||||||
|
cluster = cluster_response['DescribeClustersResponse']['DescribeClustersResult']['Clusters'][0]
|
||||||
|
cluster['ClusterSubnetGroupName'].should.equal('my_subnet_group')
|
||||||
|
|
||||||
|
|
||||||
@mock_redshift
|
@mock_redshift
|
||||||
def test_describe_non_existant_cluster():
|
def test_describe_non_existant_cluster():
|
||||||
conn = boto.redshift.connect_to_region("us-east-1")
|
conn = boto.redshift.connect_to_region("us-east-1")
|
||||||
@ -169,9 +194,69 @@ def test_modify_cluster():
|
|||||||
cluster['NodeType'].should.equal("dw.hs1.xlarge")
|
cluster['NodeType'].should.equal("dw.hs1.xlarge")
|
||||||
# cluster['ClusterSecurityGroups'].should.equal([])
|
# cluster['ClusterSecurityGroups'].should.equal([])
|
||||||
# cluster['VpcSecurityGroups'].should.equal([])
|
# cluster['VpcSecurityGroups'].should.equal([])
|
||||||
# cluster['ClusterSubnetGroupName'].should.equal(None)
|
|
||||||
cluster['PreferredMaintenanceWindow'].should.equal("Tue:03:00-Tue:11:00")
|
cluster['PreferredMaintenanceWindow'].should.equal("Tue:03:00-Tue:11:00")
|
||||||
# cluster['ClusterParameterGroups'].should.equal([])
|
# cluster['ClusterParameterGroups'].should.equal([])
|
||||||
cluster['AutomatedSnapshotRetentionPeriod'].should.equal(7)
|
cluster['AutomatedSnapshotRetentionPeriod'].should.equal(7)
|
||||||
cluster['AllowVersionUpgrade'].should.equal(False)
|
cluster['AllowVersionUpgrade'].should.equal(False)
|
||||||
cluster['NumberOfNodes'].should.equal(2)
|
cluster['NumberOfNodes'].should.equal(2)
|
||||||
|
|
||||||
|
|
||||||
|
@mock_redshift
|
||||||
|
@mock_ec2
|
||||||
|
def test_create_cluster_subnet_group():
|
||||||
|
vpc_conn = boto.connect_vpc()
|
||||||
|
vpc = vpc_conn.create_vpc("10.0.0.0/16")
|
||||||
|
subnet1 = vpc_conn.create_subnet(vpc.id, "10.0.0.0/24")
|
||||||
|
subnet2 = vpc_conn.create_subnet(vpc.id, "10.0.1.0/24")
|
||||||
|
|
||||||
|
redshift_conn = boto.connect_redshift()
|
||||||
|
|
||||||
|
redshift_conn.create_cluster_subnet_group(
|
||||||
|
"my_subnet",
|
||||||
|
"This is my subnet group",
|
||||||
|
subnet_ids=[subnet1.id, subnet2.id],
|
||||||
|
)
|
||||||
|
|
||||||
|
list(redshift_conn.describe_cluster_subnet_groups()).should.have.length_of(1)
|
||||||
|
|
||||||
|
subnets_response = redshift_conn.describe_cluster_subnet_groups("my_subnet")
|
||||||
|
my_subnet = subnets_response['DescribeClusterSubnetGroupsResponse']['DescribeClusterSubnetGroupsResult']['ClusterSubnetGroups'][0]
|
||||||
|
|
||||||
|
my_subnet['ClusterSubnetGroupName'].should.equal("my_subnet")
|
||||||
|
my_subnet['Description'].should.equal("This is my subnet group")
|
||||||
|
subnet_ids = [subnet['SubnetIdentifier'] for subnet in my_subnet['Subnets']]
|
||||||
|
set(subnet_ids).should.equal(set([subnet1.id, subnet2.id]))
|
||||||
|
|
||||||
|
|
||||||
|
@mock_redshift
|
||||||
|
def test_describe_non_existant_subnet_group():
|
||||||
|
conn = boto.redshift.connect_to_region("us-east-1")
|
||||||
|
conn.describe_cluster_subnet_groups.when.called_with("not-a-subnet-group").should.throw(ClusterSubnetGroupNotFound)
|
||||||
|
|
||||||
|
|
||||||
|
@mock_redshift
|
||||||
|
@mock_ec2
|
||||||
|
def test_delete_cluster_subnet_group():
|
||||||
|
vpc_conn = boto.connect_vpc()
|
||||||
|
vpc = vpc_conn.create_vpc("10.0.0.0/16")
|
||||||
|
subnet = vpc_conn.create_subnet(vpc.id, "10.0.0.0/24")
|
||||||
|
redshift_conn = boto.connect_redshift()
|
||||||
|
|
||||||
|
redshift_conn.create_cluster_subnet_group(
|
||||||
|
"my_subnet",
|
||||||
|
"This is my subnet group",
|
||||||
|
subnet_ids=[subnet.id],
|
||||||
|
)
|
||||||
|
|
||||||
|
subnets_response = redshift_conn.describe_cluster_subnet_groups()
|
||||||
|
subnets = subnets_response['DescribeClusterSubnetGroupsResponse']['DescribeClusterSubnetGroupsResult']['ClusterSubnetGroups']
|
||||||
|
subnets.should.have.length_of(1)
|
||||||
|
|
||||||
|
redshift_conn.delete_cluster_subnet_group("my_subnet")
|
||||||
|
|
||||||
|
subnets_response = redshift_conn.describe_cluster_subnet_groups()
|
||||||
|
subnets = subnets_response['DescribeClusterSubnetGroupsResponse']['DescribeClusterSubnetGroupsResult']['ClusterSubnetGroups']
|
||||||
|
subnets.should.have.length_of(0)
|
||||||
|
|
||||||
|
# Delete invalid id
|
||||||
|
redshift_conn.describe_cluster_subnet_groups.when.called_with("not-a-subnet-group").should.throw(ClusterSubnetGroupNotFound)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user