2022-02-20 20:53:25 +00:00
|
|
|
from . import _get_clients
|
2021-08-04 12:40:10 +00:00
|
|
|
import random
|
2021-11-01 10:31:22 +00:00
|
|
|
import pytest
|
2022-02-20 20:53:25 +00:00
|
|
|
from moto import mock_batch
|
2021-10-05 17:11:07 +00:00
|
|
|
from uuid import uuid4
|
2021-08-04 12:40:10 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_batch
|
2021-11-01 10:31:22 +00:00
|
|
|
@pytest.mark.parametrize("use_resource_reqs", [True, False])
|
|
|
|
def test_register_task_definition(use_resource_reqs):
|
2022-02-20 20:53:25 +00:00
|
|
|
_, _, _, _, batch_client = _get_clients()
|
2021-08-04 12:40:10 +00:00
|
|
|
|
2021-11-01 10:31:22 +00:00
|
|
|
resp = register_job_def(batch_client, use_resource_reqs=use_resource_reqs)
|
2021-08-04 12:40:10 +00:00
|
|
|
|
2023-06-16 10:42:07 +00:00
|
|
|
assert "jobDefinitionArn" in resp
|
|
|
|
assert "jobDefinitionName" in resp
|
|
|
|
assert "revision" in resp
|
2021-08-04 12:40:10 +00:00
|
|
|
|
2023-06-16 10:42:07 +00:00
|
|
|
assert f"{resp['jobDefinitionName']}:{resp['revision']}" in resp["jobDefinitionArn"]
|
2021-08-04 12:40:10 +00:00
|
|
|
|
|
|
|
|
2021-09-21 16:12:18 +00:00
|
|
|
@mock_batch
|
2022-02-20 20:53:25 +00:00
|
|
|
@pytest.mark.parametrize("propagate_tags", [None, True, False])
|
|
|
|
def test_register_task_definition_with_tags(propagate_tags):
|
|
|
|
_, _, _, _, batch_client = _get_clients()
|
2021-09-21 16:12:18 +00:00
|
|
|
|
2022-02-20 22:54:05 +00:00
|
|
|
job_def_name = str(uuid4())[0:8]
|
|
|
|
register_job_def_with_tags(batch_client, job_def_name, propagate_tags)
|
2021-09-21 16:12:18 +00:00
|
|
|
|
2022-02-20 22:54:05 +00:00
|
|
|
resp = batch_client.describe_job_definitions(jobDefinitionName=job_def_name)
|
2022-02-20 20:53:25 +00:00
|
|
|
job_def = resp["jobDefinitions"][0]
|
|
|
|
if propagate_tags is None:
|
2023-06-16 10:42:07 +00:00
|
|
|
assert "propagateTags" not in job_def
|
2022-02-20 20:53:25 +00:00
|
|
|
else:
|
2023-06-16 10:42:07 +00:00
|
|
|
assert job_def["propagateTags"] == propagate_tags
|
2021-09-21 16:12:18 +00:00
|
|
|
|
2022-02-20 20:53:25 +00:00
|
|
|
|
|
|
|
@mock_batch
|
|
|
|
@pytest.mark.parametrize("platform_capability", ["EC2", "FARGATE"])
|
|
|
|
def test_register_task_definition_with_platform_capability(platform_capability):
|
|
|
|
_, _, _, _, batch_client = _get_clients()
|
|
|
|
|
|
|
|
def_name = str(uuid4())[0:6]
|
|
|
|
batch_client.register_job_definition(
|
|
|
|
jobDefinitionName=def_name,
|
|
|
|
type="container",
|
|
|
|
containerProperties={
|
|
|
|
"image": "busybox",
|
|
|
|
"vcpus": 1,
|
|
|
|
"memory": 4,
|
|
|
|
"command": ["exit", "0"],
|
|
|
|
},
|
|
|
|
platformCapabilities=[platform_capability],
|
|
|
|
)
|
|
|
|
|
2023-08-21 20:33:16 +00:00
|
|
|
job_def = batch_client.describe_job_definitions(jobDefinitionName=def_name)[
|
|
|
|
"jobDefinitions"
|
|
|
|
][0]
|
|
|
|
assert job_def["platformCapabilities"] == [platform_capability]
|
|
|
|
|
|
|
|
container_props = job_def["containerProperties"]
|
|
|
|
assert container_props["resourceRequirements"] == []
|
|
|
|
assert container_props["secrets"] == []
|
|
|
|
if platform_capability == "FARGATE":
|
|
|
|
assert container_props["fargatePlatformConfiguration"] == {
|
|
|
|
"platformVersion": "LATEST"
|
|
|
|
}
|
|
|
|
else:
|
|
|
|
assert "fargatePlatformConfiguration" not in container_props
|
|
|
|
|
|
|
|
|
|
|
|
@mock_batch
|
|
|
|
def test_register_task_definition_without_command():
|
|
|
|
_, _, _, _, batch_client = _get_clients()
|
|
|
|
|
|
|
|
def_name = str(uuid4())[0:6]
|
|
|
|
batch_client.register_job_definition(
|
|
|
|
jobDefinitionName=def_name,
|
|
|
|
type="container",
|
|
|
|
containerProperties={
|
|
|
|
"image": "busybox",
|
|
|
|
"vcpus": 1,
|
|
|
|
"memory": 4,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
job_def = batch_client.describe_job_definitions(jobDefinitionName=def_name)[
|
|
|
|
"jobDefinitions"
|
|
|
|
][0]
|
|
|
|
|
|
|
|
container_props = job_def["containerProperties"]
|
|
|
|
assert container_props["command"] == []
|
2022-02-20 20:53:25 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_batch
|
|
|
|
def test_register_task_definition_with_retry_strategies():
|
|
|
|
_, _, _, _, batch_client = _get_clients()
|
|
|
|
|
|
|
|
def_name = str(uuid4())[0:6]
|
|
|
|
batch_client.register_job_definition(
|
|
|
|
jobDefinitionName=def_name,
|
|
|
|
type="container",
|
|
|
|
containerProperties={
|
|
|
|
"image": "busybox",
|
|
|
|
"vcpus": 1,
|
|
|
|
"memory": 4,
|
|
|
|
"command": ["exit", "0"],
|
|
|
|
},
|
|
|
|
retryStrategy={
|
|
|
|
"attempts": 4,
|
|
|
|
"evaluateOnExit": [
|
|
|
|
{"onStatusReason": "osr", "action": "RETRY"},
|
|
|
|
{"onStatusReason": "osr2", "action": "Exit"},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
resp = batch_client.describe_job_definitions(jobDefinitionName=def_name)
|
2023-06-16 10:42:07 +00:00
|
|
|
assert resp["jobDefinitions"][0]["retryStrategy"] == {
|
|
|
|
"attempts": 4,
|
|
|
|
"evaluateOnExit": [
|
|
|
|
{"onStatusReason": "osr", "action": "retry"},
|
|
|
|
{"onStatusReason": "osr2", "action": "exit"},
|
|
|
|
],
|
|
|
|
}
|
2021-09-21 16:12:18 +00:00
|
|
|
|
|
|
|
|
2021-08-04 12:40:10 +00:00
|
|
|
@mock_batch
|
2021-11-01 10:31:22 +00:00
|
|
|
@pytest.mark.parametrize("use_resource_reqs", [True, False])
|
|
|
|
def test_reregister_task_definition(use_resource_reqs):
|
2021-08-04 12:40:10 +00:00
|
|
|
# Reregistering task with the same name bumps the revision number
|
2022-02-20 20:53:25 +00:00
|
|
|
_, _, _, _, batch_client = _get_clients()
|
2021-08-04 12:40:10 +00:00
|
|
|
|
2021-10-05 17:11:07 +00:00
|
|
|
job_def_name = str(uuid4())[0:6]
|
2021-11-01 10:31:22 +00:00
|
|
|
resp1 = register_job_def(
|
|
|
|
batch_client, definition_name=job_def_name, use_resource_reqs=use_resource_reqs
|
|
|
|
)
|
2021-08-04 12:40:10 +00:00
|
|
|
|
2023-06-16 10:42:07 +00:00
|
|
|
assert "jobDefinitionArn" in resp1
|
|
|
|
assert resp1["jobDefinitionName"] == job_def_name
|
|
|
|
assert "revision" in resp1
|
2021-08-04 12:40:10 +00:00
|
|
|
|
|
|
|
assert resp1["jobDefinitionArn"].endswith(
|
2022-11-17 22:41:08 +00:00
|
|
|
f"{resp1['jobDefinitionName']}:{resp1['revision']}"
|
2021-08-04 12:40:10 +00:00
|
|
|
)
|
2023-06-16 10:42:07 +00:00
|
|
|
assert resp1["revision"] == 1
|
2021-08-04 12:40:10 +00:00
|
|
|
|
2021-11-01 10:31:22 +00:00
|
|
|
resp2 = register_job_def(
|
|
|
|
batch_client, definition_name=job_def_name, use_resource_reqs=use_resource_reqs
|
|
|
|
)
|
2023-06-16 10:42:07 +00:00
|
|
|
assert resp2["revision"] == 2
|
2021-08-04 12:40:10 +00:00
|
|
|
|
2023-06-16 10:42:07 +00:00
|
|
|
assert resp2["jobDefinitionArn"] != resp1["jobDefinitionArn"]
|
2021-08-04 12:40:10 +00:00
|
|
|
|
2021-11-01 10:31:22 +00:00
|
|
|
resp3 = register_job_def(
|
|
|
|
batch_client, definition_name=job_def_name, use_resource_reqs=use_resource_reqs
|
|
|
|
)
|
2023-06-16 10:42:07 +00:00
|
|
|
assert resp3["revision"] == 3
|
2021-08-04 12:40:10 +00:00
|
|
|
|
2023-06-16 10:42:07 +00:00
|
|
|
assert resp3["jobDefinitionArn"] != resp1["jobDefinitionArn"]
|
|
|
|
assert resp3["jobDefinitionArn"] != resp2["jobDefinitionArn"]
|
2021-08-04 12:40:10 +00:00
|
|
|
|
2021-11-01 10:31:22 +00:00
|
|
|
resp4 = register_job_def(
|
|
|
|
batch_client, definition_name=job_def_name, use_resource_reqs=use_resource_reqs
|
|
|
|
)
|
2023-06-16 10:42:07 +00:00
|
|
|
assert resp4["revision"] == 4
|
2021-08-04 12:40:10 +00:00
|
|
|
|
2023-06-16 10:42:07 +00:00
|
|
|
assert resp4["jobDefinitionArn"] != resp1["jobDefinitionArn"]
|
|
|
|
assert resp4["jobDefinitionArn"] != resp2["jobDefinitionArn"]
|
|
|
|
assert resp4["jobDefinitionArn"] != resp3["jobDefinitionArn"]
|
2021-08-04 12:40:10 +00:00
|
|
|
|
|
|
|
|
2022-02-20 20:53:25 +00:00
|
|
|
@mock_batch
|
|
|
|
def test_reregister_task_definition_should_not_reuse_parameters_from_inactive_definition():
|
|
|
|
# Reregistering task with the same name bumps the revision number
|
|
|
|
_, _, _, _, batch_client = _get_clients()
|
|
|
|
|
|
|
|
job_def_name = str(uuid4())[0:6]
|
|
|
|
# Register job definition with parameters
|
|
|
|
resp = batch_client.register_job_definition(
|
|
|
|
jobDefinitionName=job_def_name,
|
|
|
|
type="container",
|
|
|
|
containerProperties={
|
|
|
|
"image": "busybox",
|
|
|
|
"vcpus": 1,
|
|
|
|
"memory": 48,
|
|
|
|
"command": ["sleep", "0"],
|
|
|
|
},
|
|
|
|
parameters={"param1": "val1"},
|
|
|
|
)
|
|
|
|
job_def_arn = resp["jobDefinitionArn"]
|
|
|
|
|
|
|
|
definitions = batch_client.describe_job_definitions(jobDefinitionName=job_def_name)[
|
|
|
|
"jobDefinitions"
|
|
|
|
]
|
2023-06-16 10:42:07 +00:00
|
|
|
assert len(definitions) == 1
|
2022-02-20 20:53:25 +00:00
|
|
|
|
2023-06-16 10:42:07 +00:00
|
|
|
assert definitions[0]["parameters"] == {"param1": "val1"}
|
2022-02-20 20:53:25 +00:00
|
|
|
|
|
|
|
# Deactivate the definition
|
|
|
|
batch_client.deregister_job_definition(jobDefinition=job_def_arn)
|
|
|
|
|
|
|
|
# Second job definition does not provide any parameters
|
|
|
|
batch_client.register_job_definition(
|
|
|
|
jobDefinitionName=job_def_name,
|
|
|
|
type="container",
|
|
|
|
containerProperties={
|
|
|
|
"image": "busybox",
|
|
|
|
"vcpus": 1,
|
|
|
|
"memory": 96,
|
|
|
|
"command": ["sleep", "0"],
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
definitions = batch_client.describe_job_definitions(jobDefinitionName=job_def_name)[
|
|
|
|
"jobDefinitions"
|
|
|
|
]
|
2023-06-16 10:42:07 +00:00
|
|
|
assert len(definitions) == 2
|
2022-02-20 20:53:25 +00:00
|
|
|
|
|
|
|
# Only the inactive definition should have the parameters
|
|
|
|
actual = [(d["revision"], d["status"], d.get("parameters")) for d in definitions]
|
2023-06-16 10:42:07 +00:00
|
|
|
assert (1, "INACTIVE", {"param1": "val1"}) in actual
|
|
|
|
assert (2, "ACTIVE", {}) in actual
|
2022-02-20 20:53:25 +00:00
|
|
|
|
|
|
|
|
2021-08-04 12:40:10 +00:00
|
|
|
@mock_batch
|
2021-11-01 10:31:22 +00:00
|
|
|
@pytest.mark.parametrize("use_resource_reqs", [True, False])
|
|
|
|
def test_delete_task_definition(use_resource_reqs):
|
2022-02-20 20:53:25 +00:00
|
|
|
_, _, _, _, batch_client = _get_clients()
|
2021-08-04 12:40:10 +00:00
|
|
|
|
2021-11-01 10:31:22 +00:00
|
|
|
resp = register_job_def(
|
|
|
|
batch_client, definition_name=str(uuid4()), use_resource_reqs=use_resource_reqs
|
|
|
|
)
|
2021-10-05 17:11:07 +00:00
|
|
|
name = resp["jobDefinitionName"]
|
2021-08-04 12:40:10 +00:00
|
|
|
|
|
|
|
batch_client.deregister_job_definition(jobDefinition=resp["jobDefinitionArn"])
|
|
|
|
|
2021-10-05 17:11:07 +00:00
|
|
|
all_defs = batch_client.describe_job_definitions()["jobDefinitions"]
|
2023-06-16 10:42:07 +00:00
|
|
|
assert name in [jobdef["jobDefinitionName"] for jobdef in all_defs]
|
2022-02-20 20:53:25 +00:00
|
|
|
|
|
|
|
definitions = batch_client.describe_job_definitions(jobDefinitionName=name)[
|
|
|
|
"jobDefinitions"
|
|
|
|
]
|
2023-06-16 10:42:07 +00:00
|
|
|
assert len(definitions) == 1
|
2022-02-20 20:53:25 +00:00
|
|
|
|
2023-06-16 10:42:07 +00:00
|
|
|
assert definitions[0]["revision"] == 1
|
|
|
|
assert definitions[0]["status"] == "INACTIVE"
|
2021-09-27 17:19:44 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_batch
|
2021-11-01 10:31:22 +00:00
|
|
|
@pytest.mark.parametrize("use_resource_reqs", [True, False])
|
|
|
|
def test_delete_task_definition_by_name(use_resource_reqs):
|
2022-02-20 20:53:25 +00:00
|
|
|
_, _, _, _, batch_client = _get_clients()
|
2021-09-27 17:19:44 +00:00
|
|
|
|
2021-11-01 10:31:22 +00:00
|
|
|
resp = register_job_def(
|
|
|
|
batch_client, definition_name=str(uuid4()), use_resource_reqs=use_resource_reqs
|
|
|
|
)
|
2021-10-05 17:11:07 +00:00
|
|
|
name = resp["jobDefinitionName"]
|
2021-09-27 17:19:44 +00:00
|
|
|
|
2021-10-05 17:11:07 +00:00
|
|
|
batch_client.deregister_job_definition(jobDefinition=f"{name}:{resp['revision']}")
|
2021-09-27 17:19:44 +00:00
|
|
|
|
2021-10-05 17:11:07 +00:00
|
|
|
all_defs = batch_client.describe_job_definitions()["jobDefinitions"]
|
2022-02-20 20:53:25 +00:00
|
|
|
# We should still see our job definition as INACTIVE, as it is kept for 180 days
|
2023-06-16 10:42:07 +00:00
|
|
|
assert name in [jobdef["jobDefinitionName"] for jobdef in all_defs]
|
2022-02-20 20:53:25 +00:00
|
|
|
|
|
|
|
# Registering the job definition again should up the revision number
|
|
|
|
register_job_def(
|
|
|
|
batch_client, definition_name=name, use_resource_reqs=use_resource_reqs
|
|
|
|
)
|
|
|
|
|
|
|
|
definitions = batch_client.describe_job_definitions(jobDefinitionName=name)[
|
|
|
|
"jobDefinitions"
|
|
|
|
]
|
2023-06-16 10:42:07 +00:00
|
|
|
assert len(definitions) == 2
|
2022-02-20 20:53:25 +00:00
|
|
|
|
|
|
|
revision_status = [
|
|
|
|
{"revision": d["revision"], "status": d["status"]} for d in definitions
|
|
|
|
]
|
|
|
|
|
2023-06-16 10:42:07 +00:00
|
|
|
assert {"revision": 1, "status": "INACTIVE"} in revision_status
|
|
|
|
assert {"revision": 2, "status": "ACTIVE"} in revision_status
|
2021-08-04 12:40:10 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_batch
|
2021-11-01 10:31:22 +00:00
|
|
|
@pytest.mark.parametrize("use_resource_reqs", [True, False])
|
|
|
|
def test_describe_task_definition(use_resource_reqs):
|
2022-02-20 20:53:25 +00:00
|
|
|
_, _, _, _, batch_client = _get_clients()
|
2021-08-04 12:40:10 +00:00
|
|
|
|
2021-10-05 17:11:07 +00:00
|
|
|
sleep_def_name = f"sleep10_{str(uuid4())[0:6]}"
|
|
|
|
other_name = str(uuid4())[0:6]
|
|
|
|
tagged_name = str(uuid4())[0:6]
|
2021-11-01 10:31:22 +00:00
|
|
|
register_job_def(
|
|
|
|
batch_client,
|
|
|
|
definition_name=sleep_def_name,
|
|
|
|
use_resource_reqs=use_resource_reqs,
|
|
|
|
)
|
|
|
|
register_job_def(
|
|
|
|
batch_client,
|
|
|
|
definition_name=sleep_def_name,
|
|
|
|
use_resource_reqs=use_resource_reqs,
|
|
|
|
)
|
|
|
|
register_job_def(
|
|
|
|
batch_client, definition_name=other_name, use_resource_reqs=use_resource_reqs
|
|
|
|
)
|
2021-10-05 17:11:07 +00:00
|
|
|
register_job_def_with_tags(batch_client, definition_name=tagged_name)
|
2021-08-04 12:40:10 +00:00
|
|
|
|
2021-10-05 17:11:07 +00:00
|
|
|
resp = batch_client.describe_job_definitions(jobDefinitionName=sleep_def_name)
|
2023-06-16 10:42:07 +00:00
|
|
|
assert len(resp["jobDefinitions"]) == 2
|
2021-08-04 12:40:10 +00:00
|
|
|
|
2021-10-05 17:11:07 +00:00
|
|
|
job_defs = batch_client.describe_job_definitions()["jobDefinitions"]
|
|
|
|
all_names = [jd["jobDefinitionName"] for jd in job_defs]
|
2023-06-16 10:42:07 +00:00
|
|
|
assert sleep_def_name in all_names
|
|
|
|
assert other_name in all_names
|
|
|
|
assert tagged_name in all_names
|
2021-08-04 12:40:10 +00:00
|
|
|
|
2021-10-05 17:11:07 +00:00
|
|
|
resp = batch_client.describe_job_definitions(
|
|
|
|
jobDefinitions=[sleep_def_name, other_name]
|
|
|
|
)
|
2023-06-16 10:42:07 +00:00
|
|
|
assert len(resp["jobDefinitions"]) == 3
|
|
|
|
assert resp["jobDefinitions"][0]["tags"] == {}
|
2021-09-21 16:12:18 +00:00
|
|
|
|
2021-10-05 17:11:07 +00:00
|
|
|
resp = batch_client.describe_job_definitions(jobDefinitionName=tagged_name)
|
2023-06-16 10:42:07 +00:00
|
|
|
assert resp["jobDefinitions"][0]["tags"] == {"foo": "123", "bar": "456"}
|
2021-08-04 12:40:10 +00:00
|
|
|
|
|
|
|
for job_definition in resp["jobDefinitions"]:
|
2023-06-16 10:42:07 +00:00
|
|
|
assert job_definition["status"] == "ACTIVE"
|
|
|
|
assert "platformCapabilities" not in job_definition
|
|
|
|
assert "retryStrategy" not in job_definition
|
2021-08-04 12:40:10 +00:00
|
|
|
|
|
|
|
|
2021-11-01 10:31:22 +00:00
|
|
|
def register_job_def(batch_client, definition_name="sleep10", use_resource_reqs=True):
|
2022-03-10 14:39:59 +00:00
|
|
|
container_properties = {"image": "busybox", "command": ["sleep", "10"]}
|
2021-11-01 10:31:22 +00:00
|
|
|
|
|
|
|
if use_resource_reqs:
|
|
|
|
container_properties.update(
|
|
|
|
{
|
|
|
|
"resourceRequirements": [
|
2022-01-29 12:07:10 +00:00
|
|
|
{"value": "0.25", "type": "VCPU"},
|
|
|
|
{"value": "512", "type": "MEMORY"},
|
2021-11-01 10:31:22 +00:00
|
|
|
]
|
|
|
|
}
|
|
|
|
)
|
|
|
|
else:
|
2022-03-10 14:39:59 +00:00
|
|
|
container_properties.update({"memory": 128, "vcpus": 1})
|
2021-11-01 10:31:22 +00:00
|
|
|
|
2021-08-04 12:40:10 +00:00
|
|
|
return batch_client.register_job_definition(
|
|
|
|
jobDefinitionName=definition_name,
|
|
|
|
type="container",
|
2022-01-29 12:07:10 +00:00
|
|
|
containerProperties=container_properties,
|
2021-08-04 12:40:10 +00:00
|
|
|
)
|
2021-09-21 16:12:18 +00:00
|
|
|
|
|
|
|
|
2022-02-20 20:53:25 +00:00
|
|
|
def register_job_def_with_tags(
|
|
|
|
batch_client, definition_name="sleep10", propagate_tags=False
|
|
|
|
):
|
|
|
|
kwargs = {} if propagate_tags is None else {"propagateTags": propagate_tags}
|
2021-09-21 16:12:18 +00:00
|
|
|
return batch_client.register_job_definition(
|
|
|
|
jobDefinitionName=definition_name,
|
|
|
|
type="container",
|
|
|
|
containerProperties={
|
|
|
|
"image": "busybox",
|
|
|
|
"vcpus": 1,
|
|
|
|
"memory": random.randint(4, 128),
|
|
|
|
"command": ["sleep", "10"],
|
|
|
|
},
|
2022-03-10 14:39:59 +00:00
|
|
|
tags={"foo": "123", "bar": "456"},
|
2022-02-20 20:53:25 +00:00
|
|
|
**kwargs,
|
2021-09-21 16:12:18 +00:00
|
|
|
)
|