moto/tests/test_ec2/test_security_groups_cloudformation.py

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

285 lines
9.0 KiB
Python
Raw Normal View History

import boto3
import pytest
import json
from moto import mock_cloudformation, mock_ec2
from tests import EXAMPLE_AMI_ID
2021-10-05 17:11:07 +00:00
from string import Template
from uuid import uuid4
from botocore.exceptions import ClientError
2021-10-05 17:11:07 +00:00
SEC_GROUP_INGRESS = Template(
"""{
"AWSTemplateFormatVersion": "2010-09-09",
"Description": "AWS CloudFormation Template to create an EC2 instance",
"Parameters": {
"VPCId": {
"Type": "String",
"Description": "The VPC ID",
"AllowedPattern": "^vpc-[a-zA-Z0-9]*"
}
},
"Resources": {
"SecurityGroup": {
"Type": "AWS::EC2::SecurityGroup",
"Properties": {
"GroupDescription": "Test VPC security group",
2021-10-05 17:11:07 +00:00
"GroupName": $group_name,
"VpcId": {
"Ref": "VPCId"
}
}
},
"SSHIngressRule": {
"Type": "AWS::EC2::SecurityGroupIngress",
"Properties": {
"CidrIp": "10.0.0.0/8",
"Description": "Allow SSH traffic from 10.0.0.0/8",
"FromPort": 22,
"ToPort": 22,
"GroupId": {
"Fn::GetAtt": [
"SecurityGroup",
"GroupId"
]
},
"IpProtocol": "tcp"
}
}
}
}
"""
2021-10-05 17:11:07 +00:00
)
2021-10-05 17:11:07 +00:00
SEC_GROUP_INGRESS_WITHOUT_DESC = Template(
"""{
"AWSTemplateFormatVersion": "2010-09-09",
"Description": "AWS CloudFormation Template to create an EC2 instance",
"Parameters": {
"VPCId": {
"Type": "String",
"Description": "The VPC ID",
"AllowedPattern": "^vpc-[a-zA-Z0-9]*"
}
},
"Resources": {
"SecurityGroup": {
"Type": "AWS::EC2::SecurityGroup",
"Properties": {
"GroupDescription": "Test VPC security group",
2021-10-05 17:11:07 +00:00
"GroupName": "$group_name",
"VpcId": {
"Ref": "VPCId"
}
}
},
"SSHIngressRule": {
"Type": "AWS::EC2::SecurityGroupIngress",
"Properties": {
"CidrIp": "10.0.0.0/8",
"FromPort": 22,
"ToPort": 22,
"GroupId": {
"Fn::GetAtt": [
"SecurityGroup",
"GroupId"
]
},
"IpProtocol": "tcp"
}
}
}
}
"""
2021-10-05 17:11:07 +00:00
)
SEC_GROUP_SOURCE = {
"AWSTemplateFormatVersion": "2010-09-09",
"Resources": {
"my-security-group": {
"Type": "AWS::EC2::SecurityGroup",
"Properties": {"GroupDescription": "My other group"},
},
"Ec2Instance2": {
"Type": "AWS::EC2::Instance",
"Properties": {
"SecurityGroups": [{"Ref": "InstanceSecurityGroup"}],
"ImageId": EXAMPLE_AMI_ID,
},
},
"InstanceSecurityGroup": {
"Type": "AWS::EC2::SecurityGroup",
"Properties": {
"GroupDescription": "My security group",
"Tags": [{"Key": "bar", "Value": "baz"}],
"SecurityGroupIngress": [
{
"IpProtocol": "tcp",
"FromPort": "22",
"ToPort": "22",
"CidrIp": "123.123.123.123/32",
},
{
"IpProtocol": "tcp",
"FromPort": "80",
"ToPort": "8000",
"SourceSecurityGroupId": {"Ref": "my-security-group"},
},
],
},
},
},
}
@mock_cloudformation
@mock_ec2
def test_security_group_ingress():
cf_client = boto3.client("cloudformation", region_name="us-east-1")
ec2 = boto3.resource("ec2", region_name="us-west-1")
ec2_client = boto3.client("ec2", region_name="us-east-1")
2021-10-05 17:11:07 +00:00
group_name = str(uuid4())
vpc = ec2.create_vpc(CidrBlock="10.0.0.0/16")
cf_client.create_stack(
2021-10-05 17:11:07 +00:00
StackName=str(uuid4()),
TemplateBody=SEC_GROUP_INGRESS.substitute(group_name=group_name),
Parameters=[{"ParameterKey": "VPCId", "ParameterValue": vpc.id}],
Capabilities=["CAPABILITY_NAMED_IAM"],
OnFailure="DELETE",
)
groups = ec2_client.describe_security_groups()["SecurityGroups"]
2021-10-05 17:11:07 +00:00
group = [g for g in groups if g["GroupName"] == group_name][0]
assert group["Description"] == "Test VPC security group"
assert len(group["IpPermissions"]) == 1
ingress = group["IpPermissions"][0]
assert ingress["FromPort"] == 22
assert ingress["ToPort"] == 22
assert ingress["IpProtocol"] == "tcp"
assert ingress["IpRanges"] == [
{"CidrIp": "10.0.0.0/8", "Description": "Allow SSH traffic from 10.0.0.0/8"}
]
@mock_cloudformation
@mock_ec2
def test_delete_security_group_ingress():
cf_client = boto3.client("cloudformation", region_name="us-east-1")
ec2 = boto3.resource("ec2", region_name="us-west-1")
ec2_client = boto3.client("ec2", region_name="us-east-1")
stack_name = str(uuid4())
group_name = str(uuid4())
vpc = ec2.create_vpc(CidrBlock="10.0.0.0/16")
cf_client.create_stack(
StackName=stack_name,
TemplateBody=SEC_GROUP_INGRESS.substitute(group_name=group_name),
Parameters=[{"ParameterKey": "VPCId", "ParameterValue": vpc.id}],
Capabilities=["CAPABILITY_NAMED_IAM"],
OnFailure="DELETE",
)
sg_id = cf_client.list_stack_resources(StackName=stack_name)[
"StackResourceSummaries"
][0]["PhysicalResourceId"]
assert (
len(ec2_client.describe_security_groups(GroupIds=[sg_id])["SecurityGroups"])
== 1
)
cf_client.delete_stack(StackName=stack_name)
with pytest.raises(ClientError) as exc:
ec2_client.describe_security_groups(GroupIds=[sg_id])
assert exc.value.response["Error"]["Code"] == "InvalidGroup.NotFound"
@mock_cloudformation
@mock_ec2
def test_security_group_ingress_without_description():
cf_client = boto3.client("cloudformation", region_name="us-east-1")
ec2 = boto3.resource("ec2", region_name="us-west-1")
ec2_client = boto3.client("ec2", region_name="us-east-1")
2021-10-05 17:11:07 +00:00
group_name = str(uuid4())
vpc = ec2.create_vpc(CidrBlock="10.0.0.0/16")
cf_client.create_stack(
2021-10-05 17:11:07 +00:00
StackName=str(uuid4()),
TemplateBody=SEC_GROUP_INGRESS_WITHOUT_DESC.substitute(group_name=group_name),
Parameters=[{"ParameterKey": "VPCId", "ParameterValue": vpc.id}],
Capabilities=["CAPABILITY_NAMED_IAM"],
OnFailure="DELETE",
)
groups = ec2_client.describe_security_groups()["SecurityGroups"]
2021-10-05 17:11:07 +00:00
group = [g for g in groups if g["GroupName"] == group_name][0]
assert group["Description"] == "Test VPC security group"
assert len(group["IpPermissions"]) == 1
ingress = group["IpPermissions"][0]
assert ingress["IpRanges"] == [{"CidrIp": "10.0.0.0/8"}]
@mock_ec2
@mock_cloudformation
def test_stack_security_groups():
2021-10-05 17:11:07 +00:00
first_desc = str(uuid4())
second_desc = str(uuid4())
our_template = SEC_GROUP_SOURCE.copy()
our_template["Resources"]["my-security-group"]["Properties"][
"GroupDescription"
] = second_desc
our_template["Resources"]["InstanceSecurityGroup"]["Properties"][
"GroupDescription"
] = first_desc
template = json.dumps(our_template)
cf = boto3.client("cloudformation", region_name="us-west-1")
2021-10-05 17:11:07 +00:00
stack_name = str(uuid4())[0:6]
cf.create_stack(
2021-10-05 17:11:07 +00:00
StackName=stack_name,
TemplateBody=template,
Tags=[{"Key": "foo", "Value": "bar"}],
)
ec2 = boto3.client("ec2", region_name="us-west-1")
instance_group = ec2.describe_security_groups(
2021-10-05 17:11:07 +00:00
Filters=[{"Name": "description", "Values": [first_desc]}]
)["SecurityGroups"][0]
assert instance_group["Description"] == first_desc
assert "Tags" in instance_group
assert {"Key": "bar", "Value": "baz"} in instance_group["Tags"]
assert {"Key": "foo", "Value": "bar"} in instance_group["Tags"]
other_group = ec2.describe_security_groups(
2021-10-05 17:11:07 +00:00
Filters=[{"Name": "description", "Values": [second_desc]}]
)["SecurityGroups"][0]
2021-10-05 17:11:07 +00:00
instance = cf.list_stack_resources(StackName=stack_name)["StackResourceSummaries"][
1
]
instance_id = instance["PhysicalResourceId"]
ec2_instance = ec2.describe_instances(InstanceIds=[instance_id])["Reservations"][0][
"Instances"
][0]
assert (
ec2_instance["NetworkInterfaces"][0]["Groups"][0]["GroupId"]
== instance_group["GroupId"]
)
rule1, rule2 = instance_group["IpPermissions"]
assert int(rule1["ToPort"]) == 22
assert int(rule1["FromPort"]) == 22
assert rule1["IpRanges"][0]["CidrIp"] == "123.123.123.123/32"
assert rule1["IpProtocol"] == "tcp"
assert int(rule2["ToPort"]) == 8000
assert int(rule2["FromPort"]) == 80
assert rule2["IpProtocol"] == "tcp"
assert rule2["UserIdGroupPairs"][0]["GroupId"] == other_group["GroupId"]