2020-09-04 08:58:16 +00:00
|
|
|
import json
|
2020-12-03 18:32:06 +00:00
|
|
|
import os
|
2023-08-10 18:08:17 +00:00
|
|
|
import re
|
2023-11-30 15:55:51 +00:00
|
|
|
from datetime import datetime
|
2023-08-10 18:08:17 +00:00
|
|
|
from unittest import SkipTest, mock
|
|
|
|
|
|
|
|
import boto3
|
2023-11-30 15:55:51 +00:00
|
|
|
import pytest
|
2019-09-02 15:26:40 +00:00
|
|
|
from botocore.exceptions import ClientError
|
2023-08-10 18:08:17 +00:00
|
|
|
from dateutil.tz import tzutc
|
2019-09-02 15:26:40 +00:00
|
|
|
|
2023-11-30 15:55:51 +00:00
|
|
|
from moto import mock_stepfunctions, mock_sts
|
2022-08-13 09:49:43 +00:00
|
|
|
from moto.core import DEFAULT_ACCOUNT_ID as ACCOUNT_ID
|
2019-09-02 15:26:40 +00:00
|
|
|
|
|
|
|
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!"}}}'
|
2019-10-31 15:44:26 +00:00
|
|
|
)
|
2019-09-07 15:37:55 +00:00
|
|
|
account_id = None
|
2019-09-02 15:26:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
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()
|
2019-09-07 15:37:55 +00:00
|
|
|
)
|
2019-09-02 15:26:40 +00:00
|
|
|
#
|
2023-08-10 18:08:17 +00:00
|
|
|
assert response["ResponseMetadata"]["HTTPStatusCode"] == 200
|
|
|
|
assert isinstance(response["creationDate"], datetime)
|
|
|
|
assert response["stateMachineArn"] == (
|
2019-12-17 02:25:20 +00:00
|
|
|
"arn:aws:states:" + region + ":" + ACCOUNT_ID + ":stateMachine:" + name
|
2019-09-02 15:26:40 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
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:
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError):
|
2019-09-02 15:26:40 +00:00
|
|
|
client.create_state_machine(
|
|
|
|
name=invalid_name,
|
|
|
|
definition=str(simple_definition),
|
2019-09-07 15:37:55 +00:00
|
|
|
roleArn=_get_default_role(),
|
|
|
|
)
|
2019-09-02 15:26:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
def test_state_machine_creation_requires_valid_role_arn():
|
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
|
|
|
name = "example_step_function"
|
|
|
|
#
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError):
|
2019-09-02 15:26:40 +00:00
|
|
|
client.create_state_machine(
|
|
|
|
name=name,
|
|
|
|
definition=str(simple_definition),
|
2019-10-03 12:59:23 +00:00
|
|
|
roleArn="arn:aws:iam::1234:role/unknown_role",
|
|
|
|
)
|
2019-09-02 15:26:40 +00:00
|
|
|
|
|
|
|
|
2020-11-05 10:16:48 +00:00
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
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,
|
|
|
|
)
|
2023-08-10 18:08:17 +00:00
|
|
|
assert resp["ResponseMetadata"]["HTTPStatusCode"] == 200
|
|
|
|
assert isinstance(resp["updateDate"], datetime)
|
2020-11-05 10:16:48 +00:00
|
|
|
|
|
|
|
desc = client.describe_state_machine(stateMachineArn=state_machine_arn)
|
2023-08-10 18:08:17 +00:00
|
|
|
assert desc["definition"] == updated_definition
|
|
|
|
assert desc["roleArn"] == updated_role
|
2020-11-05 10:16:48 +00:00
|
|
|
|
|
|
|
|
2019-09-02 15:26:40 +00:00
|
|
|
@mock_stepfunctions
|
|
|
|
def test_state_machine_list_returns_empty_list_by_default():
|
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
|
|
|
#
|
2021-10-18 19:44:29 +00:00
|
|
|
sm_list = client.list_state_machines()
|
2023-08-10 18:08:17 +00:00
|
|
|
assert sm_list["stateMachines"] == []
|
2019-09-02 15:26:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
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),
|
2019-09-07 15:37:55 +00:00
|
|
|
roleArn=_get_default_role(),
|
2019-09-02 15:26:40 +00:00
|
|
|
tags=[{"key": "tag_key", "value": "tag_value"}],
|
|
|
|
)
|
2020-11-01 10:16:41 +00:00
|
|
|
machine2 = client.create_state_machine(
|
|
|
|
name="name2", definition=str(simple_definition), roleArn=_get_default_role()
|
|
|
|
)
|
2021-10-18 19:44:29 +00:00
|
|
|
sm_list = client.list_state_machines()
|
2019-09-02 15:26:40 +00:00
|
|
|
#
|
2023-08-10 18:08:17 +00:00
|
|
|
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"]
|
2019-09-02 15:26:40 +00:00
|
|
|
|
|
|
|
|
2020-11-01 10:16:41 +00:00
|
|
|
@mock_stepfunctions
|
2021-10-09 20:18:13 +00:00
|
|
|
@mock_sts
|
2020-11-01 10:16:41 +00:00
|
|
|
def test_state_machine_list_pagination():
|
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
|
|
|
for i in range(25):
|
2022-11-17 22:41:08 +00:00
|
|
|
machine_name = f"StateMachine-{i}"
|
2020-11-01 10:16:41 +00:00
|
|
|
client.create_state_machine(
|
|
|
|
name=machine_name,
|
|
|
|
definition=str(simple_definition),
|
|
|
|
roleArn=_get_default_role(),
|
|
|
|
)
|
|
|
|
|
|
|
|
resp = client.list_state_machines()
|
2023-08-10 18:08:17 +00:00
|
|
|
assert "nextToken" not in resp
|
|
|
|
assert len(resp["stateMachines"]) == 25
|
2020-11-01 10:16:41 +00:00
|
|
|
|
|
|
|
paginator = client.get_paginator("list_state_machines")
|
|
|
|
page_iterator = paginator.paginate(maxResults=5)
|
2023-08-10 18:08:17 +00:00
|
|
|
page_list = list(page_iterator)
|
|
|
|
for page in page_list:
|
|
|
|
assert len(page["stateMachines"]) == 5
|
|
|
|
assert "24" in page_list[-1]["stateMachines"][-1]["name"]
|
2020-11-01 10:16:41 +00:00
|
|
|
|
|
|
|
|
2019-09-02 15:26:40 +00:00
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
def test_state_machine_creation_is_idempotent_by_name():
|
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
|
|
|
#
|
2019-09-07 15:37:55 +00:00
|
|
|
client.create_state_machine(
|
|
|
|
name="name", definition=str(simple_definition), roleArn=_get_default_role()
|
2019-10-31 15:44:26 +00:00
|
|
|
)
|
2019-09-02 15:26:40 +00:00
|
|
|
sm_list = client.list_state_machines()
|
2023-08-10 18:08:17 +00:00
|
|
|
assert len(sm_list["stateMachines"]) == 1
|
2019-09-02 15:26:40 +00:00
|
|
|
#
|
2019-09-07 15:37:55 +00:00
|
|
|
client.create_state_machine(
|
|
|
|
name="name", definition=str(simple_definition), roleArn=_get_default_role()
|
2019-10-31 15:44:26 +00:00
|
|
|
)
|
2019-09-02 15:26:40 +00:00
|
|
|
sm_list = client.list_state_machines()
|
2023-08-10 18:08:17 +00:00
|
|
|
assert len(sm_list["stateMachines"]) == 1
|
2019-09-02 15:26:40 +00:00
|
|
|
#
|
2019-09-07 15:37:55 +00:00
|
|
|
client.create_state_machine(
|
|
|
|
name="diff_name", definition=str(simple_definition), roleArn=_get_default_role()
|
2019-10-31 15:44:26 +00:00
|
|
|
)
|
2019-09-02 15:26:40 +00:00
|
|
|
sm_list = client.list_state_machines()
|
2023-08-10 18:08:17 +00:00
|
|
|
assert len(sm_list["stateMachines"]) == 2
|
2019-09-02 15:26:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
2019-09-04 14:42:42 +00:00
|
|
|
def test_state_machine_creation_can_be_described():
|
2019-09-02 15:26:40 +00:00
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
|
|
|
#
|
2019-09-07 15:37:55 +00:00
|
|
|
sm = client.create_state_machine(
|
|
|
|
name="name", definition=str(simple_definition), roleArn=_get_default_role()
|
2019-10-31 15:44:26 +00:00
|
|
|
)
|
2019-09-02 15:26:40 +00:00
|
|
|
desc = client.describe_state_machine(stateMachineArn=sm["stateMachineArn"])
|
2023-08-10 18:08:17 +00:00
|
|
|
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"
|
2019-09-02 15:26:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
def test_state_machine_throws_error_when_describing_unknown_machine():
|
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
|
|
|
#
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError):
|
2019-09-07 15:37:55 +00:00
|
|
|
unknown_state_machine = (
|
2022-08-13 09:49:43 +00:00
|
|
|
f"arn:aws:states:{region}:{ACCOUNT_ID}:stateMachine:unknown"
|
2019-10-31 15:44:26 +00:00
|
|
|
)
|
2019-09-02 15:26:40 +00:00
|
|
|
client.describe_state_machine(stateMachineArn=unknown_state_machine)
|
|
|
|
|
|
|
|
|
2020-05-24 14:06:02 +00:00
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
def test_state_machine_throws_error_when_describing_bad_arn():
|
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
|
|
|
#
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError):
|
2020-05-24 14:06:02 +00:00
|
|
|
client.describe_state_machine(stateMachineArn="bad")
|
|
|
|
|
|
|
|
|
2019-09-02 15:26:40 +00:00
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
def test_state_machine_throws_error_when_describing_machine_in_different_account():
|
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
|
|
|
#
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError):
|
2019-09-02 15:26:40 +00:00
|
|
|
unknown_state_machine = (
|
|
|
|
"arn:aws:states:" + region + ":000000000000:stateMachine:unknown"
|
2019-10-31 15:44:26 +00:00
|
|
|
)
|
2019-09-02 15:26:40 +00:00
|
|
|
client.describe_state_machine(stateMachineArn=unknown_state_machine)
|
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
def test_state_machine_can_be_deleted():
|
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
2019-09-07 15:37:55 +00:00
|
|
|
sm = client.create_state_machine(
|
|
|
|
name="name", definition=str(simple_definition), roleArn=_get_default_role()
|
2019-10-31 15:44:26 +00:00
|
|
|
)
|
2019-09-02 15:26:40 +00:00
|
|
|
#
|
|
|
|
response = client.delete_state_machine(stateMachineArn=sm["stateMachineArn"])
|
2023-08-10 18:08:17 +00:00
|
|
|
assert response["ResponseMetadata"]["HTTPStatusCode"] == 200
|
2019-09-02 15:26:40 +00:00
|
|
|
#
|
|
|
|
sm_list = client.list_state_machines()
|
2023-08-10 18:08:17 +00:00
|
|
|
assert len(sm_list["stateMachines"]) == 0
|
2019-09-02 15:26:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
def test_state_machine_can_deleted_nonexisting_machine():
|
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
|
|
|
#
|
|
|
|
unknown_state_machine = (
|
2019-12-17 02:25:20 +00:00
|
|
|
"arn:aws:states:" + region + ":" + ACCOUNT_ID + ":stateMachine:unknown"
|
2019-10-31 15:44:26 +00:00
|
|
|
)
|
2019-09-02 15:26:40 +00:00
|
|
|
response = client.delete_state_machine(stateMachineArn=unknown_state_machine)
|
2023-08-10 18:08:17 +00:00
|
|
|
assert response["ResponseMetadata"]["HTTPStatusCode"] == 200
|
2019-09-02 15:26:40 +00:00
|
|
|
#
|
|
|
|
sm_list = client.list_state_machines()
|
2023-08-10 18:08:17 +00:00
|
|
|
assert len(sm_list["stateMachines"]) == 0
|
2019-09-02 15:26:40 +00:00
|
|
|
|
|
|
|
|
2020-11-05 10:16:48 +00:00
|
|
|
@mock_stepfunctions
|
|
|
|
def test_state_machine_tagging_non_existent_resource_fails():
|
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
2020-11-11 15:55:37 +00:00
|
|
|
non_existent_arn = f"arn:aws:states:{region}:{ACCOUNT_ID}:stateMachine:non-existent"
|
2020-11-10 17:14:50 +00:00
|
|
|
with pytest.raises(ClientError) as ex:
|
2020-11-05 10:16:48 +00:00
|
|
|
client.tag_resource(resourceArn=non_existent_arn, tags=[])
|
2023-08-10 18:08:17 +00:00
|
|
|
assert ex.value.response["Error"]["Code"] == "ResourceNotFound"
|
|
|
|
assert non_existent_arn in ex.value.response["Error"]["Message"]
|
2020-11-05 10:16:48 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
def test_state_machine_untagging_non_existent_resource_fails():
|
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
2020-11-11 15:55:37 +00:00
|
|
|
non_existent_arn = f"arn:aws:states:{region}:{ACCOUNT_ID}:stateMachine:non-existent"
|
2020-11-10 17:14:50 +00:00
|
|
|
with pytest.raises(ClientError) as ex:
|
2020-11-05 10:16:48 +00:00
|
|
|
client.untag_resource(resourceArn=non_existent_arn, tagKeys=[])
|
2023-08-10 18:08:17 +00:00
|
|
|
assert ex.value.response["Error"]["Code"] == "ResourceNotFound"
|
|
|
|
assert non_existent_arn in ex.value.response["Error"]["Message"]
|
2020-11-05 10:16:48 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
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(
|
2020-11-11 15:55:37 +00:00
|
|
|
name="test", definition=str(simple_definition), roleArn=_get_default_role()
|
2020-11-05 10:16:48 +00:00
|
|
|
)
|
|
|
|
client.tag_resource(resourceArn=machine["stateMachineArn"], tags=tags)
|
|
|
|
resp = client.list_tags_for_resource(resourceArn=machine["stateMachineArn"])
|
2023-08-10 18:08:17 +00:00
|
|
|
assert resp["tags"] == tags
|
2020-11-05 10:16:48 +00:00
|
|
|
|
|
|
|
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],
|
|
|
|
]
|
2023-08-10 18:08:17 +00:00
|
|
|
assert resp["tags"] == tags_expected
|
2020-11-05 10:16:48 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
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"])
|
2023-08-10 18:08:17 +00:00
|
|
|
assert resp["tags"] == tags
|
2020-11-05 10:16:48 +00:00
|
|
|
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]
|
2023-08-10 18:08:17 +00:00
|
|
|
assert resp["tags"] == expected_tags
|
2020-11-05 10:16:48 +00:00
|
|
|
|
|
|
|
|
2019-09-02 15:26:40 +00:00
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
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),
|
2019-09-07 15:37:55 +00:00
|
|
|
roleArn=_get_default_role(),
|
2019-09-02 15:26:40 +00:00
|
|
|
tags=[{"key": "tag_key", "value": "tag_value"}],
|
|
|
|
)
|
|
|
|
response = client.list_tags_for_resource(resourceArn=machine["stateMachineArn"])
|
|
|
|
tags = response["tags"]
|
2023-08-10 18:08:17 +00:00
|
|
|
assert len(tags) == 1
|
|
|
|
assert tags[0] == {"key": "tag_key", "value": "tag_value"}
|
2019-09-02 15:26:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
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()
|
2019-09-07 15:37:55 +00:00
|
|
|
)
|
2019-09-02 15:26:40 +00:00
|
|
|
response = client.list_tags_for_resource(resourceArn=machine["stateMachineArn"])
|
|
|
|
tags = response["tags"]
|
2023-08-10 18:08:17 +00:00
|
|
|
assert len(tags) == 0
|
2019-09-02 15:26:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
def test_state_machine_list_tags_for_nonexisting_machine():
|
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
|
|
|
#
|
2019-09-07 15:37:55 +00:00
|
|
|
non_existing_state_machine = (
|
2022-08-13 09:49:43 +00:00
|
|
|
f"arn:aws:states:{region}:{ACCOUNT_ID}:stateMachine:unknown"
|
2019-10-31 15:44:26 +00:00
|
|
|
)
|
2019-09-02 15:26:40 +00:00
|
|
|
response = client.list_tags_for_resource(resourceArn=non_existing_state_machine)
|
|
|
|
tags = response["tags"]
|
2023-08-10 18:08:17 +00:00
|
|
|
assert len(tags) == 0
|
2019-09-04 14:42:42 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
def test_state_machine_start_execution():
|
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
|
|
|
#
|
2019-09-07 15:37:55 +00:00
|
|
|
sm = client.create_state_machine(
|
|
|
|
name="name", definition=str(simple_definition), roleArn=_get_default_role()
|
2019-10-31 15:44:26 +00:00
|
|
|
)
|
2019-09-04 14:42:42 +00:00
|
|
|
execution = client.start_execution(stateMachineArn=sm["stateMachineArn"])
|
|
|
|
#
|
2023-08-10 18:08:17 +00:00
|
|
|
assert execution["ResponseMetadata"]["HTTPStatusCode"] == 200
|
2019-10-03 13:47:24 +00:00
|
|
|
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 = (
|
2022-08-13 09:49:43 +00:00
|
|
|
f"arn:aws:states:{region}:{ACCOUNT_ID}:execution:name:{uuid_regex}"
|
2019-10-31 15:44:26 +00:00
|
|
|
)
|
2023-08-10 18:08:17 +00:00
|
|
|
assert re.match(expected_exec_name, execution["executionArn"])
|
|
|
|
assert isinstance(execution["startDate"], datetime)
|
2019-09-04 14:42:42 +00:00
|
|
|
|
|
|
|
|
2020-05-24 14:06:02 +00:00
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
def test_state_machine_start_execution_bad_arn_raises_exception():
|
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
|
|
|
#
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError):
|
2020-05-24 14:06:02 +00:00
|
|
|
client.start_execution(stateMachineArn="bad")
|
|
|
|
|
|
|
|
|
2019-10-03 13:47:24 +00:00
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
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"
|
2019-10-31 15:44:26 +00:00
|
|
|
)
|
2019-10-03 13:47:24 +00:00
|
|
|
#
|
2023-08-10 18:08:17 +00:00
|
|
|
assert execution["ResponseMetadata"]["HTTPStatusCode"] == 200
|
2019-10-03 13:47:24 +00:00
|
|
|
expected_exec_name = (
|
2022-08-13 09:49:43 +00:00
|
|
|
f"arn:aws:states:{region}:{ACCOUNT_ID}:execution:name:execution_name"
|
2019-10-31 15:44:26 +00:00
|
|
|
)
|
2023-08-10 18:08:17 +00:00
|
|
|
assert execution["executionArn"] == expected_exec_name
|
|
|
|
assert isinstance(execution["startDate"], datetime)
|
2019-10-03 13:47:24 +00:00
|
|
|
|
|
|
|
|
2020-08-27 07:22:44 +00:00
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
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"
|
|
|
|
)
|
|
|
|
#
|
2020-10-06 06:04:09 +00:00
|
|
|
with pytest.raises(ClientError) as ex:
|
2020-08-27 07:22:44 +00:00
|
|
|
_ = client.start_execution(
|
|
|
|
stateMachineArn=sm["stateMachineArn"], name="execution_name"
|
|
|
|
)
|
2023-08-10 18:08:17 +00:00
|
|
|
assert ex.value.response["Error"]["Message"] == (
|
2020-08-27 07:22:44 +00:00
|
|
|
"Execution Already Exists: '" + execution_one["executionArn"] + "'"
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2020-09-04 09:22:21 +00:00
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
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
|
|
|
|
)
|
|
|
|
#
|
2023-08-10 18:08:17 +00:00
|
|
|
assert execution["ResponseMetadata"]["HTTPStatusCode"] == 200
|
2020-09-04 09:22:21 +00:00
|
|
|
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 = (
|
2022-08-13 09:49:43 +00:00
|
|
|
f"arn:aws:states:{region}:{ACCOUNT_ID}:execution:name:{uuid_regex}"
|
2020-09-04 09:22:21 +00:00
|
|
|
)
|
2023-08-10 18:08:17 +00:00
|
|
|
assert re.match(expected_exec_name, execution["executionArn"])
|
|
|
|
assert isinstance(execution["startDate"], datetime)
|
2020-09-04 09:22:21 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
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()
|
|
|
|
)
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError):
|
2020-09-04 09:22:21 +00:00
|
|
|
_ = client.start_execution(stateMachineArn=sm["stateMachineArn"], input="")
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError):
|
2020-09-04 09:22:21 +00:00
|
|
|
_ = client.start_execution(stateMachineArn=sm["stateMachineArn"], input="{")
|
|
|
|
|
|
|
|
|
2019-09-04 14:42:42 +00:00
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
def test_state_machine_list_executions():
|
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
|
|
|
#
|
2019-09-07 15:37:55 +00:00
|
|
|
sm = client.create_state_machine(
|
|
|
|
name="name", definition=str(simple_definition), roleArn=_get_default_role()
|
2019-10-31 15:44:26 +00:00
|
|
|
)
|
2019-09-04 14:42:42 +00:00
|
|
|
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"])
|
|
|
|
#
|
2023-08-10 18:08:17 +00:00
|
|
|
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]
|
2019-09-04 14:42:42 +00:00
|
|
|
|
|
|
|
|
2020-11-01 10:16:41 +00:00
|
|
|
@mock_stepfunctions
|
|
|
|
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"])
|
2023-08-10 18:08:17 +00:00
|
|
|
assert len(resp["executions"]) == 20
|
2020-11-01 10:16:41 +00:00
|
|
|
|
|
|
|
resp = client.list_executions(
|
|
|
|
stateMachineArn=sm["stateMachineArn"], statusFilter="ABORTED"
|
|
|
|
)
|
2023-08-10 18:08:17 +00:00
|
|
|
assert len(resp["executions"]) == 5
|
|
|
|
assert all(e["status"] == "ABORTED" for e in resp["executions"]) is True
|
2020-11-01 10:16:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
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"])
|
2023-08-10 18:08:17 +00:00
|
|
|
assert "nextToken" not in resp
|
|
|
|
assert len(resp["executions"]) == 100
|
2020-11-01 10:16:41 +00:00
|
|
|
|
|
|
|
paginator = client.get_paginator("list_executions")
|
|
|
|
page_iterator = paginator.paginate(
|
|
|
|
stateMachineArn=sm["stateMachineArn"], maxResults=25
|
|
|
|
)
|
|
|
|
for page in page_iterator:
|
2023-08-10 18:08:17 +00:00
|
|
|
assert len(page["executions"]) == 25
|
2020-11-01 10:16:41 +00:00
|
|
|
|
2020-11-10 17:14:50 +00:00
|
|
|
with pytest.raises(ClientError) as ex:
|
2020-11-01 10:16:41 +00:00
|
|
|
resp = client.list_executions(
|
|
|
|
stateMachineArn=sm["stateMachineArn"], maxResults=10
|
|
|
|
)
|
|
|
|
client.list_executions(
|
|
|
|
stateMachineArn=sm["stateMachineArn"],
|
|
|
|
maxResults=10,
|
|
|
|
statusFilter="ABORTED",
|
|
|
|
nextToken=resp["nextToken"],
|
|
|
|
)
|
2023-08-10 18:08:17 +00:00
|
|
|
assert ex.value.response["Error"]["Code"] == "InvalidToken"
|
|
|
|
assert "Input inconsistent with page token" in ex.value.response["Error"]["Message"]
|
2020-11-01 10:16:41 +00:00
|
|
|
|
2020-11-10 17:14:50 +00:00
|
|
|
with pytest.raises(ClientError) as ex:
|
2020-11-01 10:16:41 +00:00
|
|
|
client.list_executions(
|
|
|
|
stateMachineArn=sm["stateMachineArn"], nextToken="invalid"
|
|
|
|
)
|
2023-08-10 18:08:17 +00:00
|
|
|
assert ex.value.response["Error"]["Code"] == "InvalidToken"
|
2020-11-01 10:16:41 +00:00
|
|
|
|
|
|
|
|
2019-09-04 14:42:42 +00:00
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
def test_state_machine_list_executions_when_none_exist():
|
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
|
|
|
#
|
2019-09-07 15:37:55 +00:00
|
|
|
sm = client.create_state_machine(
|
|
|
|
name="name", definition=str(simple_definition), roleArn=_get_default_role()
|
2019-10-31 15:44:26 +00:00
|
|
|
)
|
2019-09-04 14:42:42 +00:00
|
|
|
executions = client.list_executions(stateMachineArn=sm["stateMachineArn"])
|
|
|
|
#
|
2023-08-10 18:08:17 +00:00
|
|
|
assert executions["ResponseMetadata"]["HTTPStatusCode"] == 200
|
|
|
|
assert len(executions["executions"]) == 0
|
2019-09-04 14:42:42 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
2020-09-04 08:58:16 +00:00
|
|
|
def test_state_machine_describe_execution_with_no_input():
|
2019-09-04 14:42:42 +00:00
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
|
|
|
#
|
2019-09-07 15:37:55 +00:00
|
|
|
sm = client.create_state_machine(
|
|
|
|
name="name", definition=str(simple_definition), roleArn=_get_default_role()
|
2019-10-31 15:44:26 +00:00
|
|
|
)
|
2019-09-04 14:42:42 +00:00
|
|
|
execution = client.start_execution(stateMachineArn=sm["stateMachineArn"])
|
|
|
|
description = client.describe_execution(executionArn=execution["executionArn"])
|
|
|
|
#
|
2023-08-10 18:08:17 +00:00
|
|
|
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
|
2019-09-04 14:42:42 +00:00
|
|
|
|
|
|
|
|
2020-09-04 08:58:16 +00:00
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
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"])
|
|
|
|
#
|
2023-08-10 18:08:17 +00:00
|
|
|
assert description["ResponseMetadata"]["HTTPStatusCode"] == 200
|
|
|
|
assert description["executionArn"] == execution["executionArn"]
|
|
|
|
assert 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
|
2020-09-04 08:58:16 +00:00
|
|
|
|
|
|
|
|
2019-09-04 14:42:42 +00:00
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
2020-05-24 14:06:02 +00:00
|
|
|
def test_execution_throws_error_when_describing_unknown_execution():
|
2019-09-04 14:42:42 +00:00
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
|
|
|
#
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError):
|
2022-08-13 09:49:43 +00:00
|
|
|
unknown_execution = f"arn:aws:states:{region}:{ACCOUNT_ID}:execution:unknown"
|
2019-09-04 14:42:42 +00:00
|
|
|
client.describe_execution(executionArn=unknown_execution)
|
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
def test_state_machine_can_be_described_by_execution():
|
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
|
|
|
#
|
2019-09-07 15:37:55 +00:00
|
|
|
sm = client.create_state_machine(
|
|
|
|
name="name", definition=str(simple_definition), roleArn=_get_default_role()
|
2019-10-31 15:44:26 +00:00
|
|
|
)
|
2019-09-04 14:42:42 +00:00
|
|
|
execution = client.start_execution(stateMachineArn=sm["stateMachineArn"])
|
|
|
|
desc = client.describe_state_machine_for_execution(
|
|
|
|
executionArn=execution["executionArn"]
|
|
|
|
)
|
2023-08-10 18:08:17 +00:00
|
|
|
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"]
|
2019-09-04 14:42:42 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
def test_state_machine_throws_error_when_describing_unknown_execution():
|
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
|
|
|
#
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError):
|
2022-08-13 09:49:43 +00:00
|
|
|
unknown_execution = f"arn:aws:states:{region}:{ACCOUNT_ID}:execution:unknown"
|
2019-09-04 14:42:42 +00:00
|
|
|
client.describe_state_machine_for_execution(executionArn=unknown_execution)
|
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
def test_state_machine_stop_execution():
|
|
|
|
client = boto3.client("stepfunctions", region_name=region)
|
|
|
|
#
|
2023-09-10 18:14:41 +00:00
|
|
|
sm_arn = client.create_state_machine(
|
2019-09-07 15:37:55 +00:00
|
|
|
name="name", definition=str(simple_definition), roleArn=_get_default_role()
|
2023-09-10 18:14:41 +00:00
|
|
|
)["stateMachineArn"]
|
|
|
|
start = client.start_execution(stateMachineArn=sm_arn)
|
2019-09-04 14:42:42 +00:00
|
|
|
stop = client.stop_execution(executionArn=start["executionArn"])
|
|
|
|
#
|
2023-08-10 18:08:17 +00:00
|
|
|
assert stop["ResponseMetadata"]["HTTPStatusCode"] == 200
|
|
|
|
assert isinstance(stop["stopDate"], datetime)
|
2019-09-04 14:42:42 +00:00
|
|
|
|
2023-09-10 18:14:41 +00:00
|
|
|
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)
|
|
|
|
|
2019-09-04 14:42:42 +00:00
|
|
|
|
2020-12-03 18:32:06 +00:00
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
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 = (
|
2022-08-13 09:49:43 +00:00
|
|
|
f"arn:aws:states:{region}:{ACCOUNT_ID}:execution:test-state-machine:unknown"
|
2020-12-03 18:32:06 +00:00
|
|
|
)
|
|
|
|
client.stop_execution(executionArn=unknown_execution)
|
2023-08-10 18:08:17 +00:00
|
|
|
assert ex.value.response["Error"]["Code"] == "ExecutionDoesNotExist"
|
|
|
|
assert "Execution Does Not Exist:" in ex.value.response["Error"]["Message"]
|
2020-12-03 18:32:06 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
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 = (
|
2022-08-13 09:49:43 +00:00
|
|
|
f"arn:aws:states:{region}:{ACCOUNT_ID}:execution:test-state-machine:unknown"
|
2020-12-03 18:32:06 +00:00
|
|
|
)
|
|
|
|
client.get_execution_history(executionArn=unknown_execution)
|
2023-08-10 18:08:17 +00:00
|
|
|
assert ex.value.response["Error"]["Code"] == "ExecutionDoesNotExist"
|
|
|
|
assert "Execution Does Not Exist:" in ex.value.response["Error"]["Message"]
|
2020-12-03 18:32:06 +00:00
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
def test_state_machine_get_execution_history_contains_expected_success_events_when_started():
|
|
|
|
expected_events = [
|
|
|
|
{
|
test_stepfunctions: Fix failures under non-UTC timezone (#3528)
These tests were passing with TZ=UTC, but under a non-UTC timezone
they were failing:
E AssertionError: given
E X = [{'timestamp': datetime.datetime(2020, 1, 1, 0, 0, tzinfo=tzutc()), …}, …]
E and
E Y = [{'timestamp': datetime.datetime(2020, 1, 1, 0, 0, tzinfo=tzlocal()), …}, …]
E X[0]['timestamp'] != Y[0]['timestamp']
With this fix, they pass either way.
Signed-off-by: Anders Kaseorg <andersk@mit.edu>
2020-12-09 12:25:11 +00:00
|
|
|
"timestamp": datetime(2020, 1, 1, 0, 0, 0, tzinfo=tzutc()),
|
2020-12-03 18:32:06 +00:00
|
|
|
"type": "ExecutionStarted",
|
|
|
|
"id": 1,
|
|
|
|
"previousEventId": 0,
|
|
|
|
"executionStartedEventDetails": {
|
|
|
|
"input": "{}",
|
|
|
|
"inputDetails": {"truncated": False},
|
|
|
|
"roleArn": _get_default_role(),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
test_stepfunctions: Fix failures under non-UTC timezone (#3528)
These tests were passing with TZ=UTC, but under a non-UTC timezone
they were failing:
E AssertionError: given
E X = [{'timestamp': datetime.datetime(2020, 1, 1, 0, 0, tzinfo=tzutc()), …}, …]
E and
E Y = [{'timestamp': datetime.datetime(2020, 1, 1, 0, 0, tzinfo=tzlocal()), …}, …]
E X[0]['timestamp'] != Y[0]['timestamp']
With this fix, they pass either way.
Signed-off-by: Anders Kaseorg <andersk@mit.edu>
2020-12-09 12:25:11 +00:00
|
|
|
"timestamp": datetime(2020, 1, 1, 0, 0, 10, tzinfo=tzutc()),
|
2020-12-03 18:32:06 +00:00
|
|
|
"type": "PassStateEntered",
|
|
|
|
"id": 2,
|
|
|
|
"previousEventId": 0,
|
|
|
|
"stateEnteredEventDetails": {
|
|
|
|
"name": "A State",
|
|
|
|
"input": "{}",
|
|
|
|
"inputDetails": {"truncated": False},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
test_stepfunctions: Fix failures under non-UTC timezone (#3528)
These tests were passing with TZ=UTC, but under a non-UTC timezone
they were failing:
E AssertionError: given
E X = [{'timestamp': datetime.datetime(2020, 1, 1, 0, 0, tzinfo=tzutc()), …}, …]
E and
E Y = [{'timestamp': datetime.datetime(2020, 1, 1, 0, 0, tzinfo=tzlocal()), …}, …]
E X[0]['timestamp'] != Y[0]['timestamp']
With this fix, they pass either way.
Signed-off-by: Anders Kaseorg <andersk@mit.edu>
2020-12-09 12:25:11 +00:00
|
|
|
"timestamp": datetime(2020, 1, 1, 0, 0, 10, tzinfo=tzutc()),
|
2020-12-03 18:32:06 +00:00
|
|
|
"type": "PassStateExited",
|
|
|
|
"id": 3,
|
|
|
|
"previousEventId": 2,
|
|
|
|
"stateExitedEventDetails": {
|
|
|
|
"name": "A State",
|
|
|
|
"output": "An output",
|
|
|
|
"outputDetails": {"truncated": False},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
test_stepfunctions: Fix failures under non-UTC timezone (#3528)
These tests were passing with TZ=UTC, but under a non-UTC timezone
they were failing:
E AssertionError: given
E X = [{'timestamp': datetime.datetime(2020, 1, 1, 0, 0, tzinfo=tzutc()), …}, …]
E and
E Y = [{'timestamp': datetime.datetime(2020, 1, 1, 0, 0, tzinfo=tzlocal()), …}, …]
E X[0]['timestamp'] != Y[0]['timestamp']
With this fix, they pass either way.
Signed-off-by: Anders Kaseorg <andersk@mit.edu>
2020-12-09 12:25:11 +00:00
|
|
|
"timestamp": datetime(2020, 1, 1, 0, 0, 20, tzinfo=tzutc()),
|
2020-12-03 18:32:06 +00:00
|
|
|
"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"]
|
|
|
|
)
|
2023-08-10 18:08:17 +00:00
|
|
|
assert len(execution_history["events"]) == 4
|
|
|
|
assert execution_history["events"] == expected_events
|
2020-12-03 18:32:06 +00:00
|
|
|
|
|
|
|
|
2023-06-28 21:23:33 +00:00
|
|
|
@mock.patch.dict("os.environ", {"MOTO_ENABLE_ISO_REGIONS": "true"})
|
2023-06-18 12:05:55 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"test_region", ["us-west-2", "cn-northwest-1", "us-isob-east-1"]
|
|
|
|
)
|
2021-10-18 16:13:08 +00:00
|
|
|
@mock_stepfunctions
|
2022-10-04 16:28:30 +00:00
|
|
|
def test_stepfunction_regions(test_region):
|
|
|
|
client = boto3.client("stepfunctions", region_name=test_region)
|
2021-10-18 16:13:08 +00:00
|
|
|
resp = client.list_state_machines()
|
2023-08-10 18:08:17 +00:00
|
|
|
assert resp["ResponseMetadata"]["HTTPStatusCode"] == 200
|
2021-10-18 16:13:08 +00:00
|
|
|
|
|
|
|
|
2020-12-03 18:32:06 +00:00
|
|
|
@mock_stepfunctions
|
|
|
|
@mock_sts
|
|
|
|
@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 = [
|
|
|
|
{
|
test_stepfunctions: Fix failures under non-UTC timezone (#3528)
These tests were passing with TZ=UTC, but under a non-UTC timezone
they were failing:
E AssertionError: given
E X = [{'timestamp': datetime.datetime(2020, 1, 1, 0, 0, tzinfo=tzutc()), …}, …]
E and
E Y = [{'timestamp': datetime.datetime(2020, 1, 1, 0, 0, tzinfo=tzlocal()), …}, …]
E X[0]['timestamp'] != Y[0]['timestamp']
With this fix, they pass either way.
Signed-off-by: Anders Kaseorg <andersk@mit.edu>
2020-12-09 12:25:11 +00:00
|
|
|
"timestamp": datetime(2020, 1, 1, 0, 0, 0, tzinfo=tzutc()),
|
2020-12-03 18:32:06 +00:00
|
|
|
"type": "ExecutionStarted",
|
|
|
|
"id": 1,
|
|
|
|
"previousEventId": 0,
|
|
|
|
"executionStartedEventDetails": {
|
|
|
|
"input": "{}",
|
|
|
|
"inputDetails": {"truncated": False},
|
|
|
|
"roleArn": _get_default_role(),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
test_stepfunctions: Fix failures under non-UTC timezone (#3528)
These tests were passing with TZ=UTC, but under a non-UTC timezone
they were failing:
E AssertionError: given
E X = [{'timestamp': datetime.datetime(2020, 1, 1, 0, 0, tzinfo=tzutc()), …}, …]
E and
E Y = [{'timestamp': datetime.datetime(2020, 1, 1, 0, 0, tzinfo=tzlocal()), …}, …]
E X[0]['timestamp'] != Y[0]['timestamp']
With this fix, they pass either way.
Signed-off-by: Anders Kaseorg <andersk@mit.edu>
2020-12-09 12:25:11 +00:00
|
|
|
"timestamp": datetime(2020, 1, 1, 0, 0, 10, tzinfo=tzutc()),
|
2020-12-03 18:32:06 +00:00
|
|
|
"type": "FailStateEntered",
|
|
|
|
"id": 2,
|
|
|
|
"previousEventId": 0,
|
|
|
|
"stateEnteredEventDetails": {
|
|
|
|
"name": "A State",
|
|
|
|
"input": "{}",
|
|
|
|
"inputDetails": {"truncated": False},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
test_stepfunctions: Fix failures under non-UTC timezone (#3528)
These tests were passing with TZ=UTC, but under a non-UTC timezone
they were failing:
E AssertionError: given
E X = [{'timestamp': datetime.datetime(2020, 1, 1, 0, 0, tzinfo=tzutc()), …}, …]
E and
E Y = [{'timestamp': datetime.datetime(2020, 1, 1, 0, 0, tzinfo=tzlocal()), …}, …]
E X[0]['timestamp'] != Y[0]['timestamp']
With this fix, they pass either way.
Signed-off-by: Anders Kaseorg <andersk@mit.edu>
2020-12-09 12:25:11 +00:00
|
|
|
"timestamp": datetime(2020, 1, 1, 0, 0, 10, tzinfo=tzutc()),
|
2020-12-03 18:32:06 +00:00
|
|
|
"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"]
|
|
|
|
)
|
2023-08-10 18:08:17 +00:00
|
|
|
assert len(execution_history["events"]) == 3
|
|
|
|
assert execution_history["events"] == expected_events
|
2020-12-03 18:32:06 +00:00
|
|
|
|
2023-04-12 09:26:28 +00:00
|
|
|
exc = client.describe_execution(executionArn=execution["executionArn"])
|
|
|
|
assert exc["status"] == "FAILED"
|
|
|
|
|
|
|
|
exc = client.list_executions(stateMachineArn=sm["stateMachineArn"])["executions"][0]
|
|
|
|
assert exc["status"] == "FAILED"
|
|
|
|
|
2020-12-03 18:32:06 +00:00
|
|
|
|
2023-06-27 11:26:35 +00:00
|
|
|
@mock_stepfunctions
|
|
|
|
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"
|
2023-08-10 18:08:17 +00:00
|
|
|
assert exc.value.response["Error"]["Message"] == (
|
|
|
|
f"1 validation error detected: Value '{long_name}' at 'name' "
|
|
|
|
"failed to satisfy constraint: "
|
2023-06-27 11:26:35 +00:00
|
|
|
"Member must have length less than or equal to 80"
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@mock_stepfunctions
|
|
|
|
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"
|
2023-08-10 18:08:17 +00:00
|
|
|
assert exc.value.response["Error"]["Message"] == (
|
|
|
|
f"1 validation error detected: Value '{long_name}' at 'name' "
|
|
|
|
"failed to satisfy constraint: "
|
2023-06-27 11:26:35 +00:00
|
|
|
"Member must have length less than or equal to 80"
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2019-09-07 15:37:55 +00:00
|
|
|
def _get_default_role():
|
2022-08-13 09:49:43 +00:00
|
|
|
return "arn:aws:iam::" + ACCOUNT_ID + ":role/unknown_sf_role"
|