313 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			313 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| import json
 | |
| 
 | |
| import boto3
 | |
| 
 | |
| from moto import mock_cloudformation, mock_ec2, mock_rds
 | |
| from tests.test_cloudformation.fixtures import (
 | |
|     rds_mysql_with_db_parameter_group,
 | |
|     rds_mysql_with_read_replica,
 | |
| )
 | |
| 
 | |
| 
 | |
| @mock_ec2
 | |
| @mock_rds
 | |
| @mock_cloudformation
 | |
| def test_create_subnetgroup_via_cf():
 | |
|     vpc_conn = boto3.client("ec2", "us-west-2")
 | |
|     vpc = vpc_conn.create_vpc(CidrBlock="10.0.0.0/16")["Vpc"]
 | |
|     subnet = vpc_conn.create_subnet(VpcId=vpc["VpcId"], CidrBlock="10.0.1.0/24")[
 | |
|         "Subnet"
 | |
|     ]
 | |
| 
 | |
|     rds = boto3.client("rds", region_name="us-west-2")
 | |
|     cf = boto3.client("cloudformation", region_name="us-west-2")
 | |
| 
 | |
|     template = {
 | |
|         "AWSTemplateFormatVersion": "2010-09-09",
 | |
|         "Resources": {
 | |
|             "subnet": {
 | |
|                 "Type": "AWS::RDS::DBSubnetGroup",
 | |
|                 "Properties": {
 | |
|                     "DBSubnetGroupName": "subnetgroupname",
 | |
|                     "DBSubnetGroupDescription": "subnetgroupdesc",
 | |
|                     "SubnetIds": [subnet["SubnetId"]],
 | |
|                 },
 | |
|             }
 | |
|         },
 | |
|     }
 | |
|     template_json = json.dumps(template)
 | |
|     cf.create_stack(StackName="test_stack", TemplateBody=template_json)
 | |
| 
 | |
|     response = rds.describe_db_subnet_groups()["DBSubnetGroups"]
 | |
|     assert len(response) == 1
 | |
| 
 | |
|     created_subnet = response[0]
 | |
|     assert created_subnet["DBSubnetGroupName"] == "subnetgroupname"
 | |
|     assert created_subnet["DBSubnetGroupDescription"] == "subnetgroupdesc"
 | |
|     assert created_subnet["VpcId"] == vpc["VpcId"]
 | |
| 
 | |
| 
 | |
| @mock_ec2
 | |
| @mock_rds
 | |
| @mock_cloudformation
 | |
| def test_create_dbinstance_via_cf():
 | |
|     vpc_conn = boto3.client("ec2", "us-west-2")
 | |
|     vpc = vpc_conn.create_vpc(CidrBlock="10.0.0.0/16")["Vpc"]
 | |
|     vpc_conn.create_subnet(VpcId=vpc["VpcId"], CidrBlock="10.0.1.0/24")
 | |
| 
 | |
|     rds = boto3.client("rds", region_name="us-west-2")
 | |
|     cf = boto3.client("cloudformation", region_name="us-west-2")
 | |
| 
 | |
|     template = {
 | |
|         "AWSTemplateFormatVersion": "2010-09-09",
 | |
|         "Resources": {
 | |
|             "db": {
 | |
|                 "Type": "AWS::RDS::DBInstance",
 | |
|                 "Properties": {
 | |
|                     "Port": 3307,
 | |
|                     "Engine": "mysql",
 | |
|                     # Required - throws exception when describing an instance without tags
 | |
|                     "Tags": [],
 | |
|                 },
 | |
|             }
 | |
|         },
 | |
|         "Outputs": {
 | |
|             "db_address": {"Value": {"Fn::GetAtt": ["db", "Endpoint.Address"]}},
 | |
|             "db_port": {"Value": {"Fn::GetAtt": ["db", "Endpoint.Port"]}},
 | |
|         },
 | |
|     }
 | |
|     template_json = json.dumps(template)
 | |
|     cf.create_stack(StackName="test_stack", TemplateBody=template_json)
 | |
| 
 | |
