moto/tests/test_stepfunctions/test_stepfunctions.py
2024-03-16 13:03:41 -01:00

965 lines
34 KiB
Python

import json
import os
import re
from datetime import datetime
from unittest import SkipTest, mock
import boto3
import pytest
from botocore.exceptions import ClientError
from dateutil.tz import tzutc
from moto import mock_aws
from moto.core import DEFAULT_ACCOUNT_ID as ACCOUNT_ID
region = "us-east-1"
simple_definition = (
'{"Comment": "An example of the Amazon States Language using a choice state.",'
'"StartAt": "DefaultState",'
'"States": '
'{"DefaultState": {"Type": "Fail","Error": "DefaultStateError","Cause": "No Matches!"}}}'
)
account_id = None
@mock_aws
def test_state_machine_creation_succeeds():
client = boto3.client("stepfunctions", region_name=region)
name = "example_step_function"
#
response = client.create_state_machine(
name=name, definition=str(simple_definition), roleArn=_get_default_role()
)
#
assert response["ResponseMetadata"]["HTTPStatusCode"] == 200
assert isinstance(response["creationDate"], datetime)
assert response["stateMachineArn"] == (
"arn:aws:states:" + region + ":" + ACCOUNT_ID + ":stateMachine:" + name
)
@mock_aws
def test_state_machine_creation_fails_with_invalid_names():
client = boto3.client("stepfunctions", region_name=region)
invalid_names = [
"with space",
"with<bracket",
"with>bracket",
"with{bracket",
"with}bracket",
"with[bracket",
"with]bracket",
"with?wildcard",
"with*wildcard",
'special"char',
"special#char",
"special%char",
"special\\char",
"special^char",
"special|char",
"special~char",
"special`char",
"special$char",
"special&char",
"special,char",
"special;char",
"special:char",
"special/char",
"uni\u0000code",
"uni\u0001code",
"uni\u0002code",
"uni\u0003code",
"uni\u0004code",
"uni\u0005code",
"uni\u0006code",
"uni\u0007code",
"uni\u0008code",
"uni\u0009code",
"uni\u000Acode",
"uni\u000Bcode",
"uni\u000Ccode",
"uni\u000Dcode",
"uni\u000Ecode",
"uni\u000Fcode",
"uni\u0010code",
"uni\u0011code",
"uni\u0012code",
"uni\u0013code",
"uni\u0014code",
"uni\u0015code",
"uni\u0016code",
"uni\u0017code",
"uni\u0018code",
"uni\u0019code",
"uni\u001Acode",
"uni\u001Bcode",
"uni\u001Ccode",
"uni\u001Dcode",
"uni\u001Ecode",
"uni\u001Fcode",
"uni\u007Fcode",
"uni\u0080code",
"uni\u0081code",
"uni\u0082code",
"uni\u0083code",
"uni\u0084code",
"uni\u0085code",
"uni\u0086code",
"uni\u0087code",
"uni\u0088code",
"uni\u0089code",
"uni\u008Acode",
"uni\u008Bcode",
"uni\u008Ccode",
"uni\u008Dcode",
"uni\u008Ecode",
"uni\u008Fcode",
"uni\u0090code",
"uni\u0091code",
"uni\u0092code",
"uni\u0093code",
"uni\u0094code",
"uni\u0095code",
"uni\u0096code",
"uni\u0097code",
"uni\u0098code",
"uni\u0099code",
"uni\u009Acode",
"uni\u009Bcode",
"uni\u009Ccode",
"uni\u009Dcode",
"uni\u009Ecode",
"uni\u009Fcode",
]
#
for invalid_name in invalid_names:
with pytest.raises(ClientError):
client.create_state_machine(
name=invalid_name,
definition=str(simple_definition),
roleArn=_get_default_role(),
)
@mock_aws
def test_state_machine_creation_requires_valid_role_arn():
client = boto3.client("stepfunctions", region_name=region)
name = "example_step_function"
#
with pytest.raises(ClientError):
client.create_state_machine(
name=name,
definition=str(simple_definition),
roleArn="arn:aws:iam::1234:role/unknown_role",
)
@mock_aws
def test_update_state_machine():
client = boto3.client("stepfunctions", region_name=region)
resp = client.create_state_machine(
name="test", definition=str(simple_definition), roleArn=_get_default_role()
)
state_machine_arn = resp["stateMachineArn"]
updated_role = _get_default_role() + "-updated"
updated_definition = str(simple_definition).replace(
"DefaultState", "DefaultStateUpdated"
)
resp = client.update_state_machine(
stateMachineArn=state_machine_arn,
definition=updated_definition,
roleArn=updated_role,
)
assert resp["ResponseMetadata"]["HTTPStatusCode"] == 200
assert isinstance(resp["updateDate"], datetime)
desc = client.describe_state_machine(stateMachineArn=state_machine_arn)
assert desc["definition"] == updated_definition
assert desc["roleArn"] == updated_role
@mock_aws
def test_state_machine_list_returns_empty_list_by_default():
client = boto3.client("stepfunctions", region_name=region)
#
sm_list = client.list_state_machines()
assert sm_list["stateMachines"] == []
@mock_aws
def test_state_machine_list_returns_created_state_machines():
client = boto3.client("stepfunctions", region_name=region)
#
machine1 = client.create_state_machine(
name="name1",
definition=str(simple_definition),
roleArn=_get_default_role(),
tags=[{"key": "tag_key", "value": "tag_value"}],
)
machine2 = client.create_state_machine(
name="name2", definition=str(simple_definition), roleArn=_get_default_role()
)
sm_list = client.list_state_machines()
#
assert sm_list["ResponseMetadata"]["HTTPStatusCode"] == 200
assert len(sm_list["stateMachines"]) == 2
assert isinstance(sm_list["stateMachines"][0]["creationDate"], datetime)
assert sm_list["stateMachines"][0]["creationDate"] == machine1["creationDate"]
assert sm_list["stateMachines"][0]["name"] == "name1"
assert sm_list["stateMachines"][0]["stateMachineArn"] == machine1["stateMachineArn"]
assert isinstance(sm_list["stateMachines"][1]["creationDate"], datetime)
assert sm_list["stateMachines"][1]["creationDate"] == machine2["creationDate"]
assert sm_list["stateMachines"][1]["name"] == "name2"
assert sm_list["stateMachines"][1]["stateMachineArn"] == machine2["stateMachineArn"]
@mock_aws
def test_state_machine_list_pagination():
client = boto3.client("stepfunctions", region_name=region)
for i in range(25):
machine_name = f"StateMachine-{i}"
client.create_state_machine(
name=machine_name,
definition=str(simple_definition),
roleArn=_get_default_role(),
)
resp = client.list_state_machines()
assert "nextToken" not in resp
assert len(resp["stateMachines"]) == 25
paginator = client.get_paginator("list_state_machines")
page_iterator = paginator.paginate(maxResults=5)
page_list = list(page_iterator)
for page in page_list:
assert len(page["stateMachines"]) == 5
assert "24" in page_list[-1]["stateMachines"][-1]["name"]
@mock_aws
def test_state_machine_creation_is_idempotent_by_name():
client = boto3.client("stepfunctions", region_name=region)
#
client.create_state_machine(
name="name", definition=str(simple_definition), roleArn=_get_default_role()
)
sm_list = client.list_state_machines()
assert len(sm_list["stateMachines"]) == 1
#
client.create_state_machine(
name="name", definition=str(simple_definition), roleArn=_get_default_role()
)
sm_list = client.list_state_machines()
assert len(sm_list["stateMachines"]) == 1
#
client.create_state_machine(
name="diff_name", definition=str(simple_definition), roleArn=_get_default_role()
)
sm_list = client.list_state_machines()
assert len(sm_list["stateMachines"]) == 2
@mock_aws
def test_state_machine_creation_can_be_described():
client = boto3.client("stepfunctions", region_name=region)
#
sm = client.create_state_machine(
name="name", definition=str(simple_definition), roleArn=_get_default_role()
)
desc = client.describe_state_machine(stateMachineArn=sm["stateMachineArn"])
assert desc["ResponseMetadata"]["HTTPStatusCode"] == 200
assert desc["creationDate"] == sm["creationDate"]
assert desc["definition"] == str(simple_definition)
assert desc["name"] == "name"
assert desc["roleArn"] == _get_default_role()
assert desc["stateMachineArn"] == sm["stateMachineArn"]
assert desc["status"] == "ACTIVE"
@mock_aws
def test_state_machine_throws_error_when_describing_unknown_machine():
client = boto3.client("stepfunctions", region_name=region)
#
with pytest.raises(ClientError):
unknown_state_machine = (
f"arn:aws:states:{region}:{ACCOUNT_ID}:stateMachine:unknown"
)
client.describe_state_machine(stateMachineArn=unknown_state_machine)
@mock_aws
def test_state_machine_throws_error_when_describing_bad_arn():
client = boto3.client("stepfunctions", region_name=region)
#
with pytest.raises(ClientError):
client.describe_state_machine(stateMachineArn="bad")
@mock_aws
def test_state_machine_throws_error_when_describing_machine_in_different_account():
client = boto3.client("stepfunctions", region_name=region)
#
with pytest.raises(ClientError):
unknown_state_machine = (
"arn:aws:states:" + region + ":000000000000:stateMachine:unknown"
)
client.describe_state_machine(stateMachineArn=unknown_state_machine)
@mock_aws
def test_state_machine_can_be_deleted():
client = boto3.client("stepfunctions", region_name=region)
sm = client.create_state_machine(
name="name", definition=str(simple_definition), roleArn=_get_default_role()
)
#
response = client.delete_state_machine(stateMachineArn=sm["stateMachineArn"])
assert response["ResponseMetadata"]["HTTPStatusCode"] == 200
#
sm_list = client.list_state_machines()
assert len(sm_list["stateMachines"]) == 0
@mock_aws
def test_state_machine_can_deleted_nonexisting_machine():
client = boto3.client("stepfunctions", region_name=region)
#
unknown_state_machine = (
"arn:aws:states:" + region + ":" + ACCOUNT_ID + ":stateMachine:unknown"
)
response = client.delete_state_machine(stateMachineArn=unknown_state_machine)
assert response["ResponseMetadata"]["HTTPStatusCode"] == 200
#
sm_list = client.list_state_machines()
assert len(sm_list["stateMachines"]) == 0
@mock_aws
def test_state_machine_tagging_non_existent_resource_fails():
client = boto3.client("stepfunctions", region_name=region)
non_existent_arn = f"arn:aws:states:{region}:{ACCOUNT_ID}:stateMachine:non-existent"
with pytest.raises(ClientError) as ex:
client.tag_resource(resourceArn=non_existent_arn, tags=[])
assert ex.value.response["Error"]["Code"] == "ResourceNotFound"
assert non_existent_arn in ex.value.response["Error"]["Message"]
@mock_aws
def test_state_machine_untagging_non_existent_resource_fails():
client = boto3.client("stepfunctions", region_name=region)
non_existent_arn = f"arn:aws:states:{region}:{ACCOUNT_ID}:stateMachine:non-existent"
with pytest.raises(ClientError) as ex:
client.untag_resource(resourceArn=non_existent_arn, tagKeys=[])
assert ex.value.response["Error"]["Code"] == "ResourceNotFound"
assert non_existent_arn in ex.value.response["Error"]["Message"]
@mock_aws
def test_state_machine_tagging():
client = boto3.client("stepfunctions", region_name=region)
tags = [
{"key": "tag_key1", "value": "tag_value1"},
{"key": "tag_key2", "value": "tag_value2"},
]
machine = client.create_state_machine(
name="test", definition=str(simple_definition), roleArn=_get_default_role()
)
client.tag_resource(resourceArn=machine["stateMachineArn"], tags=tags)
resp = client.list_tags_for_resource(resourceArn=machine["stateMachineArn"])
assert resp["tags"] == tags
tags_update = [
{"key": "tag_key1", "value": "tag_value1_new"},
{"key": "tag_key3", "value": "tag_value3"},
]
client.tag_resource(resourceArn=machine["stateMachineArn"], tags=tags_update)
resp = client.list_tags_for_resource(resourceArn=machine["stateMachineArn"])
tags_expected = [
tags_update[0],
tags[1],
tags_update[1],
]
assert resp["tags"] == tags_expected
@mock_aws
def test_state_machine_untagging():
client = boto3.client("stepfunctions", region_name=region)
tags = [
{"key": "tag_key1", "value": "tag_value1"},
{"key": "tag_key2", "value": "tag_value2"},
{"key": "tag_key3", "value": "tag_value3"},
]
machine = client.create_state_machine(
name="test",
definition=str(simple_definition),
roleArn=_get_default_role(),
tags=tags,
)
resp = client.list_tags_for_resource(resourceArn=machine["stateMachineArn"])
assert resp["tags"] == tags
tags_to_delete = ["tag_key1", "tag_key2"]
client.untag_resource(
resourceArn=machine["stateMachineArn"], tagKeys=tags_to_delete
)
resp = client.list_tags_for_resource(resourceArn=machine["stateMachineArn"])
expected_tags = [tag for tag in tags if tag["key"] not in tags_to_delete]
assert resp["tags"] == expected_tags
@mock_aws
def test_state_machine_list_tags_for_created_machine():
client = boto3.client("stepfunctions", region_name=region)
#
machine = client.create_state_machine(
name="name1",
definition=str(simple_definition),
roleArn=_get_default_role(),
tags=[{"key": "tag_key", "value": "tag_value"}],
)
response = client.list_tags_for_resource(resourceArn=machine["stateMachineArn"])
tags = response["tags"]
assert len(tags) == 1
assert tags[0] == {"key": "tag_key", "value": "tag_value"}
@mock_aws
def test_state_machine_list_tags_for_machine_without_tags():
client = boto3.client("stepfunctions", region_name=region)
#
machine = client.create_state_machine(
name="name1", definition=str(simple_definition), roleArn=_get_default_role()
)
response = client.list_tags_for_resource(resourceArn=machine["stateMachineArn"])
tags = response["tags"]
assert len(tags) == 0
@mock_aws
def test_state_machine_list_tags_for_nonexisting_machine():
client = boto3.client("stepfunctions", region_name=region)
#
non_existing_state_machine = (
f"arn:aws:states:{region}:{ACCOUNT_ID}:stateMachine:unknown"
)
response = client.list_tags_for_resource(resourceArn=non_existing_state_machine)
tags = response["tags"]
assert len(tags) == 0
@mock_aws
def test_state_machine_start_execution():
client = boto3.client("stepfunctions", region_name=region)
#
sm = client.create_state_machine(
name="name", definition=str(simple_definition), roleArn=_get_default_role()
)
execution = client.start_execution(stateMachineArn=sm["stateMachineArn"])
#
assert execution["ResponseMetadata"]["HTTPStatusCode"] == 200
uuid_regex = "[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}"
expected_exec_name = (
f"arn:aws:states:{region}:{ACCOUNT_ID}:execution:name:{uuid_regex}"
)
assert re.match(expected_exec_name, execution["executionArn"])
assert isinstance(execution["startDate"], datetime)
@mock_aws
def test_state_machine_start_execution_bad_arn_raises_exception():
client = boto3.client("stepfunctions", region_name=region)
#
with pytest.raises(ClientError):
client.start_execution(stateMachineArn="bad")
@mock_aws
def test_state_machine_start_execution_with_custom_name():
client = boto3.client("stepfunctions", region_name=region)
#
sm = client.create_state_machine(
name="name", definition=str(simple_definition), roleArn=_get_default_role()
)
execution = client.start_execution(
stateMachineArn=sm["stateMachineArn"], name="execution_name"
)
#
assert execution["ResponseMetadata"]["HTTPStatusCode"] == 200
expected_exec_name = (
f"arn:aws:states:{region}:{ACCOUNT_ID}:execution:name:execution_name"
)
assert execution["executionArn"] == expected_exec_name
assert isinstance(execution["startDate"], datetime)
@mock_aws
def test_state_machine_start_execution_fails_on_duplicate_execution_name():
client = boto3.client("stepfunctions", region_name=region)
#
sm = client.create_state_machine(
name="name", definition=str(simple_definition), roleArn=_get_default_role()
)
execution_one = client.start_execution(
stateMachineArn=sm["stateMachineArn"], name="execution_name"
)
#
with pytest.raises(ClientError) as ex:
_ = client.start_execution(
stateMachineArn=sm["stateMachineArn"], name="execution_name"
)
assert ex.value.response["Error"]["Message"] == (
"Execution Already Exists: '" + execution_one["executionArn"] + "'"
)
@mock_aws
def test_state_machine_start_execution_with_custom_input():
client = boto3.client("stepfunctions", region_name=region)
#
sm = client.create_state_machine(
name="name", definition=str(simple_definition), roleArn=_get_default_role()
)
execution_input = json.dumps({"input_key": "input_value"})
execution = client.start_execution(
stateMachineArn=sm["stateMachineArn"], input=execution_input
)
#
assert execution["ResponseMetadata"]["HTTPStatusCode"] == 200
uuid_regex = "[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}"
expected_exec_name = (
f"arn:aws:states:{region}:{ACCOUNT_ID}:execution:name:{uuid_regex}"
)
assert re.match(expected_exec_name, execution["executionArn"])
assert isinstance(execution["startDate"], datetime)
@mock_aws
def test_state_machine_start_execution_with_invalid_input():
client = boto3.client("stepfunctions", region_name=region)
#
sm = client.create_state_machine(
name="name", definition=str(simple_definition), roleArn=_get_default_role()
)
with pytest.raises(ClientError):
_ = client.start_execution(stateMachineArn=sm["stateMachineArn"], input="")
with pytest.raises(ClientError):
_ = client.start_execution(stateMachineArn=sm["stateMachineArn"], input="{")
@mock_aws
def test_state_machine_list_executions():
client = boto3.client("stepfunctions", region_name=region)
#
sm = client.create_state_machine(
name="name", definition=str(simple_definition), roleArn=_get_default_role()
)
execution = client.start_execution(stateMachineArn=sm["stateMachineArn"])
execution_arn = execution["executionArn"]
execution_name = execution_arn[execution_arn.rindex(":") + 1 :]
executions = client.list_executions(stateMachineArn=sm["stateMachineArn"])
#
assert executions["ResponseMetadata"]["HTTPStatusCode"] == 200
assert len(executions["executions"]) == 1
assert executions["executions"][0]["executionArn"] == execution_arn
assert executions["executions"][0]["name"] == execution_name
assert executions["executions"][0]["startDate"] == execution["startDate"]
assert executions["executions"][0]["stateMachineArn"] == sm["stateMachineArn"]
assert executions["executions"][0]["status"] == "RUNNING"
assert "stopDate" not in executions["executions"][0]
@mock_aws
def test_state_machine_list_executions_with_filter():
client = boto3.client("stepfunctions", region_name=region)
sm = client.create_state_machine(
name="name", definition=str(simple_definition), roleArn=_get_default_role()
)
for i in range(20):
execution = client.start_execution(stateMachineArn=sm["stateMachineArn"])
if not i % 4:
client.stop_execution(executionArn=execution["executionArn"])
resp = client.list_executions(stateMachineArn=sm["stateMachineArn"])
assert len(resp["executions"]) == 20
resp = client.list_executions(
stateMachineArn=sm["stateMachineArn"], statusFilter="ABORTED"
)
assert len(resp["executions"]) == 5
assert all(e["status"] == "ABORTED" for e in resp["executions"]) is True
@mock_aws
def test_state_machine_list_executions_with_pagination():
client = boto3.client("stepfunctions", region_name=region)
sm = client.create_state_machine(
name="name", definition=str(simple_definition), roleArn=_get_default_role()
)
for _ in range(100):
client.start_execution(stateMachineArn=sm["stateMachineArn"])
resp = client.list_executions(stateMachineArn=sm["stateMachineArn"])
assert "nextToken" not in resp
assert len(resp["executions"]) == 100
paginator = client.get_paginator("list_executions")
page_iterator = paginator.paginate(
stateMachineArn=sm["stateMachineArn"], maxResults=25
)
for page in page_iterator:
assert len(page["executions"]) == 25
with pytest.raises(ClientError) as ex:
resp = client.list_executions(
stateMachineArn=sm["stateMachineArn"], maxResults=10
)
client.list_executions(
stateMachineArn=sm["stateMachineArn"],
maxResults=10,
statusFilter="ABORTED",
nextToken=resp["nextToken"],
)
assert ex.value.response["Error"]["Code"] == "InvalidToken"
assert "Input inconsistent with page token" in ex.value.response["Error"]["Message"]
with pytest.raises(ClientError) as ex:
client.list_executions(
stateMachineArn=sm["stateMachineArn"], nextToken="invalid"
)
assert ex.value.response["Error"]["Code"] == "InvalidToken"
@mock_aws
def test_state_machine_list_executions_when_none_exist():
client = boto3.client("stepfunctions", region_name=region)
#
sm = client.create_state_machine(
name="name", definition=str(simple_definition), roleArn=_get_default_role()
)
executions = client.list_executions(stateMachineArn=sm["stateMachineArn"])
#
assert executions["ResponseMetadata"]["HTTPStatusCode"] == 200
assert len(executions["executions"]) == 0
@mock_aws
def test_state_machine_describe_execution_with_no_input():
client = boto3.client("stepfunctions", region_name=region)
#
sm = client.create_state_machine(
name="name", definition=str(simple_definition), roleArn=_get_default_role()
)
execution = client.start_execution(stateMachineArn=sm["stateMachineArn"])
description = client.describe_execution(executionArn=execution["executionArn"])
#
assert description["ResponseMetadata"]["HTTPStatusCode"] == 200
assert description["executionArn"] == execution["executionArn"]
assert description["input"] == '"{}"'
assert re.match("[-0-9a-z]+", description["name"])
assert description["startDate"] == execution["startDate"]
assert description["stateMachineArn"] == sm["stateMachineArn"]
assert description["status"] == "RUNNING"
assert "stopDate" not in description
@mock_aws
def test_state_machine_describe_execution_with_custom_input():
client = boto3.client("stepfunctions", region_name=region)
#
execution_input = json.dumps({"input_key": "input_val"})
sm = client.create_state_machine(
name="name", definition=str(simple_definition), roleArn=_get_default_role()
)
execution = client.start_execution(
stateMachineArn=sm["stateMachineArn"], input=execution_input
)
description = client.describe_execution(executionArn=execution["executionArn"])
#
assert description["ResponseMetadata"]["HTTPStatusCode"] == 200
assert description["executionArn"] == execution["executionArn"]
assert json.loads(description["input"]) == execution_input
assert re.match("[-a-z0-9]+", description["name"])
assert description["startDate"] == execution["startDate"]
assert description["stateMachineArn"] == sm["stateMachineArn"]
assert description["status"] == "RUNNING"
assert "stopDate" not in description
@mock_aws
def test_execution_throws_error_when_describing_unknown_execution():
client = boto3.client("stepfunctions", region_name=region)
#
with pytest.raises(ClientError):
unknown_execution = f"arn:aws:states:{region}:{ACCOUNT_ID}:execution:unknown"
client.describe_execution(executionArn=unknown_execution)
@mock_aws
def test_state_machine_can_be_described_by_execution():
client = boto3.client("stepfunctions", region_name=region)
#
sm = client.create_state_machine(
name="name", definition=str(simple_definition), roleArn=_get_default_role()
)
execution = client.start_execution(stateMachineArn=sm["stateMachineArn"])
desc = client.describe_state_machine_for_execution(
executionArn=execution["executionArn"]
)
assert desc["ResponseMetadata"]["HTTPStatusCode"] == 200
assert desc["definition"] == str(simple_definition)
assert desc["name"] == "name"
assert desc["roleArn"] == _get_default_role()
assert desc["stateMachineArn"] == sm["stateMachineArn"]
@mock_aws
def test_state_machine_throws_error_when_describing_unknown_execution():
client = boto3.client("stepfunctions", region_name=region)
#
with pytest.raises(ClientError):
unknown_execution = f"arn:aws:states:{region}:{ACCOUNT_ID}:execution:unknown"
client.describe_state_machine_for_execution(executionArn=unknown_execution)
@mock_aws
def test_state_machine_stop_execution():
client = boto3.client("stepfunctions", region_name=region)
#
sm_arn = client.create_state_machine(
name="name", definition=str(simple_definition), roleArn=_get_default_role()
)["stateMachineArn"]
start = client.start_execution(stateMachineArn=sm_arn)
stop = client.stop_execution(executionArn=start["executionArn"])
#
assert stop["ResponseMetadata"]["HTTPStatusCode"] == 200
assert isinstance(stop["stopDate"], datetime)
description = client.describe_execution(executionArn=start["executionArn"])
assert description["status"] == "ABORTED"
assert isinstance(description["stopDate"], datetime)
execution = client.list_executions(stateMachineArn=sm_arn)["executions"][0]
assert isinstance(execution["stopDate"], datetime)
@mock_aws
def test_state_machine_stop_raises_error_when_unknown_execution():
client = boto3.client("stepfunctions", region_name=region)
client.create_state_machine(
name="test-state-machine",
definition=str(simple_definition),
roleArn=_get_default_role(),
)
with pytest.raises(ClientError) as ex:
unknown_execution = (
f"arn:aws:states:{region}:{ACCOUNT_ID}:execution:test-state-machine:unknown"
)
client.stop_execution(executionArn=unknown_execution)
assert ex.value.response["Error"]["Code"] == "ExecutionDoesNotExist"
assert "Execution Does Not Exist:" in ex.value.response["Error"]["Message"]
@mock_aws
def test_state_machine_get_execution_history_throws_error_with_unknown_execution():
client = boto3.client("stepfunctions", region_name=region)
client.create_state_machine(
name="test-state-machine",
definition=str(simple_definition),
roleArn=_get_default_role(),
)
with pytest.raises(ClientError) as ex:
unknown_execution = (
f"arn:aws:states:{region}:{ACCOUNT_ID}:execution:test-state-machine:unknown"
)
client.get_execution_history(executionArn=unknown_execution)
assert ex.value.response["Error"]["Code"] == "ExecutionDoesNotExist"
assert "Execution Does Not Exist:" in ex.value.response["Error"]["Message"]
@mock_aws
def test_state_machine_get_execution_history_contains_expected_success_events_when_started():
expected_events = [
{
"timestamp": datetime(2020, 1, 1, 0, 0, 0, tzinfo=tzutc()),
"type": "ExecutionStarted",
"id": 1,
"previousEventId": 0,
"executionStartedEventDetails": {
"input": "{}",
"inputDetails": {"truncated": False},
"roleArn": _get_default_role(),
},
},
{
"timestamp": datetime(2020, 1, 1, 0, 0, 10, tzinfo=tzutc()),
"type": "PassStateEntered",
"id": 2,
"previousEventId": 0,
"stateEnteredEventDetails": {
"name": "A State",
"input": "{}",
"inputDetails": {"truncated": False},
},
},
{
"timestamp": datetime(2020, 1, 1, 0, 0, 10, tzinfo=tzutc()),
"type": "PassStateExited",
"id": 3,
"previousEventId": 2,
"stateExitedEventDetails": {
"name": "A State",
"output": "An output",
"outputDetails": {"truncated": False},
},
},
{
"timestamp": datetime(2020, 1, 1, 0, 0, 20, tzinfo=tzutc()),
"type": "ExecutionSucceeded",
"id": 4,
"previousEventId": 3,
"executionSucceededEventDetails": {
"output": "An output",
"outputDetails": {"truncated": False},
},
},
]
client = boto3.client("stepfunctions", region_name=region)
sm = client.create_state_machine(
name="test-state-machine",
definition=simple_definition,
roleArn=_get_default_role(),
)
execution = client.start_execution(stateMachineArn=sm["stateMachineArn"])
execution_history = client.get_execution_history(
executionArn=execution["executionArn"]
)
assert len(execution_history["events"]) == 4
assert execution_history["events"] == expected_events
@mock.patch.dict("os.environ", {"MOTO_ENABLE_ISO_REGIONS": "true"})
@pytest.mark.parametrize(
"test_region", ["us-west-2", "cn-northwest-1", "us-isob-east-1"]
)
@mock_aws
def test_stepfunction_regions(test_region):
client = boto3.client("stepfunctions", region_name=test_region)
resp = client.list_state_machines()
assert resp["ResponseMetadata"]["HTTPStatusCode"] == 200
@mock_aws
@mock.patch.dict(os.environ, {"SF_EXECUTION_HISTORY_TYPE": "FAILURE"})
def test_state_machine_get_execution_history_contains_expected_failure_events_when_started():
if os.environ.get("TEST_SERVER_MODE", "false").lower() == "true":
raise SkipTest("Cant pass environment variable in server mode")
expected_events = [
{
"timestamp": datetime(2020, 1, 1, 0, 0, 0, tzinfo=tzutc()),
"type": "ExecutionStarted",
"id": 1,
"previousEventId": 0,
"executionStartedEventDetails": {
"input": "{}",
"inputDetails": {"truncated": False},
"roleArn": _get_default_role(),
},
},
{
"timestamp": datetime(2020, 1, 1, 0, 0, 10, tzinfo=tzutc()),
"type": "FailStateEntered",
"id": 2,
"previousEventId": 0,
"stateEnteredEventDetails": {
"name": "A State",
"input": "{}",
"inputDetails": {"truncated": False},
},
},
{
"timestamp": datetime(2020, 1, 1, 0, 0, 10, tzinfo=tzutc()),
"type": "ExecutionFailed",
"id": 3,
"previousEventId": 2,
"executionFailedEventDetails": {
"error": "AnError",
"cause": "An error occurred!",
},
},
]
client = boto3.client("stepfunctions", region_name=region)
sm = client.create_state_machine(
name="test-state-machine",
definition=simple_definition,
roleArn=_get_default_role(),
)
execution = client.start_execution(stateMachineArn=sm["stateMachineArn"])
execution_history = client.get_execution_history(
executionArn=execution["executionArn"]
)
assert len(execution_history["events"]) == 3
assert execution_history["events"] == expected_events
exc = client.describe_execution(executionArn=execution["executionArn"])
assert exc["status"] == "FAILED"
exc = client.list_executions(stateMachineArn=sm["stateMachineArn"])["executions"][0]
assert exc["status"] == "FAILED"
@mock_aws
def test_state_machine_name_limits():
# Setup
client = boto3.client("stepfunctions", region_name=region)
long_name = "t" * 81
# Execute
with pytest.raises(ClientError) as exc:
client.create_state_machine(
name=long_name,
definition=simple_definition,
roleArn=_get_default_role(),
)
# Verify
assert exc.value.response["Error"]["Code"] == "ValidationException"
assert exc.value.response["Error"]["Message"] == (
f"1 validation error detected: Value '{long_name}' at 'name' "
"failed to satisfy constraint: "
"Member must have length less than or equal to 80"
)
@mock_aws
def test_state_machine_execution_name_limits():
# Setup
client = boto3.client("stepfunctions", region_name=region)
machine_name = "test_name"
long_name = "t" * 81
resp = client.create_state_machine(
name=machine_name,
definition=simple_definition,
roleArn=_get_default_role(),
)
# Execute
with pytest.raises(ClientError) as exc:
client.start_execution(name=long_name, stateMachineArn=resp["stateMachineArn"])
# Verify
assert exc.value.response["Error"]["Code"] == "ValidationException"
assert exc.value.response["Error"]["Message"] == (
f"1 validation error detected: Value '{long_name}' at 'name' "
"failed to satisfy constraint: "
"Member must have length less than or equal to 80"
)
def _get_default_role():
return "arn:aws:iam::" + ACCOUNT_ID + ":role/unknown_sf_role"