Step Functions Local testing with mocks provides the capability to mock AWS service integrations that are present in a state machine. This helps in testing the state machine in isolation.
This is a sample project which showcases how to run Step Functions local tests with mocks using pytest instead of running ad-hoc CLI commands. Testcontainers is used to run the Step Functions Local Docker image.
This is an implementation of the test strategy shown in the blog post above using pytest within Python to provide automated testing with strong assertion capabilities.
The SUT is a sales lead generation sample workflow implemented with AWS Step Functions. In this example, new sales leads are created in a customer relationship management system. This triggers the sample workflow execution using input data, which provides information about the contact.
Using the sales lead data, the workflow first validates the contact’s identity and address. If valid, it uses Step Functions’ AWS SDK integration for Amazon Comprehend to call the DetectSentiment API. It uses the sales lead’s comments as input for sentiment analysis.
If the comments have a positive sentiment, it adds the sales leads information to a DynamoDB table for follow-up. The event is published to Amazon EventBridge to notify subscribers.
If the sales lead data is invalid or a negative sentiment is detected, it publishes events to EventBridge for notification. No record is added to the Amazon DynamoDB table. The following Step Functions Workflow Studio diagram shows the control logic:
The goal of this example is to test AWS Step Functions state machines in isolation using mocked responses for the external services which need to conform to actual responses before testing. With mocking, developers get more control over the type of scenarios that a state machine can handle, leading to assertion of multiple behaviors. Testing a state machine with mocks can also be part of the software release. Asserting on behaviors like error handling, branching, parallel, dynamic parallel (map state) helps test the entire state machine’s behavior. For any new behavior in the state machine, such as a new type of exception from a state, you can mock and add as a test.
In this example, you test a scenario in which:
- The identity and address are successfully validated using a Lambda function.
- A positive sentiment is detected using the Comprehend.DetectSentiment API after three retries.
- A contact item is written to a DynamoDB table successfully
- An event is published to an EventBridge event bus successfully
The execution path for this test scenario is shown in the following diagram (the red and green numbers have been added). 0 represents the first execution; 1, 2, and 3 represent the max retry attempts (MaxAttempts), in case of an InternalServerException.
To use service integration mocking, create a mock configuration file with sections specifying mock AWS service responses. These are grouped into test cases that can be activated when executing state machines locally. The following example provides code snippets and the full mock configuration is available in the code repository.
To mock a successful Lambda function invocation, define a mock response that conforms to the Lambda.Invoke API response elements. Associate it to the first request attempt:
"CheckIdentityLambdaMockedSuccess": {
"0": {
"Return": {
"StatusCode": 200,
"Payload": {
"statusCode": 200,
"body": "{\"approved\": true}"
}
}
}
}
To mock the DetectSentiment retry behavior, define failure and successful mock responses that conform to the Comprehend.DetectSentiment API call. Associate the failure mocks to three request attempts, and associate the successful mock to the fourth attempt:
"DetectSentimentRetryOnErrorWithSuccess": {
"0-2": {
"Throw": {
"Error": "InternalServerException",
"Cause": "Server Exception while calling DetectSentiment API in Comprehend Service"
}
},
"3": {
"Return": {
"Sentiment": "POSITIVE",
"SentimentScore": {
"Mixed": 0.00012647535,
"Negative": 0.00008031699,
"Neutral": 0.0051454515,
"Positive": 0.9946478
}
}
}
}
Note that Step Functions Local does not validate the structure of the mocked responses. Ensure that your mocked responses conform to actual responses before testing. To review the structure of service responses, either perform the actual service calls using Step Functions or view the documentation for those services.
Next, associate the mocked responses to a test case identifier:
"RetryOnServiceExceptionTest": {
"Check Identity": "CheckIdentityLambdaMockedSuccess",
"Check Address": "CheckAddressLambdaMockedSuccess",
"DetectSentiment": "DetectSentimentRetryOnErrorWithSuccess",
"Add to FollowUp": "AddToFollowUpSuccess",
"CustomerAddedToFollowup": "CustomerAddedToFollowupSuccess"
}
With the test case and mock responses configured, you can use them for testing with Step Functions Local.
- local_testing.asl.json - State machine definition in ASL
- valid_input.json - State machine unit test input event
- MockConfigFile.json - Unit test mocked responses
- test_step_functions_local.py - Unit tests definition
- Python 3.9 or newer
- Docker
The unit test flow in test_step_funtions_local.py is:
fixture_container
downloads and starts theamazon/aws-stepfunctions-local
container locallyfixture_sfn_client
then creates the state machine using the definition from local_testing. asl.json- When a test runs it calls
execute_stepfunction
. This executes the state machine passing the input in valid_input.json and appendingtest_name
to the state machine ARN.test_name
is used to look up the TestCase entry in MockConfigFile.json which defines which mocked responses to use.execute_stepfunction
returns the list of events from the state machine execution. - After all the tests have run, the container is shut down by the
stop_step_function
finalizer added infixture_container
This example contains a sample event with new user registration data to be processed by the state machine. The unit tests will check how the state machine behaves for each of the following scenarios defined in the MockConfigFile.json file:
HappyPathTest
: every external service runs succesfully and the state machine exits with "CustomerAddedToFollowup".NegativeSentimentTest
: the contact details are properly formatted, a negative sentiment is detected within the user comments and the state machine exits with "NegativeSentimentDetected".RetryOnServiceExceptionTest
: the sentiment detection service fails three times and the state machine retries until successfully retrieving the sentiment in the fourth attempt.
Make sure docker engine is running before running the tests.
step-functions-local$ docker version
Client: Docker Engine - Community
Version: 24.0.6
API version: 1.43
...
To set it up:
step-functions-local$ cd tests
python3 -m venv venv
source venv/bin/activate
pip install --upgrade pip
pip install -r requirements.txt
To run the unit tests:
step-functions-local$ cd tests
python3 -m pytest -s unit/src/test_step_functions_local.py -v