|     summaries = cf.list_stack_resources(StackName="test_stack")[
 | |
|         "StackResourceSummaries"
 | |
|     ]
 | |
| 
 | |
|     db_instance_identifier = summaries[0]["PhysicalResourceId"]
 | |
|     resp = rds.describe_db_instances()["DBInstances"]
 | |
|     assert len(resp) == 1
 | |
| 
 | |
|     created = resp[0]
 | |
|     assert created["DBInstanceIdentifier"] == db_instance_identifier
 | |
|     assert created["Engine"] == "mysql"
 | |
|     assert created["DBInstanceStatus"] == "available"
 | |
| 
 | |
|     # Verify the stack outputs are correct
 | |
|     o = _get_stack_outputs(cf, stack_name="test_stack")
 | |
|     assert o["db_address"] == (
 | |
|         f"{db_instance_identifier}.aaaaaaaaaa.us-west-2.rds.amazonaws.com"
 | |
|     )
 | |
|     assert o["db_port"] == "3307"
 | |
| 
 | |
| 
 | |
| @mock_ec2
 | |
| @mock_rds
 | |
| @mock_cloudformation
 | |
| def test_create_dbsecuritygroup_via_cf():
 | |
|     vpc_conn = boto3.client("ec2", "us-west-2")
 | |
|     vpc = vpc_conn.create_vpc(CidrBlock="10.0.0.0/16")["Vpc"]
 | |
|     vpc_conn.create_subnet(VpcId=vpc["VpcId"], CidrBlock="10.0.1.0/24")
 | |
| 
 | |
|     rds = boto3.client("rds", region_name="us-west-2")
 | |
|     cf = boto3.client("cloudformation", region_name="us-west-2")
 | |
| 
 | |
|     template = {
 | |
|         "AWSTemplateFormatVersion": "2010-09-09",
 | |
|         "Resources": {
 | |
|             "db": {
 | |
|                 "Type": "AWS::RDS::DBSecurityGroup",
 | |
|                 "Properties": {"GroupDescription": "my sec group"},
 | |
|             }
 | |
|         },
 | |
|     }
 | |
|     template_json = json.dumps(template)
 | |
|     cf.create_stack(StackName="test_stack", TemplateBody=template_json)
 | |
| 
 | |
|     result = rds.describe_db_security_groups()["DBSecurityGroups"]
 | |
|     assert len(result) == 1
 | |
| 
 | |
|     created = result[0]
 | |
|     assert created["DBSecurityGroupDescription"] == "my sec group"
 | |
| 
 | |
| 
 | |
| @mock_cloudformation
 | |
| @mock_ec2
 | |
| @mock_rds
 | |
| def test_rds_db_parameter_groups():
 | |
|     ec2_conn = boto3.client("ec2", region_name="us-west-1")
 | |
|     ec2_conn.create_security_group(
 | |
|         GroupName="application", Description="Our Application Group"
 | |
|     )
 | |
| 
 | |
|     template_json = json.dumps(rds_mysql_with_db_parameter_group.template)
 | |
|     cf_conn = boto3.client("cloudformation", "us-west-1")
 | |
|     cf_conn.create_stack(
 | |
|         StackName="test_stack",
 | |
|         TemplateBody=template_json,
 | |
|         Parameters=[
 | |
|             {"ParameterKey": key, "ParameterValue": value}
 | |
|             for key, value in [
 | |
|                 ("DBInstanceIdentifier", "master-db"),
 | |
|                 ("DBName", "my_db"),
 | |
|                 ("DBUser", "my_user"),
 | |
|                 ("DBPassword", "my_password"),
 | |
|                 ("DBAllocatedStorage", "20"),
 | |
|                 ("DBInstanceClass", "db.m1.medium"),
 | |
|                 ("EC2SecurityGroup", "application"),
 | |
|                 ("MultiAZ", "true"),
 | |
|             ]
 | |
|         ],
 | |
|     )
 | |
| 
 | |
|     rds_conn = boto3.client("rds", region_name="us-west-1")
 | |
| 
 | |
