2750 lines
98 KiB
Python
2750 lines
98 KiB
Python
import base64
|
|
import ipaddress
|
|
import json
|
|
import os
|
|
import warnings
|
|
from unittest import SkipTest, mock
|
|
from uuid import uuid4
|
|
|
|
import boto3
|
|
import pytest
|
|
from botocore.exceptions import ClientError, ParamValidationError
|
|
from freezegun import freeze_time
|
|
|
|
from moto import mock_ec2, mock_iam, settings
|
|
from moto.core import DEFAULT_ACCOUNT_ID as ACCOUNT_ID
|
|
from tests import EXAMPLE_AMI_ID
|
|
|
|
decode_method = base64.decodebytes
|
|
|
|
|
|
@mock_ec2
|
|
def test_add_servers():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
resp = client.run_instances(ImageId=EXAMPLE_AMI_ID, MinCount=2, MaxCount=2)
|
|
for i in resp["Instances"]:
|
|
assert i["ImageId"] == EXAMPLE_AMI_ID
|
|
|
|
instances = client.describe_instances(
|
|
InstanceIds=[i["InstanceId"] for i in resp["Instances"]]
|
|
)["Reservations"][0]["Instances"]
|
|
assert len(instances) == 2
|
|
for i in instances:
|
|
assert i["ImageId"] == EXAMPLE_AMI_ID
|
|
|
|
|
|
@freeze_time("2014-01-01 05:00:00")
|
|
@mock_ec2
|
|
def test_instance_launch_and_terminate():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
|
|
with pytest.raises(ClientError) as ex:
|
|
client.run_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, DryRun=True
|
|
)
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 412
|
|
assert ex.value.response["Error"]["Code"] == "DryRunOperation"
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "An error occurred (DryRunOperation) when calling the RunInstances operation: Request would have succeeded, but DryRun flag is set"
|
|
)
|
|
|
|
reservation = client.run_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)
|
|
assert len(reservation["Instances"]) == 1
|
|
instance = reservation["Instances"][0]
|
|
assert instance["State"] == {"Code": 0, "Name": "pending"}
|
|
instance_id = instance["InstanceId"]
|
|
|
|
reservations = client.describe_instances(InstanceIds=[instance_id])["Reservations"]
|
|
assert len(reservations) == 1
|
|
assert reservations[0]["ReservationId"] == reservation["ReservationId"]
|
|
instances = reservations[0]["Instances"]
|
|
assert len(instances) == 1
|
|
instance = instances[0]
|
|
assert instance["InstanceId"] == instance_id
|
|
assert instance["State"] == {"Code": 16, "Name": "running"}
|
|
if settings.TEST_SERVER_MODE:
|
|
# Exact value can't be determined in ServerMode
|
|
assert "LaunchTime" in instance
|
|
else:
|
|
launch_time = instance["LaunchTime"].strftime("%Y-%m-%dT%H:%M:%S.000Z")
|
|
assert launch_time == "2014-01-01T05:00:00.000Z"
|
|
assert instance["VpcId"] is not None
|
|
assert instance["Placement"]["AvailabilityZone"] == "us-east-1a"
|
|
|
|
root_device_name = instance["RootDeviceName"]
|
|
mapping = instance["BlockDeviceMappings"][0]
|
|
assert mapping["DeviceName"] == root_device_name
|
|
assert mapping["Ebs"]["Status"] == "in-use"
|
|
volume_id = mapping["Ebs"]["VolumeId"]
|
|
assert volume_id.startswith("vol-")
|
|
|
|
volume = client.describe_volumes(VolumeIds=[volume_id])["Volumes"][0]
|
|
assert volume["Attachments"][0]["InstanceId"] == instance_id
|
|
assert volume["State"] == "in-use"
|
|
|
|
with pytest.raises(ClientError) as ex:
|
|
client.terminate_instances(InstanceIds=[instance_id], DryRun=True)
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 412
|
|
assert ex.value.response["Error"]["Code"] == "DryRunOperation"
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "An error occurred (DryRunOperation) when calling the TerminateInstances operation: Request would have succeeded, but DryRun flag is set"
|
|
)
|
|
|
|
response = client.terminate_instances(InstanceIds=[instance_id])
|
|
assert len(response["TerminatingInstances"]) == 1
|
|
instance = response["TerminatingInstances"][0]
|
|
assert instance["InstanceId"] == instance_id
|
|
assert instance["PreviousState"] == {"Code": 16, "Name": "running"}
|
|
assert instance["CurrentState"] == {"Code": 32, "Name": "shutting-down"}
|
|
|
|
reservations = client.describe_instances(InstanceIds=[instance_id])["Reservations"]
|
|
instance = reservations[0]["Instances"][0]
|
|
assert instance["State"] == {"Code": 48, "Name": "terminated"}
|
|
|
|
|
|
@mock_ec2
|
|
def test_instance_terminate_discard_volumes():
|
|
ec2_resource = boto3.resource("ec2", "us-west-1")
|
|
|
|
result = ec2_resource.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
BlockDeviceMappings=[
|
|
{
|
|
"DeviceName": "/dev/sda1",
|
|
"Ebs": {"VolumeSize": 50, "DeleteOnTermination": True},
|
|
}
|
|
],
|
|
)
|
|
instance = result[0]
|
|
|
|
instance_volume_ids = []
|
|
for volume in instance.volumes.all():
|
|
instance_volume_ids.append(volume.volume_id)
|
|
|
|
instance.terminate()
|
|
instance.wait_until_terminated()
|
|
|
|
all_volumes_ids = [v.id for v in list(ec2_resource.volumes.all())]
|
|
for my_id in instance_volume_ids:
|
|
assert my_id not in all_volumes_ids
|
|
|
|
|
|
@mock_ec2
|
|
def test_instance_terminate_keep_volumes_explicit():
|
|
ec2_resource = boto3.resource("ec2", "us-west-1")
|
|
|
|
result = ec2_resource.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
BlockDeviceMappings=[
|
|
{
|
|
"DeviceName": "/dev/sda1",
|
|
"Ebs": {"VolumeSize": 50, "DeleteOnTermination": False},
|
|
}
|
|
],
|
|
)
|
|
instance = result[0]
|
|
|
|
instance_volume_ids = []
|
|
for volume in instance.volumes.all():
|
|
instance_volume_ids.append(volume.volume_id)
|
|
|
|
instance.terminate()
|
|
instance.wait_until_terminated()
|
|
|
|
all_volumes_ids = [v.id for v in list(ec2_resource.volumes.all())]
|
|
for my_id in instance_volume_ids:
|
|
assert my_id in all_volumes_ids
|
|
|
|
|
|
@mock_ec2
|
|
def test_instance_terminate_keep_volumes_implicit():
|
|
ec2_resource = boto3.resource("ec2", "us-west-1")
|
|
|
|
result = ec2_resource.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
BlockDeviceMappings=[{"DeviceName": "/dev/sda1", "Ebs": {"VolumeSize": 50}}],
|
|
)
|
|
instance = result[0]
|
|
|
|
instance_volume_ids = []
|
|
for volume in instance.volumes.all():
|
|
instance_volume_ids.append(volume.volume_id)
|
|
|
|
instance.terminate()
|
|
instance.wait_until_terminated()
|
|
|
|
assert len(instance_volume_ids) == 1
|
|
volume = ec2_resource.Volume(instance_volume_ids[0])
|
|
assert volume.state == "available"
|
|
|
|
|
|
@mock_ec2
|
|
def test_instance_terminate_detach_volumes():
|
|
ec2_resource = boto3.resource("ec2", "us-west-1")
|
|
result = ec2_resource.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
BlockDeviceMappings=[
|
|
{"DeviceName": "/dev/sda1", "Ebs": {"VolumeSize": 50}},
|
|
{"DeviceName": "/dev/sda2", "Ebs": {"VolumeSize": 50}},
|
|
],
|
|
)
|
|
instance = result[0]
|
|
my_volume_ids = []
|
|
for volume in instance.volumes.all():
|
|
my_volume_ids.append(volume.volume_id)
|
|
response = instance.detach_volume(VolumeId=volume.volume_id)
|
|
assert response["State"] == "detaching"
|
|
|
|
instance.terminate()
|
|
instance.wait_until_terminated()
|
|
|
|
all_volumes_ids = [v.id for v in list(ec2_resource.volumes.all())]
|
|
for my_id in my_volume_ids:
|
|
assert my_id in all_volumes_ids
|
|
|
|
|
|
@mock_ec2
|
|
def test_instance_detach_volume_wrong_path():
|
|
ec2_resource = boto3.resource("ec2", "us-west-1")
|
|
result = ec2_resource.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
BlockDeviceMappings=[{"DeviceName": "/dev/sda1", "Ebs": {"VolumeSize": 50}}],
|
|
)
|
|
instance = result[0]
|
|
for volume in instance.volumes.all():
|
|
with pytest.raises(ClientError) as ex:
|
|
instance.detach_volume(VolumeId=volume.volume_id, Device="/dev/sdf")
|
|
|
|
assert ex.value.response["Error"]["Code"] == "InvalidAttachment.NotFound"
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== f"The volume {volume.volume_id} is not attached to instance {instance.instance_id} as device /dev/sdf"
|
|
)
|
|
|
|
|
|
@mock_ec2
|
|
def test_terminate_empty_instances():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
|
|
with pytest.raises(ClientError) as ex:
|
|
client.terminate_instances(InstanceIds=[])
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
assert ex.value.response["Error"]["Code"] == "InvalidParameterCombination"
|
|
assert ex.value.response["Error"]["Message"] == "No instances specified"
|
|
|
|
|
|
@freeze_time("2014-01-01 05:00:00")
|
|
@mock_ec2
|
|
def test_instance_attach_volume():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
reservation = client.run_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)
|
|
instance = ec2.Instance(reservation["Instances"][0]["InstanceId"])
|
|
|
|
vol1 = ec2.create_volume(Size=36, AvailabilityZone="us-east-1a")
|
|
vol1.attach_to_instance(InstanceId=instance.id, Device="/dev/sda1")
|
|
vol2 = ec2.create_volume(Size=65, AvailabilityZone="us-east-1a")
|
|
vol2.attach_to_instance(InstanceId=instance.id, Device="/dev/sdb1")
|
|
vol3 = ec2.create_volume(Size=130, AvailabilityZone="us-east-1a")
|
|
vol3.attach_to_instance(InstanceId=instance.id, Device="/dev/sdc1")
|
|
|
|
instance.reload()
|
|
|
|
assert len(instance.block_device_mappings) == 3
|
|
expected_vol3_id = [
|
|
m["Ebs"]["VolumeId"]
|
|
for m in instance.block_device_mappings
|
|
if m["DeviceName"] == "/dev/sdc1"
|
|
][0]
|
|
|
|
expected_vol3 = ec2.Volume(expected_vol3_id)
|
|
assert expected_vol3.attachments[0]["InstanceId"] == instance.id
|
|
assert expected_vol3.availability_zone == "us-east-1a"
|
|
assert expected_vol3.state == "in-use"
|
|
if not settings.TEST_SERVER_MODE:
|
|
# FreezeTime does not work in ServerMode
|
|
assert expected_vol3.attachments[0]["AttachTime"] == instance.launch_time
|
|
assert expected_vol3.create_time == instance.launch_time
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instances_by_id():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
reservation = client.run_instances(ImageId=EXAMPLE_AMI_ID, MinCount=2, MaxCount=2)
|
|
instance1 = ec2.Instance(reservation["Instances"][0]["InstanceId"])
|
|
instance2 = ec2.Instance(reservation["Instances"][1]["InstanceId"])
|
|
|
|
reservations = client.describe_instances(InstanceIds=[instance1.id])["Reservations"]
|
|
assert len(reservations) == 1
|
|
reservation = reservations[0]
|
|
assert len(reservation["Instances"]) == 1
|
|
assert reservation["Instances"][0]["InstanceId"] == instance1.id
|
|
|
|
reservations = client.describe_instances(InstanceIds=[instance1.id, instance2.id])[
|
|
"Reservations"
|
|
]
|
|
assert len(reservations) == 1
|
|
reservation = reservations[0]
|
|
assert len(reservation["Instances"]) == 2
|
|
instance_ids = [instance["InstanceId"] for instance in reservation["Instances"]]
|
|
assert set(instance_ids) == set([instance1.id, instance2.id])
|
|
|
|
# Call describe_instances with a bad id should raise an error
|
|
with pytest.raises(ClientError) as ex:
|
|
client.describe_instances(InstanceIds=[instance1.id, "i-1234abcd"])
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
assert "RequestId" in ex.value.response["ResponseMetadata"]
|
|
assert ex.value.response["Error"]["Code"] == "InvalidInstanceID.NotFound"
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_paginated_instances():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
conn = boto3.resource("ec2", "us-east-1")
|
|
instances = []
|
|
for i in range(12):
|
|
instances.extend(
|
|
conn.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)
|
|
)
|
|
|
|
resp1 = client.describe_instances(MaxResults=5)
|
|
res1 = resp1["Reservations"]
|
|
assert len(res1) == 5
|
|
next_token = resp1["NextToken"]
|
|
|
|
assert next_token is not None
|
|
|
|
resp2 = client.describe_instances(NextToken=next_token)
|
|
|
|
# at least 12 total - 5 from the first call but there may be more from servermode tests
|
|
assert len(resp2["Reservations"]) >= 7
|
|
|
|
for i in instances:
|
|
i.terminate()
|
|
|
|
|
|
@mock_ec2
|
|
def test_describe_instances_pagination_error():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
|
|
# Call describe_instances with a bad id should raise an error
|
|
with pytest.raises(ClientError) as ex:
|
|
paginator = client.get_paginator("describe_instances").paginate(
|
|
InstanceIds=["i-12345678"],
|
|
PaginationConfig={
|
|
"MaxItems": 9999,
|
|
"PageSize": 100,
|
|
},
|
|
)
|
|
for page in paginator:
|
|
dir(page)
|
|
|
|
assert ex.value.response["Error"]["Code"] == "InvalidParameterCombination"
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "The parameter instancesSet cannot be used with the parameter maxResults"
|
|
)
|
|
|
|
|
|
@mock_ec2
|
|
def test_create_with_tags():
|
|
ec2 = boto3.client("ec2", region_name="us-west-2")
|
|
instances = ec2.run_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
InstanceType="t2.micro",
|
|
TagSpecifications=[
|
|
{
|
|
"ResourceType": "instance",
|
|
"Tags": [
|
|
{"Key": "MY_TAG1", "Value": "MY_VALUE1"},
|
|
{"Key": "MY_TAG2", "Value": "MY_VALUE2"},
|
|
],
|
|
},
|
|
{
|
|
"ResourceType": "instance",
|
|
"Tags": [{"Key": "MY_TAG3", "Value": "MY_VALUE3"}],
|
|
},
|
|
],
|
|
)
|
|
assert "Tags" in instances["Instances"][0]
|
|
assert len(instances["Instances"][0]["Tags"]) == 3
|
|
|
|
|
|
@mock_ec2
|
|
def test_create_with_volume_tags():
|
|
ec2 = boto3.client("ec2", region_name="us-west-2")
|
|
volume_tags = [
|
|
{"Key": "MY_TAG1", "Value": "MY_VALUE1"},
|
|
{"Key": "MY_TAG2", "Value": "MY_VALUE2"},
|
|
]
|
|
instances = ec2.run_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=2,
|
|
MaxCount=2,
|
|
InstanceType="t2.micro",
|
|
TagSpecifications=[{"ResourceType": "volume", "Tags": volume_tags}],
|
|
).get("Instances")
|
|
instance_ids = [i["InstanceId"] for i in instances]
|
|
instances = (
|
|
ec2.describe_instances(InstanceIds=instance_ids)
|
|
.get("Reservations")[0]
|
|
.get("Instances")
|
|
)
|
|
for instance in instances:
|
|
instance_volume = instance["BlockDeviceMappings"][0]["Ebs"]
|
|
resp = ec2.describe_volumes(VolumeIds=[instance_volume["VolumeId"]])
|
|
for volume in resp["Volumes"]:
|
|
assert sorted(volume["Tags"], key=lambda i: i["Key"]) == volume_tags
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instances_filtering_by_state():
|
|
ec2 = boto3.resource("ec2", "us-west-1")
|
|
client = boto3.client("ec2", "us-west-1")
|
|
reservation = ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=3, MaxCount=3)
|
|
instance1, instance2, instance3 = reservation
|
|
|
|
client.terminate_instances(InstanceIds=[instance1.id])
|
|
|
|
instances = retrieve_all_instances(
|
|
client, [{"Name": "instance-state-name", "Values": ["running"]}]
|
|
)
|
|
instance_ids = [i["InstanceId"] for i in instances]
|
|
# Since we terminated instance1, only instance2 and instance3 should be
|
|
# returned
|
|
assert instance1.id not in instance_ids
|
|
assert instance2.id in instance_ids
|
|
assert instance3.id in instance_ids
|
|
|
|
reservations = client.describe_instances(
|
|
InstanceIds=[instance2.id],
|
|
Filters=[{"Name": "instance-state-name", "Values": ["running"]}],
|
|
)["Reservations"]
|
|
assert len(reservations) == 1
|
|
instance_ids = [instance["InstanceId"] for instance in reservations[0]["Instances"]]
|
|
assert instance_ids == [instance2.id]
|
|
|
|
reservations = client.describe_instances(
|
|
InstanceIds=[instance2.id],
|
|
Filters=[{"Name": "instance-state-name", "Values": ["terminated"]}],
|
|
)["Reservations"]
|
|
assert reservations == []
|
|
|
|
# get_all_reservations should still return all 3
|
|
instances = retrieve_all_instances(client, filters=[])
|
|
instance_ids = [i["InstanceId"] for i in instances]
|
|
assert instance1.id in instance_ids
|
|
assert instance2.id in instance_ids
|
|
assert instance3.id in instance_ids
|
|
|
|
if not settings.TEST_SERVER_MODE:
|
|
# ServerMode will just throw a generic 500
|
|
filters = [{"Name": "not-implemented-filter", "Values": ["foobar"]}]
|
|
with pytest.raises(NotImplementedError):
|
|
client.describe_instances(Filters=filters)
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instances_filtering_by_instance_id():
|
|
ec2 = boto3.resource("ec2", "us-west-1")
|
|
client = boto3.client("ec2", "us-west-1")
|
|
reservation = ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=3, MaxCount=3)
|
|
instance1, instance2, _ = reservation
|
|
|
|
def _filter(values, exists=True):
|
|
f = [{"Name": "instance-id", "Values": values}]
|
|
r = client.describe_instances(Filters=f)["Reservations"]
|
|
if exists:
|
|
assert len(r[0]["Instances"]) == len(values)
|
|
found_ids = [i["InstanceId"] for i in r[0]["Instances"]]
|
|
assert set(found_ids) == set(values)
|
|
else:
|
|
assert len(r) == 0
|
|
|
|
_filter(values=[instance1.id])
|
|
_filter(values=[instance1.id, instance2.id])
|
|
_filter(values=["non-existing-id"], exists=False)
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instances_filtering_by_instance_type():
|
|
ec2 = boto3.resource("ec2", "us-west-1")
|
|
client = boto3.client("ec2", "us-west-1")
|
|
instance1 = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, InstanceType="m1.small"
|
|
)[0]
|
|
instance2 = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, InstanceType="m1.small"
|
|
)[0]
|
|
instance3 = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, InstanceType="t1.micro"
|
|
)[0]
|
|
|
|
instances = retrieve_all_instances(
|
|
client, [{"Name": "instance-type", "Values": ["m1.small"]}]
|
|
)
|
|
instance_ids = [i["InstanceId"] for i in instances]
|
|
assert instance1.id in set(instance_ids)
|
|
assert instance2.id in set(instance_ids)
|
|
|
|
instances = retrieve_all_instances(
|
|
client, [{"Name": "instance-type", "Values": ["t1.micro"]}]
|
|
)
|
|
instance_ids = [i["InstanceId"] for i in instances]
|
|
assert instance3.id in instance_ids
|
|
|
|
instances = retrieve_all_instances(
|
|
client, [{"Name": "instance-type", "Values": ["t1.micro", "m1.small"]}]
|
|
)
|
|
instance_ids = [i["InstanceId"] for i in instances]
|
|
assert instance1.id in instance_ids
|
|
assert instance2.id in instance_ids
|
|
assert instance3.id in instance_ids
|
|
|
|
res = client.describe_instances(
|
|
Filters=[{"Name": "instance-type", "Values": ["bogus"]}]
|
|
)
|
|
assert len(res["Reservations"]) == 0
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instances_filtering_by_reason_code():
|
|
ec2 = boto3.resource("ec2", "us-west-1")
|
|
client = boto3.client("ec2", "us-west-1")
|
|
reservation = ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=3, MaxCount=3)
|
|
instance1, instance2, instance3 = reservation
|
|
instance1.stop()
|
|
instance2.terminate()
|
|
|
|
filters = [
|
|
{"Name": "state-reason-code", "Values": ["Client.UserInitiatedShutdown"]}
|
|
]
|
|
instances = retrieve_all_instances(client, filters)
|
|
instance_ids = [i["InstanceId"] for i in instances]
|
|
|
|
assert instance1.id in instance_ids
|
|
assert instance2.id in instance_ids
|
|
assert instance3.id not in instance_ids
|
|
|
|
filters = [{"Name": "state-reason-code", "Values": [""]}]
|
|
instances = retrieve_all_instances(client, filters)
|
|
instance_ids = [i["InstanceId"] for i in instances]
|
|
assert instance3.id in instance_ids
|
|
assert instance1.id not in instance_ids
|
|
assert instance2.id not in instance_ids
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instances_filtering_by_source_dest_check():
|
|
ec2 = boto3.resource("ec2", "us-west-1")
|
|
client = boto3.client("ec2", "us-west-1")
|
|
reservation = ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=2, MaxCount=2)
|
|
instance1, instance2 = reservation
|
|
client.modify_instance_attribute(
|
|
InstanceId=instance1.id, SourceDestCheck={"Value": False}
|
|
)
|
|
|
|
instances_false = retrieve_all_instances(
|
|
client, [{"Name": "source-dest-check", "Values": ["false"]}]
|
|
)
|
|
instances_true = retrieve_all_instances(
|
|
client, [{"Name": "source-dest-check", "Values": ["true"]}]
|
|
)
|
|
|
|
assert instance1.id in [i["InstanceId"] for i in instances_false]
|
|
assert instance2.id not in [i["InstanceId"] for i in instances_false]
|
|
|
|
assert instance1.id not in [i["InstanceId"] for i in instances_true]
|
|
assert instance2.id in [i["InstanceId"] for i in instances_true]
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instances_filtering_by_vpc_id():
|
|
ec2 = boto3.resource("ec2", "us-west-1")
|
|
client = boto3.client("ec2", "us-west-1")
|
|
vpc1 = ec2.create_vpc(CidrBlock="10.0.0.0/16")
|
|
subnet1 = ec2.create_subnet(VpcId=vpc1.id, CidrBlock="10.0.0.0/27")
|
|
reservation1 = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, SubnetId=subnet1.id
|
|
)
|
|
instance1 = reservation1[0]
|
|
|
|
vpc2 = ec2.create_vpc(CidrBlock="10.1.0.0/16")
|
|
subnet2 = ec2.create_subnet(VpcId=vpc2.id, CidrBlock="10.1.0.0/27")
|
|
reservation2 = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, SubnetId=subnet2.id
|
|
)
|
|
instance2 = reservation2[0]
|
|
|
|
res1 = client.describe_instances(Filters=[{"Name": "vpc-id", "Values": [vpc1.id]}])[
|
|
"Reservations"
|
|
]
|
|
assert len(res1) == 1
|
|
assert len(res1[0]["Instances"]) == 1
|
|
assert res1[0]["Instances"][0]["InstanceId"] == instance1.id
|
|
assert res1[0]["Instances"][0]["VpcId"] == vpc1.id
|
|
assert res1[0]["Instances"][0]["SubnetId"] == subnet1.id
|
|
|
|
res2 = client.describe_instances(Filters=[{"Name": "vpc-id", "Values": [vpc2.id]}])[
|
|
"Reservations"
|
|
]
|
|
assert len(res2) == 1
|
|
assert len(res2[0]["Instances"]) == 1
|
|
assert res2[0]["Instances"][0]["InstanceId"] == instance2.id
|
|
assert res2[0]["Instances"][0]["VpcId"] == vpc2.id
|
|
assert res2[0]["Instances"][0]["SubnetId"] == subnet2.id
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instances_filtering_by_dns_name():
|
|
ec2 = boto3.resource("ec2", "us-west-1")
|
|
client = boto3.client("ec2", "us-west-1")
|
|
vpc1 = ec2.create_vpc(CidrBlock="10.0.0.0/16")
|
|
subnet = ec2.create_subnet(VpcId=vpc1.id, CidrBlock="10.0.0.0/27")
|
|
client.modify_subnet_attribute(
|
|
SubnetId=subnet.id, MapPublicIpOnLaunch={"Value": True}
|
|
)
|
|
reservation1 = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, SubnetId=subnet.id
|
|
)
|
|
instance1 = reservation1[0]
|
|
|
|
reservation2 = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, SubnetId=subnet.id
|
|
)
|
|
instance2 = reservation2[0]
|
|
|
|
res1 = client.describe_instances(
|
|
Filters=[{"Name": "dns-name", "Values": [instance1.public_dns_name]}]
|
|
)["Reservations"]
|
|
assert len(res1) == 1
|
|
assert len(res1[0]["Instances"]) == 1
|
|
assert res1[0]["Instances"][0]["InstanceId"] == instance1.id
|
|
|
|
res2 = client.describe_instances(
|
|
Filters=[{"Name": "dns-name", "Values": [instance2.public_dns_name]}]
|
|
)["Reservations"]
|
|
assert len(res2) == 1
|
|
assert len(res2[0]["Instances"]) == 1
|
|
assert res2[0]["Instances"][0]["InstanceId"] == instance2.id
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instances_filtering_by_architecture():
|
|
ec2 = boto3.resource("ec2", region_name="us-west-1")
|
|
client = boto3.client("ec2", region_name="us-west-1")
|
|
ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)
|
|
|
|
reservations = client.describe_instances(
|
|
Filters=[{"Name": "architecture", "Values": ["x86_64"]}]
|
|
)["Reservations"]
|
|
# get_all_reservations should return the instance
|
|
assert len(reservations[0]["Instances"]) == 1
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instances_filtering_by_image_id():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
conn = boto3.resource("ec2", "us-east-1")
|
|
conn.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)
|
|
|
|
reservations = client.describe_instances(
|
|
Filters=[{"Name": "image-id", "Values": [EXAMPLE_AMI_ID]}]
|
|
)["Reservations"]
|
|
assert len(reservations[0]["Instances"]) >= 1, "Should return just created instance"
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instances_filtering_by_account_id():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
conn = boto3.resource("ec2", "us-east-1")
|
|
instance = conn.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)[0]
|
|
|
|
instances = retrieve_all_instances(
|
|
client, filters=[{"Name": "owner-id", "Values": [ACCOUNT_ID]}]
|
|
)
|
|
|
|
assert instance.id in [i["InstanceId"] for i in instances]
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instances_filtering_by_private_dns():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
conn = boto3.resource("ec2", "us-east-1")
|
|
conn.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, PrivateIpAddress="10.0.0.1"
|
|
)
|
|
reservations = client.describe_instances(
|
|
Filters=[{"Name": "private-dns-name", "Values": ["ip-10-0-0-1.ec2.internal"]}]
|
|
)["Reservations"]
|
|
assert len(reservations[0]["Instances"]) == 1
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instances_filtering_by_ni_private_dns():
|
|
client = boto3.client("ec2", region_name="us-west-2")
|
|
conn = boto3.resource("ec2", "us-west-2")
|
|
conn.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, PrivateIpAddress="10.0.0.1"
|
|
)
|
|
reservations = client.describe_instances(
|
|
Filters=[
|
|
{
|
|
"Name": "network-interface.private-dns-name",
|
|
"Values": ["ip-10-0-0-1.us-west-2.compute.internal"],
|
|
}
|
|
]
|
|
)["Reservations"]
|
|
assert len(reservations[0]["Instances"]) == 1
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instances_with_unknown_security_group():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
sg_id = f"sg-{str(uuid4())[0:6]}"
|
|
with pytest.raises(ClientError) as exc:
|
|
client.run_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, SecurityGroupIds=[sg_id]
|
|
)
|
|
err = exc.value.response["Error"]
|
|
assert err["Code"] == "InvalidGroup.NotFound"
|
|
assert err["Message"] == f"The security group '{sg_id}' does not exist"
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instances_filtering_by_instance_group_name():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
sec_group_name = str(uuid4())[0:6]
|
|
client.create_security_group(Description="test", GroupName=sec_group_name)
|
|
client.run_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, SecurityGroups=[sec_group_name]
|
|
)
|
|
reservations = client.describe_instances(
|
|
Filters=[{"Name": "instance.group-name", "Values": [sec_group_name]}]
|
|
)["Reservations"]
|
|
assert len(reservations[0]["Instances"]) == 1
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instances_filtering_by_instance_group_id():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
sec_group_name = str(uuid4())[0:6]
|
|
create_sg = client.create_security_group(
|
|
Description="test", GroupName=sec_group_name
|
|
)
|
|
group_id = create_sg["GroupId"]
|
|
client.run_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, SecurityGroups=[sec_group_name]
|
|
)
|
|
reservations = client.describe_instances(
|
|
Filters=[{"Name": "instance.group-id", "Values": [group_id]}]
|
|
)["Reservations"]
|
|
assert len(reservations[0]["Instances"]) == 1
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instances_filtering_by_subnet_id():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
|
|
vpc_cidr = ipaddress.ip_network("192.168.42.0/24")
|
|
subnet_cidr = ipaddress.ip_network("192.168.42.0/25")
|
|
|
|
resp = client.create_vpc(CidrBlock=str(vpc_cidr))
|
|
vpc_id = resp["Vpc"]["VpcId"]
|
|
|
|
resp = client.create_subnet(CidrBlock=str(subnet_cidr), VpcId=vpc_id)
|
|
subnet_id = resp["Subnet"]["SubnetId"]
|
|
|
|
client.run_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MaxCount=1, MinCount=1, SubnetId=subnet_id
|
|
)
|
|
|
|
reservations = client.describe_instances(
|
|
Filters=[{"Name": "subnet-id", "Values": [subnet_id]}]
|
|
)["Reservations"]
|
|
assert len(reservations) == 1
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instances_filtering_by_tag():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
reservation = ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=3, MaxCount=3)
|
|
instance1, instance2, instance3 = reservation
|
|
|
|
tag1_name = str(uuid4())[0:6]
|
|
tag1_val = str(uuid4())
|
|
tag2_name = str(uuid4())[0:6]
|
|
tag2_val = str(uuid4())
|
|
tag3_name = str(uuid4())[0:6]
|
|
|
|
instance1.create_tags(
|
|
Tags=[
|
|
{"Key": tag1_name, "Value": tag1_val},
|
|
{"Key": tag2_name, "Value": tag2_val},
|
|
{"Key": tag3_name, "Value": json.dumps(["entry1", "entry2"])},
|
|
]
|
|
)
|
|
instance2.create_tags(Tags=[{"Key": tag1_name, "Value": tag1_val}])
|
|
instance2.create_tags(Tags=[{"Key": tag2_name, "Value": "wrong value"}])
|
|
instance3.create_tags(Tags=[{"Key": tag2_name, "Value": tag2_val}])
|
|
|
|
res = client.describe_instances(
|
|
Filters=[{"Name": "tag:tag0", "Values": ["value0"]}]
|
|
)
|
|
# describe_instances should return no instances
|
|
assert len(res["Reservations"]) == 0
|
|
|
|
res = client.describe_instances(
|
|
Filters=[{"Name": f"tag:{tag1_name}", "Values": [tag1_val]}]
|
|
)
|
|
# describe_instances should return both instances with this tag value
|
|
assert len(res["Reservations"]) == 1
|
|
assert len(res["Reservations"][0]["Instances"]) == 2
|
|
assert res["Reservations"][0]["Instances"][0]["InstanceId"] == instance1.id
|
|
assert res["Reservations"][0]["Instances"][1]["InstanceId"] == instance2.id
|
|
|
|
res = client.describe_instances(
|
|
Filters=[
|
|
{"Name": f"tag:{tag1_name}", "Values": [tag1_val]},
|
|
{"Name": f"tag:{tag2_name}", "Values": [tag2_val]},
|
|
]
|
|
)
|
|
# describe_instances should return the instance with both tag values
|
|
assert len(res["Reservations"]) == 1
|
|
assert len(res["Reservations"][0]["Instances"]) == 1
|
|
assert res["Reservations"][0]["Instances"][0]["InstanceId"] == instance1.id
|
|
|
|
res = client.describe_instances(
|
|
Filters=[{"Name": f"tag:{tag2_name}", "Values": [tag2_val, "bogus"]}]
|
|
)
|
|
# describe_instances should return both instances with one of the
|
|
# acceptable tag values
|
|
assert len(res["Reservations"]) == 1
|
|
assert len(res["Reservations"][0]["Instances"]) == 2
|
|
assert res["Reservations"][0]["Instances"][0]["InstanceId"] == instance1.id
|
|
assert res["Reservations"][0]["Instances"][1]["InstanceId"] == instance3.id
|
|
|
|
# We should be able to use tags containing special characters
|
|
res = client.describe_instances(
|
|
Filters=[
|
|
{"Name": f"tag:{tag3_name}", "Values": [json.dumps(["entry1", "entry2"])]}
|
|
]
|
|
)
|
|
assert len(res["Reservations"]) == 1
|
|
assert len(res["Reservations"][0]["Instances"]) == 1
|
|
assert res["Reservations"][0]["Instances"][0]["InstanceId"] == instance1.id
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instances_filtering_by_tag_value():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
reservation = ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=3, MaxCount=3)
|
|
instance1, instance2, instance3 = reservation
|
|
|
|
tag1_name = str(uuid4())[0:6]
|
|
tag1_val = str(uuid4())
|
|
tag2_name = str(uuid4())[0:6]
|
|
tag2_val = str(uuid4())
|
|
instance1.create_tags(Tags=[{"Key": tag1_name, "Value": tag1_val}])
|
|
instance1.create_tags(Tags=[{"Key": tag2_name, "Value": tag2_val}])
|
|
instance2.create_tags(Tags=[{"Key": tag1_name, "Value": tag1_val}])
|
|
instance2.create_tags(Tags=[{"Key": tag2_name, "Value": "wrong value"}])
|
|
instance3.create_tags(Tags=[{"Key": tag2_name, "Value": tag2_val}])
|
|
|
|
res = client.describe_instances(
|
|
Filters=[{"Name": "tag-value", "Values": ["value0"]}]
|
|
)
|
|
# describe_instances should return no instances
|
|
assert len(res["Reservations"]) == 0
|
|
|
|
res = client.describe_instances(
|
|
Filters=[{"Name": "tag-value", "Values": [tag1_val]}]
|
|
)
|
|
# describe_instances should return both instances with this tag value
|
|
assert len(res["Reservations"]) == 1
|
|
assert len(res["Reservations"][0]["Instances"]) == 2
|
|
assert res["Reservations"][0]["Instances"][0]["InstanceId"] == instance1.id
|
|
assert res["Reservations"][0]["Instances"][1]["InstanceId"] == instance2.id
|
|
|
|
res = client.describe_instances(
|
|
Filters=[{"Name": "tag-value", "Values": [tag2_val, tag1_val]}]
|
|
)
|
|
# describe_instances should return both instances with one of the
|
|
# acceptable tag values
|
|
assert len(res["Reservations"]) == 1
|
|
assert len(res["Reservations"][0]["Instances"]) == 3
|
|
assert res["Reservations"][0]["Instances"][0]["InstanceId"] == instance1.id
|
|
assert res["Reservations"][0]["Instances"][1]["InstanceId"] == instance2.id
|
|
assert res["Reservations"][0]["Instances"][2]["InstanceId"] == instance3.id
|
|
|
|
res = client.describe_instances(
|
|
Filters=[{"Name": "tag-value", "Values": [tag2_val, "bogus"]}]
|
|
)
|
|
# describe_instances should return both instances with one of the
|
|
# acceptable tag values
|
|
assert len(res["Reservations"]) == 1
|
|
assert len(res["Reservations"][0]["Instances"]) == 2
|
|
assert res["Reservations"][0]["Instances"][0]["InstanceId"] == instance1.id
|
|
assert res["Reservations"][0]["Instances"][1]["InstanceId"] == instance3.id
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instances_filtering_by_tag_name():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
reservation = ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=3, MaxCount=3)
|
|
instance1, instance2, instance3 = reservation
|
|
|
|
tag1 = str(uuid4())
|
|
tag3 = str(uuid4())
|
|
|
|
instance1.create_tags(Tags=[{"Key": tag1, "Value": ""}])
|
|
instance1.create_tags(Tags=[{"Key": "tag2", "Value": ""}])
|
|
instance2.create_tags(Tags=[{"Key": tag1, "Value": ""}])
|
|
instance2.create_tags(Tags=[{"Key": "tag2X", "Value": ""}])
|
|
instance3.create_tags(Tags=[{"Key": tag3, "Value": ""}])
|
|
|
|
res = client.describe_instances(Filters=[{"Name": "tag-key", "Values": ["tagX"]}])
|
|
# describe_instances should return no instances
|
|
assert len(res["Reservations"]) == 0
|
|
|
|
res = client.describe_instances(Filters=[{"Name": "tag-key", "Values": [tag1]}])
|
|
# describe_instances should return both instances with this tag value
|
|
assert len(res["Reservations"]) == 1
|
|
assert len(res["Reservations"][0]["Instances"]) == 2
|
|
assert res["Reservations"][0]["Instances"][0]["InstanceId"] == instance1.id
|
|
assert res["Reservations"][0]["Instances"][1]["InstanceId"] == instance2.id
|
|
|
|
res = client.describe_instances(
|
|
Filters=[{"Name": "tag-key", "Values": [tag1, tag3]}]
|
|
)
|
|
# describe_instances should return both instances with one of the
|
|
# acceptable tag values
|
|
assert len(res["Reservations"]) == 1
|
|
assert len(res["Reservations"][0]["Instances"]) == 3
|
|
assert res["Reservations"][0]["Instances"][0]["InstanceId"] == instance1.id
|
|
assert res["Reservations"][0]["Instances"][1]["InstanceId"] == instance2.id
|
|
assert res["Reservations"][0]["Instances"][2]["InstanceId"] == instance3.id
|
|
|
|
|
|
@mock_ec2
|
|
def test_instance_start_and_stop():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
reservation = ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=2, MaxCount=2)
|
|
instance1, instance2 = reservation
|
|
|
|
instance_ids = [instance1.id, instance2.id]
|
|
|
|
with pytest.raises(ClientError) as ex:
|
|
client.stop_instances(InstanceIds=instance_ids, DryRun=True)
|
|
assert ex.value.response["Error"]["Code"] == "DryRunOperation"
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 412
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "An error occurred (DryRunOperation) when calling the StopInstances operation: Request would have succeeded, but DryRun flag is set"
|
|
)
|
|
|
|
stopped_instances = client.stop_instances(InstanceIds=instance_ids)[
|
|
"StoppingInstances"
|
|
]
|
|
|
|
for instance in stopped_instances:
|
|
assert instance["PreviousState"] == {"Code": 16, "Name": "running"}
|
|
assert instance["CurrentState"] == {"Code": 64, "Name": "stopping"}
|
|
|
|
instance1.reload()
|
|
assert instance1.state == {"Code": 80, "Name": "stopped"}
|
|
|
|
with pytest.raises(ClientError) as ex:
|
|
client.start_instances(InstanceIds=[instance1.id], DryRun=True)
|
|
assert ex.value.response["Error"]["Code"] == "DryRunOperation"
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 412
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "An error occurred (DryRunOperation) when calling the StartInstances operation: Request would have succeeded, but DryRun flag is set"
|
|
)
|
|
|
|
instance1.reload()
|
|
# The DryRun-operation did not change anything
|
|
assert instance1.state == {"Code": 80, "Name": "stopped"}
|
|
|
|
started_instances = client.start_instances(InstanceIds=[instance1.id])[
|
|
"StartingInstances"
|
|
]
|
|
assert started_instances[0]["CurrentState"] == {"Code": 0, "Name": "pending"}
|
|
assert started_instances[0]["PreviousState"] == {"Code": 80, "Name": "stopped"}
|
|
|
|
|
|
@mock_ec2
|
|
def test_instance_reboot():
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
response = ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)
|
|
instance = response[0]
|
|
assert instance.state == {"Code": 0, "Name": "pending"}
|
|
instance.reload()
|
|
assert instance.state == {"Code": 16, "Name": "running"}
|
|
|
|
with pytest.raises(ClientError) as ex:
|
|
instance.reboot(DryRun=True)
|
|
assert ex.value.response["Error"]["Code"] == "DryRunOperation"
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 412
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "An error occurred (DryRunOperation) when calling the RebootInstances operation: Request would have succeeded, but DryRun flag is set"
|
|
)
|
|
|
|
assert instance.state == {"Code": 16, "Name": "running"}
|
|
|
|
instance.reboot()
|
|
assert instance.state == {"Code": 16, "Name": "running"}
|
|
|
|
|
|
@mock_ec2
|
|
def test_instance_attribute_instance_type():
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
response = ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)
|
|
instance = response[0]
|
|
assert instance.instance_type == "m1.small"
|
|
|
|
with pytest.raises(ClientError) as ex:
|
|
instance.modify_attribute(InstanceType={"Value": "m1.medium"}, DryRun=True)
|
|
assert ex.value.response["Error"]["Code"] == "DryRunOperation"
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 412
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "An error occurred (DryRunOperation) when calling the ModifyInstanceAttribute operation: Request would have succeeded, but DryRun flag is set"
|
|
)
|
|
|
|
instance.modify_attribute(InstanceType={"Value": "m1.medium"})
|
|
|
|
assert instance.instance_type == "m1.medium"
|
|
assert instance.describe_attribute(Attribute="instanceType")["InstanceType"] == {
|
|
"Value": "m1.medium"
|
|
}
|
|
|
|
|
|
@mock_ec2
|
|
def test_modify_instance_attribute_security_groups():
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
response = ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)
|
|
instance = response[0]
|
|
old_groups = instance.describe_attribute(Attribute="groupSet")["Groups"]
|
|
assert old_groups == []
|
|
|
|
sg_id = ec2.create_security_group(
|
|
GroupName=str(uuid4()), Description="this is a test security group"
|
|
).id
|
|
sg_id2 = ec2.create_security_group(
|
|
GroupName=str(uuid4()), Description="this is a test security group 2"
|
|
).id
|
|
|
|
with pytest.raises(ClientError) as ex:
|
|
instance.modify_attribute(Groups=[sg_id, sg_id2], DryRun=True)
|
|
assert ex.value.response["Error"]["Code"] == "DryRunOperation"
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 412
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "An error occurred (DryRunOperation) when calling the ModifyInstanceAttribute operation: Request would have succeeded, but DryRun flag is set"
|
|
)
|
|
|
|
instance.modify_attribute(Groups=[sg_id, sg_id2])
|
|
|
|
new_groups = instance.describe_attribute(Attribute="groupSet")["Groups"]
|
|
assert len(new_groups) == 2
|
|
assert {"GroupId": sg_id} in new_groups
|
|
assert {"GroupId": sg_id2} in new_groups
|
|
|
|
|
|
@mock_ec2
|
|
def test_instance_attribute_user_data():
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
res = ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)
|
|
instance = res[0]
|
|
|
|
with pytest.raises(ClientError) as ex:
|
|
instance.modify_attribute(
|
|
UserData={"Value": "this is my user data"}, DryRun=True
|
|
)
|
|
assert ex.value.response["Error"]["Code"] == "DryRunOperation"
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 412
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "An error occurred (DryRunOperation) when calling the ModifyInstanceAttribute operation: Request would have succeeded, but DryRun flag is set"
|
|
)
|
|
|
|
instance.modify_attribute(UserData={"Value": "this is my user data"})
|
|
|
|
attribute = instance.describe_attribute(Attribute="userData")["UserData"]
|
|
retrieved_user_data = attribute["Value"].encode("utf-8")
|
|
assert decode_method(retrieved_user_data) == b"this is my user data"
|
|
|
|
|
|
@mock_ec2
|
|
def test_instance_attribute_source_dest_check():
|
|
ec2 = boto3.resource("ec2", region_name="us-west-1")
|
|
instance = ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)[0]
|
|
|
|
instance_attribute = instance.describe_attribute(Attribute="sourceDestCheck")
|
|
assert instance_attribute.get("SourceDestCheck") == {"Value": True}
|
|
|
|
# Set to false (note: Boto converts bool to string, eg 'false')
|
|
|
|
with pytest.raises(ClientError) as ex:
|
|
instance.modify_attribute(SourceDestCheck={"Value": False}, DryRun=True)
|
|
assert ex.value.response["Error"]["Code"] == "DryRunOperation"
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 412
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "An error occurred (DryRunOperation) when calling the ModifyInstanceAttribute operation: Request would have succeeded, but DryRun flag is set"
|
|
)
|
|
|
|
instance.modify_attribute(SourceDestCheck={"Value": False})
|
|
|
|
instance_attribute = instance.describe_attribute(Attribute="sourceDestCheck")
|
|
assert instance_attribute.get("SourceDestCheck") == {"Value": False}
|
|
|
|
# Set back to true
|
|
instance.modify_attribute(SourceDestCheck={"Value": True})
|
|
|
|
instance_attribute = instance.describe_attribute(Attribute="sourceDestCheck")
|
|
assert instance_attribute.get("SourceDestCheck") == {"Value": True}
|
|
|
|
|
|
@mock_ec2
|
|
def test_user_data_with_run_instance():
|
|
user_data = b"some user data"
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
instance = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, UserData=user_data
|
|
)[0]
|
|
|
|
attribute = instance.describe_attribute(Attribute="userData")["UserData"]
|
|
retrieved_user_data = attribute["Value"].encode("utf-8")
|
|
decoded_user_data = decode_method(retrieved_user_data)
|
|
assert decoded_user_data == b"some user data"
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_with_security_group_name():
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
|
|
sec_group_name = str(uuid4())[0:6]
|
|
|
|
with pytest.raises(ClientError) as ex:
|
|
ec2.create_security_group(
|
|
GroupName=sec_group_name, Description="d", DryRun=True
|
|
)
|
|
assert ex.value.response["Error"]["Code"] == "DryRunOperation"
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 412
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "An error occurred (DryRunOperation) when calling the CreateSecurityGroup operation: Request would have succeeded, but DryRun flag is set"
|
|
)
|
|
|
|
group = ec2.create_security_group(
|
|
GroupName=sec_group_name, Description="some description"
|
|
)
|
|
|
|
res = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, SecurityGroups=[sec_group_name]
|
|
)
|
|
instance = res[0]
|
|
|
|
assert instance.security_groups == [
|
|
{"GroupName": sec_group_name, "GroupId": group.id}
|
|
]
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_with_security_group_id():
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
sec_group_name = str(uuid4())
|
|
group = ec2.create_security_group(
|
|
GroupName=sec_group_name, Description="some description"
|
|
)
|
|
instance = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, SecurityGroupIds=[group.id]
|
|
)[0]
|
|
|
|
assert instance.security_groups == [
|
|
{"GroupName": sec_group_name, "GroupId": group.id}
|
|
]
|
|
|
|
|
|
@mock_ec2
|
|
@pytest.mark.parametrize("hibernate", [True, False])
|
|
def test_run_instance_with_additional_args(hibernate):
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
instance = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
InstanceType="t1.micro",
|
|
Placement={"AvailabilityZone": "us-east-1b"},
|
|
HibernationOptions={"Configured": hibernate},
|
|
)[0]
|
|
|
|
assert instance.instance_type == "t1.micro"
|
|
assert instance.placement["AvailabilityZone"] == "us-east-1b"
|
|
assert instance.hibernation_options == {"Configured": hibernate}
|
|
|
|
reservations = client.describe_instances(InstanceIds=[instance.id])["Reservations"]
|
|
instance = reservations[0]["Instances"][0]
|
|
assert instance["HibernationOptions"] == {"Configured": hibernate}
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_with_default_placement():
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
instance = ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)[0]
|
|
|
|
assert instance.placement["AvailabilityZone"] == "us-east-1a"
|
|
|
|
|
|
@mock_ec2
|
|
@mock.patch(
|
|
"moto.ec2.models.instances.settings.EC2_ENABLE_INSTANCE_TYPE_VALIDATION",
|
|
new_callable=mock.PropertyMock(return_value=True),
|
|
)
|
|
def test_run_instance_with_invalid_instance_type(m_flag):
|
|
if settings.TEST_SERVER_MODE:
|
|
raise SkipTest(
|
|
"It is not possible to set the environment variable in server mode"
|
|
)
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
with pytest.raises(ClientError) as ex:
|
|
ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
InstanceType="invalid_type",
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
Placement={"AvailabilityZone": "us-east-1b"},
|
|
)
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "The instance type 'invalid_type' does not exist"
|
|
)
|
|
assert m_flag is True
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_with_availability_zone_not_from_region():
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
with pytest.raises(ClientError) as ex:
|
|
ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
InstanceType="t2.nano",
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
Placement={"AvailabilityZone": "us-west-1b"},
|
|
)
|
|
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "Invalid Availability Zone (us-west-1b)"
|
|
)
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_with_subnet():
|
|
client = boto3.client("ec2", region_name="eu-central-1")
|
|
|
|
ip_networks = [
|
|
(ipaddress.ip_network("10.0.0.0/16"), ipaddress.ip_network("10.0.99.0/24")),
|
|
(
|
|
ipaddress.ip_network("192.168.42.0/24"),
|
|
ipaddress.ip_network("192.168.42.0/25"),
|
|
),
|
|
]
|
|
|
|
# Tests instances are created with the correct IPs
|
|
for vpc_cidr, subnet_cidr in ip_networks:
|
|
resp = client.create_vpc(
|
|
CidrBlock=str(vpc_cidr),
|
|
AmazonProvidedIpv6CidrBlock=False,
|
|
DryRun=False,
|
|
InstanceTenancy="default",
|
|
)
|
|
vpc_id = resp["Vpc"]["VpcId"]
|
|
|
|
resp = client.create_subnet(CidrBlock=str(subnet_cidr), VpcId=vpc_id)
|
|
subnet_id = resp["Subnet"]["SubnetId"]
|
|
|
|
resp = client.run_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MaxCount=1, MinCount=1, SubnetId=subnet_id
|
|
)
|
|
instance = resp["Instances"][0]
|
|
assert instance["SubnetId"] == subnet_id
|
|
|
|
priv_ipv4 = ipaddress.ip_address(str(instance["PrivateIpAddress"]))
|
|
assert priv_ipv4 in subnet_cidr
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_with_specified_private_ipv4():
|
|
client = boto3.client("ec2", region_name="eu-central-1")
|
|
|
|
vpc_cidr = ipaddress.ip_network("192.168.42.0/24")
|
|
subnet_cidr = ipaddress.ip_network("192.168.42.0/25")
|
|
|
|
resp = client.create_vpc(
|
|
CidrBlock=str(vpc_cidr),
|
|
AmazonProvidedIpv6CidrBlock=False,
|
|
DryRun=False,
|
|
InstanceTenancy="default",
|
|
)
|
|
vpc_id = resp["Vpc"]["VpcId"]
|
|
|
|
resp = client.create_subnet(CidrBlock=str(subnet_cidr), VpcId=vpc_id)
|
|
subnet_id = resp["Subnet"]["SubnetId"]
|
|
|
|
resp = client.run_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MaxCount=1,
|
|
MinCount=1,
|
|
SubnetId=subnet_id,
|
|
PrivateIpAddress="192.168.42.5",
|
|
)
|
|
instance = resp["Instances"][0]
|
|
assert instance["SubnetId"] == subnet_id
|
|
assert instance["PrivateIpAddress"] == "192.168.42.5"
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_with_placement():
|
|
client = boto3.client("ec2", region_name="eu-central-1")
|
|
host_id = "h-asdfasdfasdf"
|
|
|
|
instance = client.run_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MaxCount=1, MinCount=1, Placement={"HostId": host_id}
|
|
)["Instances"][0]
|
|
instance_id = instance["InstanceId"]
|
|
assert instance["Placement"]["HostId"] == host_id
|
|
|
|
resp = client.describe_instances(InstanceIds=[instance_id])["Reservations"][0][
|
|
"Instances"
|
|
][0]
|
|
assert resp["Placement"]["HostId"] == host_id
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_mapped_public_ipv4():
|
|
client = boto3.client("ec2", region_name="eu-central-1")
|
|
|
|
vpc_cidr = ipaddress.ip_network("192.168.42.0/24")
|
|
subnet_cidr = ipaddress.ip_network("192.168.42.0/25")
|
|
|
|
resp = client.create_vpc(
|
|
CidrBlock=str(vpc_cidr),
|
|
AmazonProvidedIpv6CidrBlock=False,
|
|
DryRun=False,
|
|
InstanceTenancy="default",
|
|
)
|
|
vpc_id = resp["Vpc"]["VpcId"]
|
|
|
|
resp = client.create_subnet(CidrBlock=str(subnet_cidr), VpcId=vpc_id)
|
|
subnet_id = resp["Subnet"]["SubnetId"]
|
|
client.modify_subnet_attribute(
|
|
SubnetId=subnet_id, MapPublicIpOnLaunch={"Value": True}
|
|
)
|
|
|
|
resp = client.run_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MaxCount=1, MinCount=1, SubnetId=subnet_id
|
|
)
|
|
instance = resp["Instances"][0]
|
|
assert "PublicDnsName" in instance
|
|
assert "PublicIpAddress" in instance
|
|
assert len(instance["PublicDnsName"]) > 0
|
|
assert len(instance["PublicIpAddress"]) > 0
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_with_nic_autocreated():
|
|
ec2 = boto3.resource("ec2", "us-west-1")
|
|
client = boto3.client("ec2", "us-west-1")
|
|
vpc = ec2.create_vpc(CidrBlock="10.0.0.0/16")
|
|
subnet = ec2.create_subnet(VpcId=vpc.id, CidrBlock="10.0.0.0/18")
|
|
security_group1 = ec2.create_security_group(
|
|
GroupName=str(uuid4()), Description="n/a"
|
|
)
|
|
security_group2 = ec2.create_security_group(
|
|
GroupName=str(uuid4()), Description="n/a"
|
|
)
|
|
private_ip = "10.0.0.1"
|
|
|
|
instance = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
SubnetId=subnet.id,
|
|
SecurityGroups=[security_group1.group_name],
|
|
SecurityGroupIds=[security_group2.group_id],
|
|
PrivateIpAddress=private_ip,
|
|
)[0]
|
|
|
|
instance_eni = instance.network_interfaces_attribute
|
|
assert len(instance_eni) == 1
|
|
|
|
nii = instance_eni[0]["NetworkInterfaceId"]
|
|
|
|
my_enis = client.describe_network_interfaces(NetworkInterfaceIds=[nii])[
|
|
"NetworkInterfaces"
|
|
]
|
|
assert len(my_enis) == 1
|
|
eni = my_enis[0]
|
|
|
|
assert instance.subnet_id == subnet.id
|
|
assert len(instance.security_groups) == 2
|
|
assert set([group["GroupId"] for group in instance.security_groups]) == {
|
|
security_group1.id,
|
|
security_group2.id,
|
|
}
|
|
|
|
assert eni["SubnetId"] == subnet.id
|
|
assert len(eni["Groups"]) == 2
|
|
assert set([group["GroupId"] for group in eni["Groups"]]) == {
|
|
security_group1.id,
|
|
security_group2.id,
|
|
}
|
|
assert len(eni["PrivateIpAddresses"]) == 1
|
|
assert eni["PrivateIpAddresses"][0]["PrivateIpAddress"] == private_ip
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_with_nic_preexisting():
|
|
ec2 = boto3.resource("ec2", "us-west-1")
|
|
client = boto3.client("ec2", "us-west-1")
|
|
vpc = ec2.create_vpc(CidrBlock="10.0.0.0/16")
|
|
subnet = ec2.create_subnet(VpcId=vpc.id, CidrBlock="10.0.0.0/18")
|
|
security_group1 = ec2.create_security_group(
|
|
GroupName=str(uuid4()), Description="n/a"
|
|
)
|
|
security_group2 = ec2.create_security_group(
|
|
GroupName=str(uuid4()), Description="n/a"
|
|
)
|
|
private_ip = "54.0.0.1"
|
|
eni = ec2.create_network_interface(
|
|
SubnetId=subnet.id,
|
|
PrivateIpAddress=private_ip,
|
|
Groups=[security_group1.group_id],
|
|
)
|
|
|
|
instance = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
NetworkInterfaces=[{"DeviceIndex": 0, "NetworkInterfaceId": eni.id}],
|
|
SecurityGroupIds=[security_group2.group_id],
|
|
)[0]
|
|
|
|
assert instance.subnet_id == subnet.id
|
|
|
|
nii = instance.network_interfaces_attribute[0]["NetworkInterfaceId"]
|
|
|
|
all_enis = client.describe_network_interfaces(NetworkInterfaceIds=[nii])[
|
|
"NetworkInterfaces"
|
|
]
|
|
assert len(all_enis) == 1
|
|
|
|
instance_enis = instance.network_interfaces_attribute
|
|
assert len(instance_enis) == 1
|
|
instance_eni = instance_enis[0]
|
|
assert instance_eni["NetworkInterfaceId"] == eni.id
|
|
|
|
assert instance_eni["SubnetId"] == subnet.id
|
|
assert len(instance_eni["Groups"]) == 2
|
|
assert set([group["GroupId"] for group in instance_eni["Groups"]]) == {
|
|
security_group1.id,
|
|
security_group2.id,
|
|
}
|
|
assert len(instance_eni["PrivateIpAddresses"]) == 1
|
|
assert instance_eni["PrivateIpAddresses"][0]["PrivateIpAddress"] == private_ip
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_with_new_nic_and_security_groups():
|
|
ec2 = boto3.resource("ec2", "us-west-1")
|
|
client = boto3.client("ec2", "us-west-1")
|
|
security_group1 = ec2.create_security_group(
|
|
GroupName=str(uuid4()), Description="n/a"
|
|
)
|
|
security_group2 = ec2.create_security_group(
|
|
GroupName=str(uuid4()), Description="n/a"
|
|
)
|
|
|
|
instance = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
NetworkInterfaces=[
|
|
{
|
|
"DeviceIndex": 0,
|
|
"Groups": [security_group1.group_id, security_group2.group_id],
|
|
}
|
|
],
|
|
)[0]
|
|
|
|
nii = instance.network_interfaces_attribute[0]["NetworkInterfaceId"]
|
|
|
|
all_enis = client.describe_network_interfaces(NetworkInterfaceIds=[nii])[
|
|
"NetworkInterfaces"
|
|
]
|
|
assert len(all_enis) == 1
|
|
|
|
instance_enis = instance.network_interfaces_attribute
|
|
assert len(instance_enis) == 1
|
|
instance_eni = instance_enis[0]
|
|
|
|
assert len(instance_eni["Groups"]) == 2
|
|
assert set([group["GroupId"] for group in instance_eni["Groups"]]) == {
|
|
security_group1.id,
|
|
security_group2.id,
|
|
}
|
|
|
|
|
|
@mock_ec2
|
|
def test_instance_with_nic_attach_detach():
|
|
ec2 = boto3.resource("ec2", "us-west-1")
|
|
client = boto3.client("ec2", "us-west-1")
|
|
vpc = ec2.create_vpc(CidrBlock="10.0.0.0/16")
|
|
subnet = ec2.create_subnet(VpcId=vpc.id, CidrBlock="10.0.0.0/18")
|
|
security_group1 = ec2.create_security_group(
|
|
GroupName=str(uuid4()), Description="n/a"
|
|
)
|
|
security_group2 = ec2.create_security_group(
|
|
GroupName=str(uuid4()), Description="n/a"
|
|
)
|
|
|
|
instance = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
SecurityGroupIds=[security_group1.group_id],
|
|
)[0]
|
|
|
|
eni = ec2.create_network_interface(SubnetId=subnet.id, Groups=[security_group2.id])
|
|
eni_id = eni.id
|
|
|
|
# Check initial instance and ENI data
|
|
assert len(instance.network_interfaces_attribute) == 1
|
|
|
|
assert [group["GroupId"] for group in eni.groups] == [security_group2.id]
|
|
|
|
# Attach
|
|
with pytest.raises(ClientError) as ex:
|
|
client.attach_network_interface(
|
|
NetworkInterfaceId=eni_id,
|
|
InstanceId=instance.id,
|
|
DeviceIndex=1,
|
|
DryRun=True,
|
|
)
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 412
|
|
assert ex.value.response["Error"]["Code"] == "DryRunOperation"
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "An error occurred (DryRunOperation) when calling the AttachNetworkInterface operation: Request would have succeeded, but DryRun flag is set"
|
|
)
|
|
|
|
client.attach_network_interface(
|
|
NetworkInterfaceId=eni_id, InstanceId=instance.id, DeviceIndex=1
|
|
)
|
|
|
|
# Check attached instance and ENI data
|
|
instance.reload()
|
|
assert len(instance.network_interfaces_attribute) == 2
|
|
instance_eni = instance.network_interfaces_attribute[1]
|
|
assert instance_eni["NetworkInterfaceId"] == eni_id
|
|
assert len(instance_eni["Groups"]) == 2
|
|
assert set([group["GroupId"] for group in instance_eni["Groups"]]) == {
|
|
security_group1.id,
|
|
security_group2.id,
|
|
}
|
|
|
|
eni = client.describe_network_interfaces(
|
|
Filters=[{"Name": "network-interface-id", "Values": [eni_id]}]
|
|
)["NetworkInterfaces"][0]
|
|
assert len(eni["Groups"]) == 2
|
|
assert set([group["GroupId"] for group in eni["Groups"]]) == {
|
|
security_group1.id,
|
|
security_group2.id,
|
|
}
|
|
|
|
# Detach
|
|
with pytest.raises(ClientError) as ex:
|
|
client.detach_network_interface(
|
|
AttachmentId=instance_eni["Attachment"]["AttachmentId"], DryRun=True
|
|
)
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 412
|
|
assert ex.value.response["Error"]["Code"] == "DryRunOperation"
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "An error occurred (DryRunOperation) when calling the DetachNetworkInterface operation: Request would have succeeded, but DryRun flag is set"
|
|
)
|
|
|
|
client.detach_network_interface(
|
|
AttachmentId=instance_eni["Attachment"]["AttachmentId"]
|
|
)
|
|
|
|
# Check detached instance and ENI data
|
|
instance.reload()
|
|
assert len(instance.network_interfaces_attribute) == 1
|
|
|
|
eni = client.describe_network_interfaces(
|
|
Filters=[{"Name": "network-interface-id", "Values": [eni_id]}]
|
|
)["NetworkInterfaces"][0]
|
|
assert [group["GroupId"] for group in eni["Groups"]] == [security_group2.id]
|
|
|
|
# Detach with invalid attachment ID
|
|
with pytest.raises(ClientError) as ex:
|
|
client.detach_network_interface(AttachmentId="eni-attach-1234abcd")
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
assert "RequestId" in ex.value.response["ResponseMetadata"]
|
|
assert ex.value.response["Error"]["Code"] == "InvalidAttachmentID.NotFound"
|
|
|
|
|
|
@mock_ec2
|
|
def test_ec2_classic_has_public_ip_address():
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
instance = ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)[0]
|
|
assert instance.public_ip_address is not None
|
|
assert instance.public_ip_address.replace(".", "-") in instance.public_dns_name
|
|
assert instance.private_ip_address is not None
|
|
assert instance.private_ip_address.replace(".", "-") in instance.private_dns_name
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_with_keypair():
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
|
|
instance = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, KeyName="keypair_name"
|
|
)[0]
|
|
|
|
assert instance.key_name == "keypair_name"
|
|
|
|
|
|
@mock_ec2
|
|
def test_describe_instances_with_keypair_filter():
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
for i in range(3):
|
|
key_name = "kp-single" if i % 2 else "kp-multiple"
|
|
ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, KeyName=key_name
|
|
)
|
|
test_data = [
|
|
(["kp-single"], 1),
|
|
(["kp-multiple"], 2),
|
|
(["kp-single", "kp-multiple"], 3),
|
|
]
|
|
for filter_values, expected_instance_count in test_data:
|
|
_filter = [{"Name": "key-name", "Values": filter_values}]
|
|
instances_found = list(ec2.instances.filter(Filters=_filter))
|
|
assert len(instances_found) == expected_instance_count
|
|
|
|
|
|
@mock_ec2
|
|
@mock.patch(
|
|
"moto.ec2.models.instances.settings.ENABLE_KEYPAIR_VALIDATION",
|
|
new_callable=mock.PropertyMock(return_value=True),
|
|
)
|
|
def test_run_instance_with_invalid_keypair(m_flag):
|
|
if settings.TEST_SERVER_MODE:
|
|
raise SkipTest(
|
|
"It is not possible to set the environment variable in server mode"
|
|
)
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
keypair_name = "keypair_name"
|
|
ec2.create_key_pair(KeyName=keypair_name)
|
|
|
|
with pytest.raises(ClientError) as ex:
|
|
ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, KeyName="not a key name"
|
|
)[0]
|
|
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
assert ex.value.response["Error"]["Code"] == "InvalidKeyPair.NotFound"
|
|
assert m_flag is True
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_with_block_device_mappings():
|
|
ec2_client = boto3.client("ec2", region_name="us-east-1")
|
|
|
|
kwargs = {
|
|
"MinCount": 1,
|
|
"MaxCount": 1,
|
|
"ImageId": EXAMPLE_AMI_ID,
|
|
"KeyName": "the_key",
|
|
"InstanceType": "t1.micro",
|
|
"BlockDeviceMappings": [{"DeviceName": "/dev/sda2", "Ebs": {"VolumeSize": 50}}],
|
|
}
|
|
|
|
instance_id = ec2_client.run_instances(**kwargs)["Instances"][0]["InstanceId"]
|
|
|
|
instances = ec2_client.describe_instances(InstanceIds=[instance_id])
|
|
volume = instances["Reservations"][0]["Instances"][0]["BlockDeviceMappings"][0][
|
|
"Ebs"
|
|
]
|
|
|
|
volumes = ec2_client.describe_volumes(VolumeIds=[volume["VolumeId"]])
|
|
assert volumes["Volumes"][0]["Size"] == 50
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_with_block_device_mappings_missing_ebs():
|
|
ec2_client = boto3.client("ec2", region_name="us-east-1")
|
|
|
|
kwargs = {
|
|
"MinCount": 1,
|
|
"MaxCount": 1,
|
|
"ImageId": EXAMPLE_AMI_ID,
|
|
"KeyName": "the_key",
|
|
"InstanceType": "t1.micro",
|
|
"BlockDeviceMappings": [{"DeviceName": "/dev/sda2"}],
|
|
}
|
|
with pytest.raises(ClientError) as ex:
|
|
ec2_client.run_instances(**kwargs)
|
|
|
|
assert ex.value.response["Error"]["Code"] == "MissingParameter"
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "The request must contain the parameter ebs"
|
|
)
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_with_block_device_mappings_using_no_device():
|
|
ec2_client = boto3.client("ec2", region_name="us-east-1")
|
|
|
|
kwargs = {
|
|
"MinCount": 1,
|
|
"MaxCount": 1,
|
|
"ImageId": EXAMPLE_AMI_ID,
|
|
"KeyName": "the_key",
|
|
"InstanceType": "t1.micro",
|
|
"BlockDeviceMappings": [{"DeviceName": "/dev/sda2", "NoDevice": ""}],
|
|
}
|
|
resp = ec2_client.run_instances(**kwargs)
|
|
instance_id = resp["Instances"][0]["InstanceId"]
|
|
|
|
instances = ec2_client.describe_instances(InstanceIds=[instance_id])
|
|
# Assuming that /dev/sda2 is not the root device and that there is a /dev/sda1, boto would
|
|
# create an instance with one block device instead of two. However, moto's modeling of
|
|
# BlockDeviceMappings is simplified, so we will accept that moto creates an instance without
|
|
# block devices for now
|
|
# assert "BlockDeviceMappings" not in instances["Reservations"][0]["Instances"][0]
|
|
|
|
# moto gives the key with an empty list instead of not having it at all, that's also fine
|
|
assert instances["Reservations"][0]["Instances"][0]["BlockDeviceMappings"] == []
|
|
|
|
# passing None with NoDevice should raise ParamValidationError
|
|
kwargs["BlockDeviceMappings"][0]["NoDevice"] = None
|
|
with pytest.raises(ParamValidationError) as ex:
|
|
ec2_client.run_instances(**kwargs)
|
|
|
|
# passing a string other than "" with NoDevice should raise InvalidRequest
|
|
kwargs["BlockDeviceMappings"][0]["NoDevice"] = "yes"
|
|
with pytest.raises(ClientError) as ex:
|
|
ec2_client.run_instances(**kwargs)
|
|
|
|
assert ex.value.response["Error"]["Code"] == "InvalidRequest"
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
assert ex.value.response["Error"]["Message"] == "The request received was invalid"
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_with_block_device_mappings_missing_size():
|
|
ec2_client = boto3.client("ec2", region_name="us-east-1")
|
|
|
|
kwargs = {
|
|
"MinCount": 1,
|
|
"MaxCount": 1,
|
|
"ImageId": EXAMPLE_AMI_ID,
|
|
"KeyName": "the_key",
|
|
"InstanceType": "t1.micro",
|
|
"BlockDeviceMappings": [
|
|
{"DeviceName": "/dev/sda2", "Ebs": {"VolumeType": "standard"}}
|
|
],
|
|
}
|
|
with pytest.raises(ClientError) as ex:
|
|
ec2_client.run_instances(**kwargs)
|
|
|
|
assert ex.value.response["Error"]["Code"] == "MissingParameter"
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "The request must contain the parameter size or snapshotId"
|
|
)
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_with_block_device_mappings_from_snapshot():
|
|
ec2_client = boto3.client("ec2", region_name="us-east-1")
|
|
ec2_resource = boto3.resource("ec2", region_name="us-east-1")
|
|
|
|
volume_details = {
|
|
"AvailabilityZone": "1a",
|
|
"Size": 30,
|
|
}
|
|
|
|
volume = ec2_resource.create_volume(**volume_details)
|
|
snapshot = volume.create_snapshot()
|
|
kwargs = {
|
|
"MinCount": 1,
|
|
"MaxCount": 1,
|
|
"ImageId": EXAMPLE_AMI_ID,
|
|
"KeyName": "the_key",
|
|
"InstanceType": "t1.micro",
|
|
"BlockDeviceMappings": [
|
|
{"DeviceName": "/dev/sda2", "Ebs": {"SnapshotId": snapshot.snapshot_id}}
|
|
],
|
|
}
|
|
|
|
resp = ec2_client.run_instances(**kwargs)
|
|
instance_id = resp["Instances"][0]["InstanceId"]
|
|
|
|
instances = ec2_client.describe_instances(InstanceIds=[instance_id])
|
|
volume = instances["Reservations"][0]["Instances"][0]["BlockDeviceMappings"][0][
|
|
"Ebs"
|
|
]
|
|
|
|
volumes = ec2_client.describe_volumes(VolumeIds=[volume["VolumeId"]])
|
|
|
|
assert volumes["Volumes"][0]["Size"] == 30
|
|
assert volumes["Volumes"][0]["SnapshotId"] == snapshot.snapshot_id
|
|
|
|
|
|
@mock_ec2
|
|
def test_describe_instance_status_no_instances():
|
|
if settings.TEST_SERVER_MODE:
|
|
raise SkipTest("ServerMode is not guaranteed to be empty")
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
all_status = client.describe_instance_status()["InstanceStatuses"]
|
|
assert len(all_status) == 0
|
|
|
|
|
|
@mock_ec2
|
|
def test_describe_instance_status_with_instances():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
instance = ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)[0]
|
|
|
|
all_status = client.describe_instance_status()["InstanceStatuses"]
|
|
instance_ids = [s["InstanceId"] for s in all_status]
|
|
assert instance.id in instance_ids
|
|
|
|
my_status = [s for s in all_status if s["InstanceId"] == instance.id][0]
|
|
assert my_status["InstanceStatus"]["Status"] == "ok"
|
|
assert my_status["SystemStatus"]["Status"] == "ok"
|
|
|
|
|
|
@mock_ec2
|
|
def test_describe_instance_status_with_instance_filter_deprecated():
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
|
|
# We want to filter based on this one
|
|
instance = ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)[0]
|
|
|
|
# This is just to setup the test
|
|
ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)
|
|
|
|
all_status = client.describe_instance_status(InstanceIds=[instance.id])[
|
|
"InstanceStatuses"
|
|
]
|
|
assert len(all_status) == 1
|
|
assert all_status[0]["InstanceId"] == instance.id
|
|
|
|
# Call get_all_instance_status with a bad id should raise an error
|
|
with pytest.raises(ClientError) as ex:
|
|
client.describe_instance_status(InstanceIds=[instance.id, "i-1234abcd"])
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
assert "RequestId" in ex.value.response["ResponseMetadata"]
|
|
assert ex.value.response["Error"]["Code"] == "InvalidInstanceID.NotFound"
|
|
|
|
|
|
@mock_ec2
|
|
def test_describe_instance_credit_specifications():
|
|
conn = boto3.client("ec2", region_name="us-west-1")
|
|
|
|
# We want to filter based on this one
|
|
reservation = conn.run_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)
|
|
result = conn.describe_instance_credit_specifications(
|
|
InstanceIds=[reservation["Instances"][0]["InstanceId"]]
|
|
)
|
|
assert (
|
|
result["InstanceCreditSpecifications"][0]["InstanceId"]
|
|
== reservation["Instances"][0]["InstanceId"]
|
|
)
|
|
|
|
|
|
@mock_ec2
|
|
def test_describe_instance_status_with_instance_filter():
|
|
conn = boto3.client("ec2", region_name="us-west-1")
|
|
|
|
# We want to filter based on this one
|
|
reservation = conn.run_instances(ImageId=EXAMPLE_AMI_ID, MinCount=3, MaxCount=3)
|
|
instance1 = reservation["Instances"][0]
|
|
instance2 = reservation["Instances"][1]
|
|
instance3 = reservation["Instances"][2]
|
|
conn.stop_instances(InstanceIds=[instance1["InstanceId"]])
|
|
stopped_instance_ids = [instance1["InstanceId"]]
|
|
running_instance_ids = sorted([instance2["InstanceId"], instance3["InstanceId"]])
|
|
all_instance_ids = sorted(stopped_instance_ids + running_instance_ids)
|
|
|
|
# Filter instance using the state name
|
|
state_name_filter = {
|
|
"running_and_stopped": [
|
|
{"Name": "instance-state-name", "Values": ["running", "stopped"]}
|
|
],
|
|
"running": [{"Name": "instance-state-name", "Values": ["running"]}],
|
|
"stopped": [{"Name": "instance-state-name", "Values": ["stopped"]}],
|
|
}
|
|
|
|
found_statuses = conn.describe_instance_status(
|
|
IncludeAllInstances=True, Filters=state_name_filter["running_and_stopped"]
|
|
)["InstanceStatuses"]
|
|
found_instance_ids = [status["InstanceId"] for status in found_statuses]
|
|
for _id in all_instance_ids:
|
|
assert _id in found_instance_ids
|
|
|
|
found_statuses = conn.describe_instance_status(
|
|
IncludeAllInstances=True, Filters=state_name_filter["running"]
|
|
)["InstanceStatuses"]
|
|
found_instance_ids = [status["InstanceId"] for status in found_statuses]
|
|
for _id in stopped_instance_ids:
|
|
assert _id not in found_instance_ids
|
|
for _id in running_instance_ids:
|
|
assert _id in found_instance_ids
|
|
|
|
found_statuses = conn.describe_instance_status(
|
|
IncludeAllInstances=True, Filters=state_name_filter["stopped"]
|
|
)["InstanceStatuses"]
|
|
found_instance_ids = [status["InstanceId"] for status in found_statuses]
|
|
for _id in stopped_instance_ids:
|
|
assert _id in found_instance_ids
|
|
for _id in running_instance_ids:
|
|
assert _id not in found_instance_ids
|
|
|
|
# Filter instance using the state code
|
|
state_code_filter = {
|
|
"running_and_stopped": [
|
|
{"Name": "instance-state-code", "Values": ["16", "80"]}
|
|
],
|
|
"running": [{"Name": "instance-state-code", "Values": ["16"]}],
|
|
"stopped": [{"Name": "instance-state-code", "Values": ["80"]}],
|
|
}
|
|
|
|
found_statuses = conn.describe_instance_status(
|
|
IncludeAllInstances=True, Filters=state_code_filter["running_and_stopped"]
|
|
)["InstanceStatuses"]
|
|
found_instance_ids = [status["InstanceId"] for status in found_statuses]
|
|
for _id in all_instance_ids:
|
|
assert _id in found_instance_ids
|
|
|
|
found_statuses = conn.describe_instance_status(
|
|
IncludeAllInstances=True, Filters=state_code_filter["running"]
|
|
)["InstanceStatuses"]
|
|
found_instance_ids = [status["InstanceId"] for status in found_statuses]
|
|
for _id in stopped_instance_ids:
|
|
assert _id not in found_instance_ids
|
|
for _id in running_instance_ids:
|
|
assert _id in found_instance_ids
|
|
|
|
found_statuses = conn.describe_instance_status(
|
|
IncludeAllInstances=True, Filters=state_code_filter["stopped"]
|
|
)["InstanceStatuses"]
|
|
found_instance_ids = [status["InstanceId"] for status in found_statuses]
|
|
for _id in stopped_instance_ids:
|
|
assert _id in found_instance_ids
|
|
for _id in running_instance_ids:
|
|
assert _id not in found_instance_ids
|
|
|
|
|
|
@mock_ec2
|
|
def test_describe_instance_status_with_non_running_instances():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
reservation = ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=3, MaxCount=3)
|
|
instance1, instance2, instance3 = reservation
|
|
instance1.stop()
|
|
instance2.terminate()
|
|
|
|
all_running_status = client.describe_instance_status()["InstanceStatuses"]
|
|
assert instance1.id not in [status["InstanceId"] for status in all_running_status]
|
|
assert instance2.id not in [status["InstanceId"] for status in all_running_status]
|
|
assert instance3.id in [status["InstanceId"] for status in all_running_status]
|
|
|
|
my_status = [s for s in all_running_status if s["InstanceId"] == instance3.id][0]
|
|
assert my_status["InstanceState"] == {"Code": 16, "Name": "running"}
|
|
|
|
all_status = client.describe_instance_status(IncludeAllInstances=True)[
|
|
"InstanceStatuses"
|
|
]
|
|
assert instance1.id in [status["InstanceId"] for status in all_status]
|
|
assert instance2.id in [status["InstanceId"] for status in all_status]
|
|
assert instance3.id in [status["InstanceId"] for status in all_status]
|
|
|
|
status1 = next((s for s in all_status if s["InstanceId"] == instance1.id), None)
|
|
assert status1["InstanceState"] == {"Code": 80, "Name": "stopped"}
|
|
|
|
status2 = next((s for s in all_status if s["InstanceId"] == instance2.id), None)
|
|
assert status2["InstanceState"] == {"Code": 48, "Name": "terminated"}
|
|
|
|
status3 = next((s for s in all_status if s["InstanceId"] == instance3.id), None)
|
|
assert status3["InstanceState"] == {"Code": 16, "Name": "running"}
|
|
|
|
|
|
@mock_ec2
|
|
def test_get_instance_by_security_group():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
ec2 = boto3.resource("ec2", region_name="us-east-1")
|
|
|
|
instance = ec2.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)[0]
|
|
|
|
security_group = ec2.create_security_group(
|
|
GroupName=str(uuid4())[0:6], Description="test"
|
|
)
|
|
|
|
with pytest.raises(ClientError) as ex:
|
|
client.modify_instance_attribute(
|
|
InstanceId=instance.id, Groups=[security_group.id], DryRun=True
|
|
)
|
|
assert ex.value.response["Error"]["Code"] == "DryRunOperation"
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 412
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "An error occurred (DryRunOperation) when calling the ModifyInstanceAttribute operation: Request would have succeeded, but DryRun flag is set"
|
|
)
|
|
|
|
client.modify_instance_attribute(InstanceId=instance.id, Groups=[security_group.id])
|
|
|
|
instance.reload()
|
|
security_group_instances = instance.describe_attribute(Attribute="groupSet")[
|
|
"Groups"
|
|
]
|
|
|
|
assert len(security_group_instances) == 1
|
|
assert security_group_instances == [{"GroupId": security_group.id}]
|
|
|
|
|
|
@mock_ec2
|
|
def test_modify_delete_on_termination():
|
|
ec2_client = boto3.resource("ec2", region_name="us-west-1")
|
|
result = ec2_client.create_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)
|
|
instance = result[0]
|
|
instance.load()
|
|
assert instance.block_device_mappings[0]["Ebs"]["DeleteOnTermination"] is True
|
|
instance.modify_attribute(
|
|
BlockDeviceMappings=[
|
|
{"DeviceName": "/dev/sda1", "Ebs": {"DeleteOnTermination": False}}
|
|
]
|
|
)
|
|
instance.load()
|
|
assert instance.block_device_mappings[0]["Ebs"]["DeleteOnTermination"] is False
|
|
|
|
|
|
@mock_ec2
|
|
def test_create_instance_with_default_options():
|
|
client = boto3.client("ec2", region_name="eu-west-1")
|
|
|
|
def assert_instance(instance):
|
|
# TODO: Add additional asserts for default instance response
|
|
assert instance["ImageId"] == EXAMPLE_AMI_ID
|
|
assert "KeyName" not in instance
|
|
|
|
resp = client.run_instances(ImageId=EXAMPLE_AMI_ID, MaxCount=1, MinCount=1)
|
|
assert_instance(resp["Instances"][0])
|
|
|
|
resp = client.describe_instances(InstanceIds=[resp["Instances"][0]["InstanceId"]])
|
|
assert_instance(resp["Reservations"][0]["Instances"][0])
|
|
|
|
|
|
@mock_ec2
|
|
def test_create_instance_ebs_optimized():
|
|
ec2_resource = boto3.resource("ec2", region_name="eu-west-1")
|
|
|
|
instance = ec2_resource.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MaxCount=1, MinCount=1, EbsOptimized=True
|
|
)[0]
|
|
instance.load()
|
|
assert instance.ebs_optimized is True
|
|
|
|
instance.modify_attribute(EbsOptimized={"Value": False})
|
|
instance.load()
|
|
assert instance.ebs_optimized is False
|
|
|
|
instance = ec2_resource.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MaxCount=1, MinCount=1
|
|
)[0]
|
|
instance.load()
|
|
assert instance.ebs_optimized is False
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_multiple_instances_in_same_command():
|
|
instance_count = 4
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
instances = client.run_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=instance_count, MaxCount=instance_count
|
|
)
|
|
reservation_id = instances["ReservationId"]
|
|
|
|
# TODO: use this filter when implemented
|
|
# client.describe_instances(Filters=[{"Name": "reservation-id", "Values": [instances["ReservationId"]]}])["Reservations"]
|
|
all_reservations = retrieve_all_reservations(client)
|
|
my_reservation = [
|
|
r for r in all_reservations if r["ReservationId"] == reservation_id
|
|
][0]
|
|
|
|
assert len(my_reservation["Instances"]) == instance_count
|
|
|
|
instances = my_reservation["Instances"]
|
|
for i in range(0, instance_count):
|
|
assert instances[i]["AmiLaunchIndex"] == i
|
|
|
|
|
|
@mock_ec2
|
|
def test_describe_instance_attribute():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
security_group_id = client.create_security_group(
|
|
GroupName=str(uuid4()), Description="this is a test security group"
|
|
)["GroupId"]
|
|
resp = client.run_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
SecurityGroupIds=[security_group_id],
|
|
)
|
|
instance_id = resp["Instances"][0]["InstanceId"]
|
|
|
|
valid_instance_attributes = [
|
|
"instanceType",
|
|
"kernel",
|
|
"ramdisk",
|
|
"userData",
|
|
"disableApiTermination",
|
|
"instanceInitiatedShutdownBehavior",
|
|
"rootDeviceName",
|
|
"blockDeviceMapping",
|
|
"productCodes",
|
|
"sourceDestCheck",
|
|
"groupSet",
|
|
"ebsOptimized",
|
|
"sriovNetSupport",
|
|
]
|
|
|
|
for valid_instance_attribute in valid_instance_attributes:
|
|
response = client.describe_instance_attribute(
|
|
InstanceId=instance_id, Attribute=valid_instance_attribute
|
|
)
|
|
if valid_instance_attribute == "groupSet":
|
|
assert "Groups" in response
|
|
assert len(response["Groups"]) == 1
|
|
assert response["Groups"][0]["GroupId"] == security_group_id
|
|
elif valid_instance_attribute == "userData":
|
|
assert "UserData" in response
|
|
assert response["UserData"] == {}
|
|
|
|
invalid_instance_attributes = [
|
|
"abc",
|
|
"Kernel",
|
|
"RamDisk",
|
|
"userdata",
|
|
"iNsTaNcEtYpE",
|
|
]
|
|
|
|
for invalid_instance_attribute in invalid_instance_attributes:
|
|
with pytest.raises(ClientError) as ex:
|
|
client.describe_instance_attribute(
|
|
InstanceId=instance_id, Attribute=invalid_instance_attribute
|
|
)
|
|
assert ex.value.response["Error"]["Code"] == "InvalidParameterValue"
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
message = f"Value ({invalid_instance_attribute}) for parameter attribute is invalid. Unknown attribute."
|
|
assert ex.value.response["Error"]["Message"] == message
|
|
|
|
|
|
@mock_ec2
|
|
def test_warn_on_invalid_ami():
|
|
if settings.TEST_SERVER_MODE:
|
|
raise SkipTest("Can't capture warnings in server mode.")
|
|
ec2 = boto3.resource("ec2", "us-east-1")
|
|
with pytest.warns(
|
|
PendingDeprecationWarning,
|
|
match=r"Could not find AMI with image-id:invalid-ami.+",
|
|
):
|
|
ec2.create_instances(ImageId="invalid-ami", MinCount=1, MaxCount=1)
|
|
|
|
|
|
@mock_ec2
|
|
@mock.patch(
|
|
"moto.ec2.models.instances.settings.ENABLE_AMI_VALIDATION",
|
|
new_callable=mock.PropertyMock(return_value=True),
|
|
)
|
|
def test_error_on_invalid_ami(m_flag):
|
|
if settings.TEST_SERVER_MODE:
|
|
raise SkipTest("Can't capture warnings in server mode.")
|
|
ec2 = boto3.resource("ec2", "us-east-1")
|
|
with pytest.raises(ClientError) as ex:
|
|
ec2.create_instances(ImageId="ami-invalid", MinCount=1, MaxCount=1)
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
assert ex.value.response["Error"]["Code"] == "InvalidAMIID.NotFound"
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "The image id '[['ami-invalid']]' does not exist"
|
|
)
|
|
|
|
assert m_flag is True
|
|
|
|
|
|
@mock_ec2
|
|
@mock.patch(
|
|
"moto.ec2.models.instances.settings.ENABLE_AMI_VALIDATION",
|
|
new_callable=mock.PropertyMock(return_value=True),
|
|
)
|
|
def test_error_on_invalid_ami_format(m_flag):
|
|
if settings.TEST_SERVER_MODE:
|
|
raise SkipTest(
|
|
"It is not possible to set the environment variable in server mode"
|
|
)
|
|
ec2 = boto3.resource("ec2", "us-east-1")
|
|
with pytest.raises(ClientError) as ex:
|
|
ec2.create_instances(ImageId="invalid-ami-format", MinCount=1, MaxCount=1)
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
assert ex.value.response["Error"]["Code"] == "InvalidAMIID.Malformed"
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== 'Invalid id: "[\'invalid-ami-format\']" (expecting "ami-...")'
|
|
)
|
|
|
|
assert m_flag is True
|
|
|
|
|
|
@mock_ec2
|
|
def test_filter_wildcard_in_specified_tag_only():
|
|
ec2_client = boto3.client("ec2", region_name="us-west-1")
|
|
|
|
name = str(uuid4())[0:6]
|
|
tags_name = [{"Key": "Name", "Value": f"{name} in wonderland"}]
|
|
ec2_client.run_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MaxCount=1,
|
|
MinCount=1,
|
|
TagSpecifications=[{"ResourceType": "instance", "Tags": tags_name}],
|
|
)
|
|
|
|
tags_owner = [{"Key": "Owner", "Value": f"{name} in wonderland"}]
|
|
ec2_client.run_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MaxCount=1,
|
|
MinCount=1,
|
|
TagSpecifications=[{"ResourceType": "instance", "Tags": tags_owner}],
|
|
)
|
|
|
|
# should only match the Name tag
|
|
response = ec2_client.describe_instances(
|
|
Filters=[{"Name": "tag:Name", "Values": [f"*{name}*"]}]
|
|
)
|
|
instances = [i for r in response["Reservations"] for i in r["Instances"]]
|
|
assert len(instances) == 1
|
|
assert instances[0]["Tags"][0]["Key"] == "Name"
|
|
|
|
|
|
@mock_ec2
|
|
def test_instance_termination_protection():
|
|
client = boto3.client("ec2", region_name="us-west-1")
|
|
|
|
resp = client.run_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)
|
|
instance_id = resp["Instances"][0]["InstanceId"]
|
|
|
|
client.modify_instance_attribute(
|
|
InstanceId=instance_id, DisableApiTermination={"Value": True}
|
|
)
|
|
client.stop_instances(InstanceIds=[instance_id], Force=True)
|
|
|
|
with pytest.raises(ClientError) as ex:
|
|
client.terminate_instances(InstanceIds=[instance_id])
|
|
error = ex.value.response["Error"]
|
|
assert error["Code"] == "OperationNotPermitted"
|
|
assert (
|
|
f"The instance '{instance_id}' may not be terminated"
|
|
in ex.value.response["Error"]["Message"]
|
|
)
|
|
|
|
# Use alternate request syntax for setting attribute.
|
|
client.modify_instance_attribute(
|
|
InstanceId=instance_id, Attribute="disableApiTermination", Value="false"
|
|
)
|
|
client.terminate_instances(InstanceIds=[instance_id])
|
|
|
|
resp = client.describe_instances(InstanceIds=[instance_id])
|
|
instances = resp["Reservations"][0]["Instances"]
|
|
assert len(instances) == 1
|
|
instance = instances[0]
|
|
assert instance["State"]["Name"] == "terminated"
|
|
|
|
|
|
@mock_ec2
|
|
def test_terminate_unknown_instances():
|
|
client = boto3.client("ec2", region_name="us-west-1")
|
|
|
|
# Correct error message for single unknown instance
|
|
with pytest.raises(ClientError) as ex:
|
|
client.terminate_instances(InstanceIds=["i-12345678"])
|
|
error = ex.value.response["Error"]
|
|
assert error["Code"] == "InvalidInstanceID.NotFound"
|
|
assert error["Message"] == "The instance ID 'i-12345678' does not exist"
|
|
|
|
# Correct error message for multiple unknown instances
|
|
with pytest.raises(ClientError) as ex:
|
|
client.terminate_instances(InstanceIds=["i-12345678", "i-12345668"])
|
|
error = ex.value.response["Error"]
|
|
assert error["Code"] == "InvalidInstanceID.NotFound"
|
|
assert error["Message"] == "The instance IDs 'i-12345678, i-12345668' do not exist"
|
|
|
|
# Create an instance
|
|
resp = client.run_instances(ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1)
|
|
instance_id = resp["Instances"][0]["InstanceId"]
|
|
|
|
# Correct error message if one instance is known
|
|
with pytest.raises(ClientError) as ex:
|
|
client.terminate_instances(InstanceIds=["i-12345678", instance_id])
|
|
error = ex.value.response["Error"]
|
|
assert error["Code"] == "InvalidInstanceID.NotFound"
|
|
assert error["Message"] == "The instance ID 'i-12345678' does not exist"
|
|
|
|
# status = still running
|
|
resp = client.describe_instances(InstanceIds=[instance_id])
|
|
instance = resp["Reservations"][0]["Instances"][0]
|
|
assert instance["State"]["Name"] == "running"
|
|
|
|
|
|
@mock_ec2
|
|
def test_instance_lifecycle():
|
|
ec2_resource = boto3.resource("ec2", "us-west-1")
|
|
|
|
result = ec2_resource.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
BlockDeviceMappings=[
|
|
{
|
|
"DeviceName": "/dev/sda1",
|
|
"Ebs": {"VolumeSize": 50, "DeleteOnTermination": True},
|
|
}
|
|
],
|
|
)
|
|
instance = result[0]
|
|
|
|
assert instance.instance_lifecycle is None
|
|
|
|
|
|
# The default AMIs are not loaded for our test case, to speed things up
|
|
# But we do need it for this specific test
|
|
@mock.patch.dict(os.environ, {"MOTO_EC2_LOAD_DEFAULT_AMIS": "true"})
|
|
@mock_ec2
|
|
@pytest.mark.parametrize(
|
|
"launch_template_kind", ("LaunchTemplateId", "LaunchTemplateName")
|
|
)
|
|
def test_create_instance_with_launch_template_id_produces_no_warning(
|
|
launch_template_kind,
|
|
):
|
|
if settings.TEST_SERVER_MODE:
|
|
raise SkipTest("Can't set environment variables in ServerMode")
|
|
client, resource = (
|
|
boto3.client("ec2", region_name="us-west-1"),
|
|
boto3.resource("ec2", region_name="us-west-1"),
|
|
)
|
|
|
|
template = client.create_launch_template(
|
|
LaunchTemplateName=str(uuid4()), LaunchTemplateData={"ImageId": EXAMPLE_AMI_ID}
|
|
)["LaunchTemplate"]
|
|
|
|
with warnings.catch_warnings(record=True) as ws:
|
|
resource.create_instances(
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
LaunchTemplate={launch_template_kind: template[launch_template_kind]},
|
|
)
|
|
# We could have other warnings in this method, coming from botocore for instance
|
|
# But we should not receive a warning that the AMI could not be found
|
|
messages = [str(w.message) for w in ws]
|
|
assert all(["Could not find AMI" not in msg for msg in messages])
|
|
|
|
|
|
@mock_ec2
|
|
def test_create_instance_from_launch_template__process_tags():
|
|
client = boto3.client("ec2", region_name="us-west-1")
|
|
|
|
template = client.create_launch_template(
|
|
LaunchTemplateName=str(uuid4()),
|
|
LaunchTemplateData={
|
|
"ImageId": EXAMPLE_AMI_ID,
|
|
"TagSpecifications": [
|
|
{"ResourceType": "instance", "Tags": [{"Key": "k", "Value": "v"}]}
|
|
],
|
|
},
|
|
)["LaunchTemplate"]
|
|
|
|
instance = client.run_instances(
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
LaunchTemplate={"LaunchTemplateId": template["LaunchTemplateId"]},
|
|
)["Instances"][0]
|
|
|
|
assert instance["Tags"] == [{"Key": "k", "Value": "v"}]
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_and_associate_public_ip():
|
|
ec2 = boto3.resource("ec2", "us-west-1")
|
|
vpc = ec2.create_vpc(CidrBlock="10.0.0.0/16")
|
|
subnet = ec2.create_subnet(VpcId=vpc.id, CidrBlock="10.0.0.0/18")
|
|
|
|
# Do not pass AssociatePublicIpAddress-argument
|
|
instance = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
NetworkInterfaces=[{"DeviceIndex": 0, "SubnetId": subnet.id}],
|
|
)[0]
|
|
interfaces = instance.network_interfaces_attribute
|
|
addresses = interfaces[0]["PrivateIpAddresses"][0]
|
|
assert addresses["Primary"] is True
|
|
assert "PrivateIpAddress" in addresses
|
|
assert "Association" not in addresses
|
|
|
|
# Pass AssociatePublicIpAddress=False
|
|
instance = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
NetworkInterfaces=[
|
|
{"DeviceIndex": 0, "SubnetId": subnet.id, "AssociatePublicIpAddress": False}
|
|
],
|
|
)[0]
|
|
interfaces = instance.network_interfaces_attribute
|
|
addresses = interfaces[0]["PrivateIpAddresses"][0]
|
|
assert addresses["Primary"] is True
|
|
assert "PrivateIpAddress" in addresses
|
|
assert "Association" not in addresses
|
|
|
|
# Pass AssociatePublicIpAddress=True
|
|
instance = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
NetworkInterfaces=[
|
|
{"DeviceIndex": 0, "SubnetId": subnet.id, "AssociatePublicIpAddress": True}
|
|
],
|
|
)[0]
|
|
interfaces = instance.network_interfaces_attribute
|
|
addresses = interfaces[0]["PrivateIpAddresses"][0]
|
|
assert addresses["Primary"] is True
|
|
assert "PrivateIpAddress" in addresses
|
|
assert "Association" in addresses
|
|
# Only now should we have a PublicIp
|
|
assert addresses["Association"]["IpOwnerId"] == ACCOUNT_ID
|
|
assert "PublicIp" in addresses["Association"]
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_cannot_have_subnet_and_networkinterface_parameter():
|
|
ec2 = boto3.resource("ec2", "us-west-1")
|
|
vpc = ec2.create_vpc(CidrBlock="10.0.0.0/16")
|
|
subnet = ec2.create_subnet(VpcId=vpc.id, CidrBlock="10.0.0.0/18")
|
|
|
|
with pytest.raises(ClientError) as exc:
|
|
ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
SubnetId=subnet.id,
|
|
NetworkInterfaces=[{"DeviceIndex": 0}],
|
|
)
|
|
err = exc.value.response["Error"]
|
|
assert err["Code"] == "InvalidParameterCombination"
|
|
assert (
|
|
err["Message"]
|
|
== "Network interfaces and an instance-level subnet ID may not be specified on the same request"
|
|
)
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_in_subnet_with_nic_private_ip():
|
|
vpc_cidr_block = "10.26.0.0/16"
|
|
subnet_cidr_block = "10.26.1.0/24"
|
|
private_ip = "10.26.1.3"
|
|
ec2 = boto3.resource("ec2", region_name="eu-west-1")
|
|
vpc = ec2.create_vpc(CidrBlock=vpc_cidr_block)
|
|
subnet = ec2.create_subnet(
|
|
VpcId=vpc.id,
|
|
CidrBlock=subnet_cidr_block,
|
|
)
|
|
my_interface = {
|
|
"SubnetId": subnet.id,
|
|
"DeviceIndex": 0,
|
|
"PrivateIpAddress": private_ip,
|
|
}
|
|
[instance] = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, NetworkInterfaces=[my_interface], MinCount=1, MaxCount=1
|
|
)
|
|
assert instance.private_ip_address == private_ip
|
|
|
|
interfaces = instance.network_interfaces_attribute
|
|
address = interfaces[0]["PrivateIpAddresses"][0]
|
|
assert "Association" not in address
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_instance_in_subnet_with_nic_private_ip_and_public_association():
|
|
vpc_cidr_block = "10.26.0.0/16"
|
|
subnet_cidr_block = "10.26.1.0/24"
|
|
primary_private_ip = "10.26.1.3"
|
|
other_private_ip = "10.26.1.4"
|
|
ec2 = boto3.resource("ec2", region_name="eu-west-1")
|
|
vpc = ec2.create_vpc(CidrBlock=vpc_cidr_block)
|
|
subnet = ec2.create_subnet(
|
|
VpcId=vpc.id,
|
|
CidrBlock=subnet_cidr_block,
|
|
)
|
|
my_interface = {
|
|
"SubnetId": subnet.id,
|
|
"DeviceIndex": 0,
|
|
"AssociatePublicIpAddress": True,
|
|
"PrivateIpAddresses": [
|
|
{"Primary": True, "PrivateIpAddress": primary_private_ip},
|
|
{"Primary": False, "PrivateIpAddress": other_private_ip},
|
|
],
|
|
}
|
|
[instance] = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, NetworkInterfaces=[my_interface], MinCount=1, MaxCount=1
|
|
)
|
|
assert instance.private_ip_address == primary_private_ip
|
|
|
|
interfaces = instance.network_interfaces_attribute
|
|
address = interfaces[0]["PrivateIpAddresses"][0]
|
|
assert address["Association"]["IpOwnerId"] == ACCOUNT_ID
|
|
|
|
|
|
@mock_ec2
|
|
def test_describe_instances_dryrun():
|
|
client = boto3.client("ec2", region_name="us-east-1")
|
|
|
|
with pytest.raises(ClientError) as ex:
|
|
client.describe_instances(DryRun=True)
|
|
assert ex.value.response["ResponseMetadata"]["HTTPStatusCode"] == 412
|
|
assert ex.value.response["Error"]["Code"] == "DryRunOperation"
|
|
assert (
|
|
ex.value.response["Error"]["Message"]
|
|
== "An error occurred (DryRunOperation) when calling the DescribeInstances operation: Request would have succeeded, but DryRun flag is set"
|
|
)
|
|
|
|
|
|
@mock_ec2
|
|
def test_describe_instances_filter_vpcid_via_networkinterface():
|
|
vpc_cidr_block = "10.26.0.0/16"
|
|
subnet_cidr_block = "10.26.1.0/24"
|
|
ec2 = boto3.resource("ec2", region_name="eu-west-1")
|
|
vpc = ec2.create_vpc(CidrBlock=vpc_cidr_block)
|
|
subnet = ec2.create_subnet(
|
|
VpcId=vpc.id, CidrBlock=subnet_cidr_block, AvailabilityZone="eu-west-1a"
|
|
)
|
|
my_interface = {
|
|
"SubnetId": subnet.id,
|
|
"DeviceIndex": 0,
|
|
"PrivateIpAddresses": [{"Primary": True, "PrivateIpAddress": "10.26.1.3"}],
|
|
}
|
|
instance = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID, NetworkInterfaces=[my_interface], MinCount=1, MaxCount=1
|
|
)[0]
|
|
|
|
_filter = [{"Name": "vpc-id", "Values": [vpc.id]}]
|
|
found = list(ec2.instances.filter(Filters=_filter))
|
|
assert len(found) == 1
|
|
assert found == [instance]
|
|
|
|
|
|
@mock_ec2
|
|
@mock_iam
|
|
def test_instance_iam_instance_profile():
|
|
ec2_resource = boto3.resource("ec2", "us-west-1")
|
|
iam = boto3.client("iam", "us-west-1")
|
|
profile_name = "fake_profile"
|
|
profile = iam.create_instance_profile(
|
|
InstanceProfileName=profile_name,
|
|
)
|
|
|
|
result1 = ec2_resource.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
IamInstanceProfile={
|
|
"Name": profile_name,
|
|
},
|
|
)
|
|
instance = result1[0]
|
|
assert "Arn" in instance.iam_instance_profile
|
|
assert "Id" in instance.iam_instance_profile
|
|
assert profile["InstanceProfile"]["Arn"] == instance.iam_instance_profile["Arn"]
|
|
|
|
result2 = ec2_resource.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
IamInstanceProfile={
|
|
"Arn": profile["InstanceProfile"]["Arn"],
|
|
},
|
|
)
|
|
instance = result2[0]
|
|
assert "Arn" in instance.iam_instance_profile
|
|
assert "Id" in instance.iam_instance_profile
|
|
assert profile["InstanceProfile"]["Arn"] == instance.iam_instance_profile["Arn"]
|
|
|
|
tag_key = str(uuid4())[0:6]
|
|
with pytest.raises(ClientError) as exc:
|
|
ec2_resource.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=1,
|
|
MaxCount=1,
|
|
IamInstanceProfile={"Arn": "unknown:instance:profile"},
|
|
TagSpecifications=[
|
|
{"ResourceType": "instance", "Tags": [{"Key": tag_key, "Value": "val"}]}
|
|
],
|
|
)
|
|
err = exc.value.response["Error"]
|
|
assert err["Code"] == "NoSuchEntity"
|
|
assert err["Message"] == "Instance profile unknown:instance:profile not found"
|
|
|
|
ec2_client = boto3.client("ec2", "us-west-1")
|
|
filters = [{"Name": "tag-key", "Values": [tag_key]}]
|
|
assert retrieve_all_instances(ec2_client, filters) == []
|
|
|
|
|
|
def retrieve_all_reservations(client, filters=[]): # pylint: disable=W0102
|
|
resp = client.describe_instances(Filters=filters)
|
|
all_reservations = resp["Reservations"]
|
|
next_token = resp.get("NextToken")
|
|
while next_token:
|
|
resp = client.describe_instances(Filters=filters, NextToken=next_token)
|
|
all_reservations.extend(resp["Reservations"])
|
|
next_token = resp.get("NextToken")
|
|
return all_reservations
|
|
|
|
|
|
def retrieve_all_instances(client, filters=[]): # pylint: disable=W0102
|
|
reservations = retrieve_all_reservations(client, filters)
|
|
return [i for r in reservations for i in r["Instances"]]
|
|
|
|
|
|
@mock_ec2
|
|
def test_run_multiple_instances_with_single_nic_template():
|
|
ec2 = boto3.resource("ec2", "us-west-1")
|
|
client = boto3.client("ec2", "us-west-1")
|
|
vpc = ec2.create_vpc(CidrBlock="10.0.0.0/16")
|
|
subnet = ec2.create_subnet(VpcId=vpc.id, CidrBlock="10.0.0.0/18")
|
|
security_group1 = ec2.create_security_group(
|
|
GroupName=str(uuid4()), Description="n/a"
|
|
)
|
|
|
|
instances = ec2.create_instances(
|
|
ImageId=EXAMPLE_AMI_ID,
|
|
MinCount=2,
|
|
MaxCount=2,
|
|
NetworkInterfaces=[
|
|
{
|
|
"AssociatePublicIpAddress": False,
|
|
"DeleteOnTermination": True,
|
|
"DeviceIndex": 0,
|
|
"Groups": [security_group1.group_id],
|
|
"SubnetId": subnet.id,
|
|
"InterfaceType": "interface",
|
|
}
|
|
],
|
|
)
|
|
|
|
enis = []
|
|
|
|
for instance in instances:
|
|
instance_eni = instance.network_interfaces_attribute
|
|
assert len(instance_eni) == 1
|
|
|
|
nii = instance_eni[0]["NetworkInterfaceId"]
|
|
|
|
my_enis = client.describe_network_interfaces(NetworkInterfaceIds=[nii])[
|
|
"NetworkInterfaces"
|
|
]
|
|
assert len(my_enis) == 1
|
|
eni = my_enis[0]
|
|
|
|
assert instance.subnet_id == subnet.id
|
|
|
|
assert eni["SubnetId"] == subnet.id
|
|
assert len(eni["Groups"]) == 1
|
|
assert [group["GroupId"] for group in eni["Groups"]] == [security_group1.id]
|
|
assert len(eni["PrivateIpAddresses"]) == 1
|
|
assert eni["PrivateIpAddresses"][0]["PrivateIpAddress"] is not None
|
|
|
|
enis.append(eni)
|
|
|
|
instance_0_ip = enis[0]["PrivateIpAddresses"][0]["PrivateIpAddress"]
|
|
instance_1_ip = enis[1]["PrivateIpAddresses"][0]["PrivateIpAddress"]
|
|
|
|
assert instance_0_ip != instance_1_ip
|
|
|
|
|
|
@mock_ec2
|
|
def test_describe_instance_without_enhanced_monitoring():
|
|
conn = boto3.client("ec2", region_name="us-west-1")
|
|
|
|
instance = conn.run_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, Monitoring={"Enabled": True}
|
|
)
|
|
assert instance["Instances"][0]["Monitoring"] == {"State": "enabled"}
|
|
|
|
result = conn.describe_instances(
|
|
InstanceIds=[instance["Instances"][0]["InstanceId"]]
|
|
)["Reservations"][0]["Instances"]
|
|
|
|
assert result[0]["Monitoring"] == {"State": "enabled"}
|
|
|
|
|
|
@mock_ec2
|
|
def test_describe_instance_with_enhanced_monitoring():
|
|
conn = boto3.client("ec2", region_name="us-west-1")
|
|
|
|
instance = conn.run_instances(
|
|
ImageId=EXAMPLE_AMI_ID, MinCount=1, MaxCount=1, Monitoring={"Enabled": False}
|
|
)
|
|
assert instance["Instances"][0]["Monitoring"] == {"State": "disabled"}
|
|
|
|
result = conn.describe_instances(
|
|
InstanceIds=[instance["Instances"][0]["InstanceId"]]
|
|
)["Reservations"][0]["Instances"]
|
|
|
|
assert result[0]["Monitoring"] == {"State": "disabled"}
|