Add cluster security groups.
This commit is contained in:
parent
ef3e5448ea
commit
25a31ee88a
@ -24,13 +24,20 @@ class ClusterNotFoundError(RedshiftClientError):
|
|||||||
"Cluster {0} not found.".format(cluster_identifier))
|
"Cluster {0} not found.".format(cluster_identifier))
|
||||||
|
|
||||||
|
|
||||||
class ClusterSubnetGroupNotFound(RedshiftClientError):
|
class ClusterSubnetGroupNotFoundError(RedshiftClientError):
|
||||||
def __init__(self, subnet_identifier):
|
def __init__(self, subnet_identifier):
|
||||||
super(ClusterSubnetGroupNotFound, self).__init__(
|
super(ClusterSubnetGroupNotFoundError, self).__init__(
|
||||||
'ClusterSubnetGroupNotFound',
|
'ClusterSubnetGroupNotFound',
|
||||||
"Subnet group {0} not found.".format(subnet_identifier))
|
"Subnet group {0} not found.".format(subnet_identifier))
|
||||||
|
|
||||||
|
|
||||||
|
class ClusterSecurityGroupNotFoundError(RedshiftClientError):
|
||||||
|
def __init__(self, group_identifier):
|
||||||
|
super(ClusterSecurityGroupNotFoundError, self).__init__(
|
||||||
|
'ClusterSecurityGroupNotFound',
|
||||||
|
"Security group {0} not found.".format(group_identifier))
|
||||||
|
|
||||||
|
|
||||||
class InvalidSubnetError(RedshiftClientError):
|
class InvalidSubnetError(RedshiftClientError):
|
||||||
def __init__(self, subnet_identifier):
|
def __init__(self, subnet_identifier):
|
||||||
super(InvalidSubnetError, self).__init__(
|
super(InvalidSubnetError, self).__init__(
|
||||||
|
@ -3,17 +3,23 @@ from __future__ import unicode_literals
|
|||||||
import boto.redshift
|
import boto.redshift
|
||||||
from moto.core import BaseBackend
|
from moto.core import BaseBackend
|
||||||
from moto.ec2 import ec2_backends
|
from moto.ec2 import ec2_backends
|
||||||
from .exceptions import ClusterNotFoundError, ClusterSubnetGroupNotFound, InvalidSubnetError
|
from .exceptions import (
|
||||||
|
ClusterNotFoundError,
|
||||||
|
ClusterSecurityGroupNotFoundError,
|
||||||
|
ClusterSubnetGroupNotFoundError,
|
||||||
|
InvalidSubnetError,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class Cluster(object):
|
class Cluster(object):
|
||||||
def __init__(self, cluster_identifier, node_type, master_username,
|
def __init__(self, redshift_backend, cluster_identifier, node_type, master_username,
|
||||||
master_user_password, db_name, cluster_type, cluster_security_groups,
|
master_user_password, db_name, cluster_type, cluster_security_groups,
|
||||||
vpc_security_group_ids, cluster_subnet_group_name, availability_zone,
|
vpc_security_group_ids, cluster_subnet_group_name, availability_zone,
|
||||||
preferred_maintenance_window, cluster_parameter_group_name,
|
preferred_maintenance_window, cluster_parameter_group_name,
|
||||||
automated_snapshot_retention_period, port, cluster_version,
|
automated_snapshot_retention_period, port, cluster_version,
|
||||||
allow_version_upgrade, number_of_nodes, publicly_accessible,
|
allow_version_upgrade, number_of_nodes, publicly_accessible,
|
||||||
encrypted, region):
|
encrypted, region):
|
||||||
|
self.redshift_backend = redshift_backend
|
||||||
self.cluster_identifier = cluster_identifier
|
self.cluster_identifier = cluster_identifier
|
||||||
self.node_type = node_type
|
self.node_type = node_type
|
||||||
self.master_username = master_username
|
self.master_username = master_username
|
||||||
@ -46,6 +52,14 @@ class Cluster(object):
|
|||||||
else:
|
else:
|
||||||
self.number_of_nodes = 1
|
self.number_of_nodes = 1
|
||||||
|
|
||||||
|
@property
|
||||||
|
def security_groups(self):
|
||||||
|
return [
|
||||||
|
security_group for security_group
|
||||||
|
in self.redshift_backend.describe_cluster_security_groups()
|
||||||
|
if security_group.cluster_security_group_name in self.cluster_security_groups
|
||||||
|
]
|
||||||
|
|
||||||
def to_json(self):
|
def to_json(self):
|
||||||
return {
|
return {
|
||||||
"MasterUsername": self.master_username,
|
"MasterUsername": self.master_username,
|
||||||
@ -62,7 +76,10 @@ class Cluster(object):
|
|||||||
"DBName": self.db_name,
|
"DBName": self.db_name,
|
||||||
"PreferredMaintenanceWindow": self.preferred_maintenance_window,
|
"PreferredMaintenanceWindow": self.preferred_maintenance_window,
|
||||||
"ClusterParameterGroups": [],
|
"ClusterParameterGroups": [],
|
||||||
"ClusterSecurityGroups": [],
|
"ClusterSecurityGroups": [{
|
||||||
|
"Status": "active",
|
||||||
|
"ClusterSecurityGroupName": group.cluster_security_group_name,
|
||||||
|
} for group in self.security_groups],
|
||||||
"Port": self.port,
|
"Port": self.port,
|
||||||
"NodeType": self.node_type,
|
"NodeType": self.node_type,
|
||||||
"ClusterIdentifier": self.cluster_identifier,
|
"ClusterIdentifier": self.cluster_identifier,
|
||||||
@ -104,11 +121,26 @@ class SubnetGroup(object):
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class SecurityGroup(object):
|
||||||
|
def __init__(self, cluster_security_group_name, description):
|
||||||
|
self.cluster_security_group_name = cluster_security_group_name
|
||||||
|
self.description = description
|
||||||
|
|
||||||
|
def to_json(self):
|
||||||
|
return {
|
||||||
|
"EC2SecurityGroups": [],
|
||||||
|
"IPRanges": [],
|
||||||
|
"Description": self.description,
|
||||||
|
"ClusterSecurityGroupName": self.cluster_security_group_name,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
class RedshiftBackend(BaseBackend):
|
class RedshiftBackend(BaseBackend):
|
||||||
|
|
||||||
def __init__(self, ec2_backend):
|
def __init__(self, ec2_backend):
|
||||||
self.clusters = {}
|
self.clusters = {}
|
||||||
self.subnet_groups = {}
|
self.subnet_groups = {}
|
||||||
|
self.security_groups = {}
|
||||||
self.ec2_backend = ec2_backend
|
self.ec2_backend = ec2_backend
|
||||||
|
|
||||||
def reset(self):
|
def reset(self):
|
||||||
@ -118,7 +150,7 @@ class RedshiftBackend(BaseBackend):
|
|||||||
|
|
||||||
def create_cluster(self, **cluster_kwargs):
|
def create_cluster(self, **cluster_kwargs):
|
||||||
cluster_identifier = cluster_kwargs['cluster_identifier']
|
cluster_identifier = cluster_kwargs['cluster_identifier']
|
||||||
cluster = Cluster(**cluster_kwargs)
|
cluster = Cluster(self, **cluster_kwargs)
|
||||||
self.clusters[cluster_identifier] = cluster
|
self.clusters[cluster_identifier] = cluster
|
||||||
return cluster
|
return cluster
|
||||||
|
|
||||||
@ -157,19 +189,38 @@ class RedshiftBackend(BaseBackend):
|
|||||||
self.subnet_groups[cluster_subnet_group_name] = subnet_group
|
self.subnet_groups[cluster_subnet_group_name] = subnet_group
|
||||||
return subnet_group
|
return subnet_group
|
||||||
|
|
||||||
def describe_cluster_subnet_groups(self, subnet_identifier):
|
def describe_cluster_subnet_groups(self, subnet_identifier=None):
|
||||||
subnet_groups = self.subnet_groups.values()
|
subnet_groups = self.subnet_groups.values()
|
||||||
if subnet_identifier:
|
if subnet_identifier:
|
||||||
if subnet_identifier in self.subnet_groups:
|
if subnet_identifier in self.subnet_groups:
|
||||||
return [self.subnet_groups[subnet_identifier]]
|
return [self.subnet_groups[subnet_identifier]]
|
||||||
else:
|
else:
|
||||||
raise ClusterSubnetGroupNotFound(subnet_identifier)
|
raise ClusterSubnetGroupNotFoundError(subnet_identifier)
|
||||||
return subnet_groups
|
return subnet_groups
|
||||||
|
|
||||||
def delete_cluster_subnet_group(self, subnet_identifier):
|
def delete_cluster_subnet_group(self, subnet_identifier):
|
||||||
if subnet_identifier in self.subnet_groups:
|
if subnet_identifier in self.subnet_groups:
|
||||||
return self.subnet_groups.pop(subnet_identifier)
|
return self.subnet_groups.pop(subnet_identifier)
|
||||||
raise ClusterSubnetGroupNotFound(subnet_identifier)
|
raise ClusterSubnetGroupNotFoundError(subnet_identifier)
|
||||||
|
|
||||||
|
def create_cluster_security_group(self, cluster_security_group_name, description):
|
||||||
|
security_group = SecurityGroup(cluster_security_group_name, description)
|
||||||
|
self.security_groups[cluster_security_group_name] = security_group
|
||||||
|
return security_group
|
||||||
|
|
||||||
|
def describe_cluster_security_groups(self, security_group_name=None):
|
||||||
|
security_groups = self.security_groups.values()
|
||||||
|
if security_group_name:
|
||||||
|
if security_group_name in self.security_groups:
|
||||||
|
return [self.security_groups[security_group_name]]
|
||||||
|
else:
|
||||||
|
raise ClusterSecurityGroupNotFoundError(security_group_name)
|
||||||
|
return security_groups
|
||||||
|
|
||||||
|
def delete_cluster_security_group(self, security_group_identifier):
|
||||||
|
if security_group_identifier in self.security_groups:
|
||||||
|
return self.security_groups.pop(security_group_identifier)
|
||||||
|
raise ClusterSecurityGroupNotFoundError(security_group_identifier)
|
||||||
|
|
||||||
|
|
||||||
redshift_backends = {}
|
redshift_backends = {}
|
||||||
|
@ -20,8 +20,8 @@ class RedshiftResponse(BaseResponse):
|
|||||||
"master_user_password": self._get_param('MasterUserPassword'),
|
"master_user_password": self._get_param('MasterUserPassword'),
|
||||||
"db_name": self._get_param('DBName'),
|
"db_name": self._get_param('DBName'),
|
||||||
"cluster_type": self._get_param('ClusterType'),
|
"cluster_type": self._get_param('ClusterType'),
|
||||||
"cluster_security_groups": self._get_multi_param('ClusterSecurityGroups'),
|
"cluster_security_groups": self._get_multi_param('ClusterSecurityGroups.member'),
|
||||||
"vpc_security_group_ids": self._get_multi_param('VpcSecurityGroupIds'),
|
"vpc_security_group_ids": self._get_multi_param('VpcSecurityGroupIds.member'),
|
||||||
"cluster_subnet_group_name": self._get_param('ClusterSubnetGroupName'),
|
"cluster_subnet_group_name": self._get_param('ClusterSubnetGroupName'),
|
||||||
"availability_zone": self._get_param('AvailabilityZone'),
|
"availability_zone": self._get_param('AvailabilityZone'),
|
||||||
"preferred_maintenance_window": self._get_param('PreferredMaintenanceWindow'),
|
"preferred_maintenance_window": self._get_param('PreferredMaintenanceWindow'),
|
||||||
@ -35,7 +35,6 @@ class RedshiftResponse(BaseResponse):
|
|||||||
"encrypted": self._get_param("Encrypted"),
|
"encrypted": self._get_param("Encrypted"),
|
||||||
"region": self.region,
|
"region": self.region,
|
||||||
}
|
}
|
||||||
|
|
||||||
cluster = self.redshift_backend.create_cluster(**cluster_kwargs)
|
cluster = self.redshift_backend.create_cluster(**cluster_kwargs)
|
||||||
|
|
||||||
return json.dumps({
|
return json.dumps({
|
||||||
@ -71,8 +70,8 @@ class RedshiftResponse(BaseResponse):
|
|||||||
"node_type": self._get_param('NodeType'),
|
"node_type": self._get_param('NodeType'),
|
||||||
"master_user_password": self._get_param('MasterUserPassword'),
|
"master_user_password": self._get_param('MasterUserPassword'),
|
||||||
"cluster_type": self._get_param('ClusterType'),
|
"cluster_type": self._get_param('ClusterType'),
|
||||||
"cluster_security_groups": self._get_multi_param('ClusterSecurityGroups'),
|
"cluster_security_groups": self._get_multi_param('ClusterSecurityGroups.member'),
|
||||||
"vpc_security_group_ids": self._get_multi_param('VpcSecurityGroupIds'),
|
"vpc_security_group_ids": self._get_multi_param('VpcSecurityGroupIds.member'),
|
||||||
"cluster_subnet_group_name": self._get_param('ClusterSubnetGroupName'),
|
"cluster_subnet_group_name": self._get_param('ClusterSubnetGroupName'),
|
||||||
"preferred_maintenance_window": self._get_param('PreferredMaintenanceWindow'),
|
"preferred_maintenance_window": self._get_param('PreferredMaintenanceWindow'),
|
||||||
"cluster_parameter_group_name": self._get_param('ClusterParameterGroupName'),
|
"cluster_parameter_group_name": self._get_param('ClusterParameterGroupName'),
|
||||||
@ -160,3 +159,50 @@ class RedshiftResponse(BaseResponse):
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
def create_cluster_security_group(self):
|
||||||
|
cluster_security_group_name = self._get_param('ClusterSecurityGroupName')
|
||||||
|
description = self._get_param('Description')
|
||||||
|
|
||||||
|
security_group = self.redshift_backend.create_cluster_security_group(
|
||||||
|
cluster_security_group_name=cluster_security_group_name,
|
||||||
|
description=description,
|
||||||
|
)
|
||||||
|
|
||||||
|
return json.dumps({
|
||||||
|
"CreateClusterSecurityGroupResponse": {
|
||||||
|
"CreateClusterSecurityGroupResult": {
|
||||||
|
"ClusterSecurityGroup": security_group.to_json(),
|
||||||
|
},
|
||||||
|
"ResponseMetadata": {
|
||||||
|
"RequestId": "384ac68d-3775-11df-8963-01868b7c937a",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
def describe_cluster_security_groups(self):
|
||||||
|
cluster_security_group_name = self._get_param("ClusterSecurityGroupName")
|
||||||
|
security_groups = self.redshift_backend.describe_cluster_security_groups(cluster_security_group_name)
|
||||||
|
|
||||||
|
return json.dumps({
|
||||||
|
"DescribeClusterSecurityGroupsResponse": {
|
||||||
|
"DescribeClusterSecurityGroupsResult": {
|
||||||
|
"ClusterSecurityGroups": [security_group.to_json() for security_group in security_groups]
|
||||||
|
},
|
||||||
|
"ResponseMetadata": {
|
||||||
|
"RequestId": "384ac68d-3775-11df-8963-01868b7c937a",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
def delete_cluster_security_group(self):
|
||||||
|
security_group_identifier = self._get_param("ClusterSecurityGroupName")
|
||||||
|
self.redshift_backend.delete_cluster_security_group(security_group_identifier)
|
||||||
|
|
||||||
|
return json.dumps({
|
||||||
|
"DeleteClusterSecurityGroupResponse": {
|
||||||
|
"ResponseMetadata": {
|
||||||
|
"RequestId": "384ac68d-3775-11df-8963-01868b7c937a",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
@ -1,7 +1,11 @@
|
|||||||
from __future__ import unicode_literals
|
from __future__ import unicode_literals
|
||||||
|
|
||||||
import boto
|
import boto
|
||||||
from boto.redshift.exceptions import ClusterNotFound, ClusterSubnetGroupNotFound
|
from boto.redshift.exceptions import (
|
||||||
|
ClusterNotFound,
|
||||||
|
ClusterSecurityGroupNotFound,
|
||||||
|
ClusterSubnetGroupNotFound,
|
||||||
|
)
|
||||||
import sure # noqa
|
import sure # noqa
|
||||||
|
|
||||||
from moto import mock_ec2, mock_redshift
|
from moto import mock_ec2, mock_redshift
|
||||||
@ -19,7 +23,6 @@ def test_create_cluster():
|
|||||||
master_user_password="password",
|
master_user_password="password",
|
||||||
db_name="my_db",
|
db_name="my_db",
|
||||||
cluster_type="multi-node",
|
cluster_type="multi-node",
|
||||||
# cluster_security_groups=None,
|
|
||||||
# vpc_security_group_ids=None,
|
# vpc_security_group_ids=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",
|
||||||
@ -130,6 +133,33 @@ def test_create_cluster_in_subnet_group():
|
|||||||
cluster['ClusterSubnetGroupName'].should.equal('my_subnet_group')
|
cluster['ClusterSubnetGroupName'].should.equal('my_subnet_group')
|
||||||
|
|
||||||
|
|
||||||
|
@mock_redshift
|
||||||
|
def test_create_cluster_with_security_group():
|
||||||
|
conn = boto.redshift.connect_to_region("us-east-1")
|
||||||
|
conn.create_cluster_security_group(
|
||||||
|
"security_group1",
|
||||||
|
"This is my security group",
|
||||||
|
)
|
||||||
|
conn.create_cluster_security_group(
|
||||||
|
"security_group2",
|
||||||
|
"This is my security group",
|
||||||
|
)
|
||||||
|
|
||||||
|
cluster_identifier = 'my_cluster'
|
||||||
|
conn.create_cluster(
|
||||||
|
cluster_identifier,
|
||||||
|
node_type="dw.hs1.xlarge",
|
||||||
|
master_username="username",
|
||||||
|
master_user_password="password",
|
||||||
|
cluster_security_groups=["security_group1", "security_group2"]
|
||||||
|
)
|
||||||
|
|
||||||
|
cluster_response = conn.describe_clusters(cluster_identifier)
|
||||||
|
cluster = cluster_response['DescribeClustersResponse']['DescribeClustersResult']['Clusters'][0]
|
||||||
|
group_names = [group['ClusterSecurityGroupName'] for group in cluster['ClusterSecurityGroups']]
|
||||||
|
set(group_names).should.equal(set(["security_group1", "security_group2"]))
|
||||||
|
|
||||||
|
|
||||||
@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")
|
||||||
@ -164,6 +194,10 @@ def test_delete_cluster():
|
|||||||
def test_modify_cluster():
|
def test_modify_cluster():
|
||||||
conn = boto.connect_redshift()
|
conn = boto.connect_redshift()
|
||||||
cluster_identifier = 'my_cluster'
|
cluster_identifier = 'my_cluster'
|
||||||
|
conn.create_cluster_security_group(
|
||||||
|
"security_group",
|
||||||
|
"This is my security group",
|
||||||
|
)
|
||||||
|
|
||||||
conn.create_cluster(
|
conn.create_cluster(
|
||||||
cluster_identifier,
|
cluster_identifier,
|
||||||
@ -177,7 +211,7 @@ def test_modify_cluster():
|
|||||||
cluster_type="multi-node",
|
cluster_type="multi-node",
|
||||||
node_type="dw.hs1.xlarge",
|
node_type="dw.hs1.xlarge",
|
||||||
number_of_nodes=2,
|
number_of_nodes=2,
|
||||||
# cluster_security_groups=None,
|
cluster_security_groups="security_group",
|
||||||
# vpc_security_group_ids=None,
|
# vpc_security_group_ids=None,
|
||||||
master_user_password="new_password",
|
master_user_password="new_password",
|
||||||
# cluster_parameter_group_name=None,
|
# cluster_parameter_group_name=None,
|
||||||
@ -192,7 +226,7 @@ def test_modify_cluster():
|
|||||||
|
|
||||||
cluster['ClusterIdentifier'].should.equal("new_identifier")
|
cluster['ClusterIdentifier'].should.equal("new_identifier")
|
||||||
cluster['NodeType'].should.equal("dw.hs1.xlarge")
|
cluster['NodeType'].should.equal("dw.hs1.xlarge")
|
||||||
# cluster['ClusterSecurityGroups'].should.equal([])
|
cluster['ClusterSecurityGroups'][0]['ClusterSecurityGroupName'].should.equal("security_group")
|
||||||
# cluster['VpcSecurityGroups'].should.equal([])
|
# cluster['VpcSecurityGroups'].should.equal([])
|
||||||
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([])
|
||||||
@ -217,8 +251,6 @@ def test_create_cluster_subnet_group():
|
|||||||
subnet_ids=[subnet1.id, subnet2.id],
|
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")
|
subnets_response = redshift_conn.describe_cluster_subnet_groups("my_subnet")
|
||||||
my_subnet = subnets_response['DescribeClusterSubnetGroupsResponse']['DescribeClusterSubnetGroupsResult']['ClusterSubnetGroups'][0]
|
my_subnet = subnets_response['DescribeClusterSubnetGroupsResponse']['DescribeClusterSubnetGroupsResult']['ClusterSubnetGroups'][0]
|
||||||
|
|
||||||
@ -259,4 +291,48 @@ def test_delete_cluster_subnet_group():
|
|||||||
subnets.should.have.length_of(0)
|
subnets.should.have.length_of(0)
|
||||||
|
|
||||||
# Delete invalid id
|
# Delete invalid id
|
||||||
redshift_conn.describe_cluster_subnet_groups.when.called_with("not-a-subnet-group").should.throw(ClusterSubnetGroupNotFound)
|
redshift_conn.delete_cluster_subnet_group.when.called_with("not-a-subnet-group").should.throw(ClusterSubnetGroupNotFound)
|
||||||
|
|
||||||
|
|
||||||
|
@mock_redshift
|
||||||
|
def test_create_cluster_security_group():
|
||||||
|
conn = boto.connect_redshift()
|
||||||
|
conn.create_cluster_security_group(
|
||||||
|
"my_security_group",
|
||||||
|
"This is my security group",
|
||||||
|
)
|
||||||
|
|
||||||
|
groups_response = conn.describe_cluster_security_groups("my_security_group")
|
||||||
|
my_group = groups_response['DescribeClusterSecurityGroupsResponse']['DescribeClusterSecurityGroupsResult']['ClusterSecurityGroups'][0]
|
||||||
|
|
||||||
|
my_group['ClusterSecurityGroupName'].should.equal("my_security_group")
|
||||||
|
my_group['Description'].should.equal("This is my security group")
|
||||||
|
list(my_group['IPRanges']).should.equal([])
|
||||||
|
|
||||||
|
|
||||||
|
@mock_redshift
|
||||||
|
def test_describe_non_existant_security_group():
|
||||||
|
conn = boto.redshift.connect_to_region("us-east-1")
|
||||||
|
conn.describe_cluster_security_groups.when.called_with("not-a-security-group").should.throw(ClusterSecurityGroupNotFound)
|
||||||
|
|
||||||
|
|
||||||
|
@mock_redshift
|
||||||
|
def test_delete_cluster_security_group():
|
||||||
|
conn = boto.connect_redshift()
|
||||||
|
conn.create_cluster_security_group(
|
||||||
|
"my_security_group",
|
||||||
|
"This is my security group",
|
||||||
|
)
|
||||||
|
|
||||||
|
groups_response = conn.describe_cluster_security_groups()
|
||||||
|
groups = groups_response['DescribeClusterSecurityGroupsResponse']['DescribeClusterSecurityGroupsResult']['ClusterSecurityGroups']
|
||||||
|
groups.should.have.length_of(1)
|
||||||
|
|
||||||
|
conn.delete_cluster_security_group("my_security_group")
|
||||||
|
|
||||||
|
groups_response = conn.describe_cluster_security_groups()
|
||||||
|
groups = groups_response['DescribeClusterSecurityGroupsResponse']['DescribeClusterSecurityGroupsResult']['ClusterSecurityGroups']
|
||||||
|
groups.should.have.length_of(0)
|
||||||
|
|
||||||
|
# Delete invalid id
|
||||||
|
conn.delete_cluster_security_group.when.called_with("not-a-security-group").should.throw(ClusterSecurityGroupNotFound)
|
||||||
|
Loading…
Reference in New Issue
Block a user