|     db_parameter_groups = rds_conn.describe_db_parameter_groups()
 | |
|     assert len(db_parameter_groups["DBParameterGroups"]) == 1
 | |
|     db_parameter_group_name = db_parameter_groups["DBParameterGroups"][0][
 | |
|         "DBParameterGroupName"
 | |
|     ]
 | |
| 
 | |
|     found_cloudformation_set_parameter = False
 | |
|     for db_parameter in rds_conn.describe_db_parameters(
 | |
|         DBParameterGroupName=db_parameter_group_name
 | |
|     )["Parameters"]:
 | |
|         if (
 | |
|             db_parameter["ParameterName"] == "BACKLOG_QUEUE_LIMIT"
 | |
|             and db_parameter["ParameterValue"] == "2048"
 | |
|         ):
 | |
|             found_cloudformation_set_parameter = True
 | |
| 
 | |
|     assert found_cloudformation_set_parameter is True
 | |
| 
 | |
| 
 | |
| @mock_cloudformation
 | |
| @mock_ec2
 | |
| @mock_rds
 | |
| def test_rds_mysql_with_read_replica():
 | |
|     ec2_conn = boto3.client("ec2", region_name="us-west-1")
 | |
|     ec2_conn.create_security_group(
 | |
|         GroupName="application", Description="Our Application Group"
 | |
|     )
 | |
| 
 | |
|     template_json = json.dumps(rds_mysql_with_read_replica.template)
 | |
|     cf = boto3.client("cloudformation", "us-west-1")
 | |
|     db_identifier = "master-db"
 | |
|     cf.create_stack(
 | |
|         StackName="test_stack",
 | |
|         TemplateBody=template_json,
 | |
|         Parameters=[
 | |
|             {"ParameterKey": "DBInstanceIdentifier", "ParameterValue": db_identifier},
 | |
|             {"ParameterKey": "DBName", "ParameterValue": "my_db"},
 | |
|             {"ParameterKey": "DBUser", "ParameterValue": "my_user"},
 | |
|             {"ParameterKey": "DBPassword", "ParameterValue": "my_password"},
 | |
|             {"ParameterKey": "DBAllocatedStorage", "ParameterValue": "20"},
 | |
|             {"ParameterKey": "DBInstanceClass", "ParameterValue": "db.m1.medium"},
 | |
|             {"ParameterKey": "EC2SecurityGroup", "ParameterValue": "application"},
 | |
|             {"ParameterKey": "MultiAZ", "ParameterValue": "true"},
 | |
|         ],
 | |
|     )
 | |
| 
 | |
|     rds = boto3.client("rds", region_name="us-west-1")
 | |
| 
 | |
|     primary = rds.describe_db_instances(DBInstanceIdentifier=db_identifier)[
 | |
|         "DBInstances"
 | |
|     ][0]
 | |
|     assert primary["MasterUsername"] == "my_user"
 | |
|     assert primary["AllocatedStorage"] == 20
 | |
|     assert primary["DBInstanceClass"] == "db.m1.medium"
 | |
|     assert primary["MultiAZ"]
 | |
|     assert len(primary["ReadReplicaDBInstanceIdentifiers"]) == 1
 | |
|     replica_id = primary["ReadReplicaDBInstanceIdentifiers"][0]
 | |
| 
 | |
|     replica = rds.describe_db_instances(DBInstanceIdentifier=replica_id)["DBInstances"][
 | |
|         0
 | |
|     ]
 | |
|     assert replica["DBInstanceClass"] == "db.m1.medium"
 | |
| 
 | |
|     security_group_name = primary["DBSecurityGroups"][0]["DBSecurityGroupName"]
 | |
|     security_group = rds.describe_db_security_groups(
 | |
|         DBSecurityGroupName=security_group_name
 | |
|     )["DBSecurityGroups"][0]
 | |
|     assert (
 | |
|         security_group["EC2SecurityGroups"][0]["EC2SecurityGroupName"] == "application"
 | |
|     )
 | |
| 
 | |
| 
 | |
| @mock_cloudformation
 | |
