This project contains automated test sample code samples for serverless applications written in Python. The project demonstrates several techniques for executing tests including mocking, emulation and testing in the cloud specifically when interacting with the Amazon DynamoDB service. Based on current tooling, we recommend customers focus on testing in the cloud as much as possible.
The project uses the AWS Serverless Application Model (SAM) CLI for configuration, testing and deployment.
- app.py - Lambda handler code to test
- template.yaml - SAM script for deployment
- mock_test.py - Unit test using mocks
- test_api_gateway.py - Integration tests on a live stack
The sample project allows a user to call an API endpoint generate a custom "hello" message, and also tracks the messages it generates. A user provides an "id", which the endpoint uses to look up the person's name associated with that id, and generates a message. The message is recorded in DynamoDB and also returned to the caller:
This project consists of an API Gateway, a single AWS Lambda function, and a Amazon DynamoDB table.
The DynamoDB Table is a single-table design, as both the name lookup and the message tracking use the same table. The table schema is defined as follows:
- For all records, the "Partition Key" is the id.
- For name records, the "Sort Key" is set to a constant = "NAME#"
- For message history records, the "Sort Key" is set to "TS#" appended with the current date-time stamp.
- The payloads are in a field named "data".
Data persistence brings additional testing considerations.
First, the data store must be pre-populated with data to test certain functionality. In our example, we need a valid id
to retrieve a name to test our function. Therefore, we will add data to the data stores prior to running the tests. This data seeding operation is performed in the test setup.
Second, the data store will be populated as a side-effect of our testing. In our example, items of recorded "Hello" messages will be populated in our DynamoDB table. To prevent unintended side-effects, we will clean-up data generated during the test execution. This data cleaning operation is performed in the test tear-down.
Third, any identifying values should be unique to the specific test. This will prevent collisions between tests should there be an issue with tear-down. Each test can define a unique postfix to prevent the issues.
In the unit test, all references and calls to the DynamoDB service are mocked on line 18.
The unit test establishes the DYNAMODB_TABLE_NAME environment variable that the Lambda function uses to reference the DynamoDB table. DYNAMODB_TABLE_NAME is definied in the setUp method of test class in mock_test.py.
In a unit test, you must create a mocked version of the DynamoDB table. The example approach in the setUp method of test class in mock_test.py reads in the DynamoDB table schema directly the SAM Template so that the definition is maintained in one place. This simple technique works if there are no intrinsics (like !If or !Ref) in the resource properties for KeySchema, AttributeDefinitions, & BillingMode. Once the mocked table is created, test data is populated.
With the mocked DynamoDB table created and the DYNAMODB_TABLE_NAME set to the mocked table name, the Lambda function will use the mocked DynamoDB table when executing.
The unit test tear-down removes the mocked DynamoDB table and clears the DYNAMODB_TABLE_NAME environment variable.
To run the unit test, execute the following
# Run from the project directory serverless-test-samples/python-test-samples/apigw-lambda-dynamodb
# Create and Activate a Python Virtual Environment
# One-time setup
pip3 install virtualenv
python3 -m venv venv
source ./venv/bin/activate
# install dependencies
pip3 install -r tests/requirements.txt
# run unit tests with mocks
python3 -m pytest -s tests/unit -v
For integration tests, the full stack is deployed before testing:
# Run from the project directory serverless-test-samples/python-test-samples/apigw-lambda-dynamodb
sam build
sam deploy --guided
The integration test setup determines both the API endpoint and the name of the DynamoDB table in the stack.
The integration test then populates data into the DyanamoDB table.
The integration test tear-down removes the seed data, as well as data generated during the test.
To run the integration test, create the environment variable "AWS_SAM_STACK_NAME" with the name of the test stack, and execute the test.
# Run from the project directory serverless-test-samples/python-test-samples/apigw-lambda-dynamodb
# Set the environment variables AWS_SAM_STACK_NAME and (optionally)AWS_DEFAULT_REGION
# to match the name of the stack and the region where you will test
AWS_SAM_STACK_NAME=<stack-name> AWS_DEFAULT_REGION=<region_name> python -m pytest -s tests/integration -v