ECS: Fix container memory validation (#7280)

This commit is contained in:
Miki Watanabe 2024-02-05 16:16:58 -05:00 committed by GitHub
parent c70ca1d477
commit 9aef6944da
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
2 changed files with 32 additions and 7 deletions

View File

@ -1178,8 +1178,9 @@ class EC2ContainerServiceBackend(BaseBackend):
raise EcsClientException( raise EcsClientException(
f"Tasks using the Fargate launch type do not support pidMode '{pid_mode}'. The supported value for pidMode is 'task'." f"Tasks using the Fargate launch type do not support pidMode '{pid_mode}'. The supported value for pidMode is 'task'."
) )
self._validate_container_defs(
self._validate_container_defs(memory, container_definitions) memory, container_definitions, requires_compatibilities
)
if family in self.task_definitions: if family in self.task_definitions:
last_id = self._get_last_task_definition_revision_id(family) last_id = self._get_last_task_definition_revision_id(family)
@ -1214,13 +1215,19 @@ class EC2ContainerServiceBackend(BaseBackend):
return task_definition return task_definition
@staticmethod @staticmethod
def _validate_container_defs(memory: Optional[str], container_definitions: List[Dict[str, Any]]) -> None: # type: ignore[misc] def _validate_container_defs(memory: Optional[str], container_definitions: List[Dict[str, Any]], requires_compatibilities: Optional[List[str]]) -> None: # type: ignore[misc]
# The capitalised keys are passed by Cloudformation # The capitalised keys are passed by Cloudformation
for cd in container_definitions: for cd in container_definitions:
if "name" not in cd and "Name" not in cd: if "name" not in cd and "Name" not in cd:
raise TaskDefinitionMissingPropertyError("name") raise TaskDefinitionMissingPropertyError("name")
if "image" not in cd and "Image" not in cd: if "image" not in cd and "Image" not in cd:
raise TaskDefinitionMissingPropertyError("image") raise TaskDefinitionMissingPropertyError("image")
if (
requires_compatibilities
and "EC2" in requires_compatibilities
and ("memory" not in cd and "Memory" not in cd and not memory)
):
raise TaskDefinitionMemoryError(cd["name"])
if ( if (
"memory" not in cd "memory" not in cd
and "Memory" not in cd and "Memory" not in cd

View File

@ -324,16 +324,16 @@ def test_register_task_definition_fargate_with_pid_mode():
@mock_aws @mock_aws
def test_register_task_definition_memory_validation(): def test_register_task_definition_memory_validation_ec2():
client = boto3.client("ecs", region_name=ECS_REGION) client = boto3.client("ecs", region_name=ECS_REGION)
container_name = "hello_world" container_name = "hello_world"
bad_definition1 = dict( bad_definition1 = dict(
family="test_ecs_task", family="test_ecs_task",
containerDefinitions=[ containerDefinitions=[
{"name": container_name, "image": "hello-world:latest", "memory": 400}, {"name": container_name, "image": "hello-world:latest"},
{"name": f"{container_name}2", "image": "hello-world:latest"}, {"name": f"{container_name}2", "image": "hello-world:latest"},
], ],
requiresCompatibilities=["FARGATE"], requiresCompatibilities=["EC2"],
) )
with pytest.raises(ClientError) as exc: with pytest.raises(ClientError) as exc:
@ -344,10 +344,28 @@ def test_register_task_definition_memory_validation():
assert ex.response["Error"]["Code"] == "ClientException" assert ex.response["Error"]["Code"] == "ClientException"
assert ( assert (
ex.response["Error"]["Message"] ex.response["Error"]["Message"]
== f"Invalid setting for container '{container_name}2'. At least one of 'memory' or 'memoryReservation' must be specified." == f"Invalid setting for container '{container_name}'. At least one of 'memory' or 'memoryReservation' must be specified."
) )
@mock_aws
def test_register_task_definition_memory_validation_fargate():
client = boto3.client("ecs", region_name=ECS_REGION)
container_name = "hello_world"
good_definition1 = dict(
family="test_ecs_task",
memory="1024",
containerDefinitions=[
{"name": container_name, "image": "hello-world:latest"},
{"name": f"{container_name}2", "image": "hello-world:latest"},
],
requiresCompatibilities=["FARGATE"],
)
response = client.register_task_definition(**good_definition1)
assert response["ResponseMetadata"]["HTTPStatusCode"] == 200
@mock_aws @mock_aws
@pytest.mark.parametrize( @pytest.mark.parametrize(
"ecs_def,missing_prop", "ecs_def,missing_prop",