| @mock_ec2
 | |
| @mock_rds
 | |
| def test_rds_mysql_with_read_replica_in_vpc():
 | |
|     template_json = json.dumps(rds_mysql_with_read_replica.template)
 | |
|     cf = boto3.client("cloudformation", "eu-central-1")
 | |
|     db_identifier = "master-db"
 | |
|     cf.create_stack(
 | |
|         StackName="test_stack",
 | |
|         TemplateBody=template_json,
 | |
|         Parameters=[
 | |
|             {"ParameterKey": "DBInstanceIdentifier", "ParameterValue": db_identifier},
 | |
|             {"ParameterKey": "DBName", "ParameterValue": "my_db"},
 | |
|             {"ParameterKey": "DBUser", "ParameterValue": "my_user"},
 | |
|             {"ParameterKey": "DBPassword", "ParameterValue": "my_password"},
 | |
|             {"ParameterKey": "DBAllocatedStorage", "ParameterValue": "20"},
 | |
|             {"ParameterKey": "DBInstanceClass", "ParameterValue": "db.m1.medium"},
 | |
|             {"ParameterKey": "MultiAZ", "ParameterValue": "true"},
 | |
|         ],
 | |
|     )
 | |
| 
 | |
|     rds = boto3.client("rds", region_name="eu-central-1")
 | |
|     primary = rds.describe_db_instances(DBInstanceIdentifier=db_identifier)[
 | |
|         "DBInstances"
 | |
|     ][0]
 | |
| 
 | |
|     subnet_group_name = primary["DBSubnetGroup"]["DBSubnetGroupName"]
 | |
|     subnet_group = rds.describe_db_subnet_groups(DBSubnetGroupName=subnet_group_name)[
 | |
|         "DBSubnetGroups"
 | |
|     ][0]
 | |
|     assert subnet_group["DBSubnetGroupDescription"] == "my db subnet group"
 | |
| 
 | |
| 
 | |
| @mock_ec2
 | |
| @mock_rds
 | |
| @mock_cloudformation
 | |
| def test_delete_dbinstance_via_cf():
 | |
|     vpc_conn = boto3.client("ec2", "us-west-2")
 | |
|     vpc = vpc_conn.create_vpc(CidrBlock="10.0.0.0/16")["Vpc"]
 | |
|     vpc_conn.create_subnet(VpcId=vpc["VpcId"], CidrBlock="10.0.1.0/24")
 | |
| 
 | |
|     rds = boto3.client("rds", region_name="us-west-2")
 | |
|     cf = boto3.client("cloudformation", region_name="us-west-2")
 | |
| 
 | |
|     template = {
 | |
|         "AWSTemplateFormatVersion": "2010-09-09",
 | |
|         "Resources": {
 | |
|             "db": {
 | |
|                 "Type": "AWS::RDS::DBInstance",
 | |
|                 "Properties": {
 | |
|                     "Port": 3307,
 | |
|                     "Engine": "mysql",
 | |
|                     # Required - throws exception when describing an instance without tags
 | |
|                     "Tags": [],
 | |
|                 },
 | |
|             }
 | |
|         },
 | |
|     }
 | |
|     template_json = json.dumps(template)
 | |
|     cf.create_stack(StackName="test_stack", TemplateBody=template_json)
 | |
| 
 | |
|     resp = rds.describe_db_instances()["DBInstances"]
 | |
|     assert len(resp) == 1
 | |
| 
 | |
|     cf.delete_stack(StackName="test_stack")
 | |
| 
 | |
|     resp = rds.describe_db_instances()["DBInstances"]
 | |
|     assert len(resp) == 0
 | |
| 
 | |
| 
 | |
| def _get_stack_outputs(cf_client, stack_name):
 | |
|     """Returns the outputs for the first entry in describe_stacks."""
 | |
|     stack_description = cf_client.describe_stacks(StackName=stack_name)["Stacks"][0]
 | |
|     return {
 | |
|         output["OutputKey"]: output["OutputValue"]
 | |
|         for output in stack_description["Outputs"]
 | |
|     }
 |