Test 01 Fastapi Provider¶
Test the FastAPI provider with Pact.
This module demonstrates how to write a provider test using Pact Python's upcoming version 3. Pact, being a consumer-driven testing tool, requires that the provider respond to the requests defined by the consumer. The consumer defines the expected interactions with the provider, and the provider is expected to respond with the expected responses.
This module tests the FastAPI provider defined in src/fastapi.py
against the
mock consumer. The mock consumer is set up by Pact and will replay the requests
defined by the consumers. Pact will then validate that the provider responds
with the expected responses.
The provider will be expected to be in a given state in order to respond to
certain requests. For example, when fetching a user's information, the provider
will need to have a user with the given ID in the database. In order to avoid
side effects, the provider's database calls are mocked out using functionalities
from unittest.mock
.
Note that Pact requires that the provider be running on an accessible URL. This
means that FastAPI's [TestClient
][fastapi.testclient.TestClient] cannot be used
to test the provider. Instead, the provider is run in a separate thread using
Python's Thread
class.
Attributes¶
PROVIDER_URL = URL('http://localhost:8000')
module-attribute
¶
Classes¶
Server
¶
Bases: Server
Custom server class to run the FastAPI server in a separate thread.
Thanks to this StackOverflow answer for this solution.
Functions¶
install_signal_handlers() -> None
¶
Prevent the server from installing signal handlers.
This is required to run the FastAPI server in a separate process. The
default behaviour of uvicorn.Server
is to install signal handlers which
would interfere with the signal handlers of the main process.
Source code in examples/tests/v3/test_01_fastapi_provider.py
run_in_thread() -> Generator[str, None, None]
¶
Run the FastAPI server in a separate thread.
This method runs the FastAPI server in a separate thread and yields the URL of the server. The server is started in a separate thread to allow the tests to run in the main thread.
Source code in examples/tests/v3/test_01_fastapi_provider.py
Functions¶
mock_delete_request_to_delete_user() -> None
¶
Mock the database for the delete request to delete a user.
As with the mock_post_request_to_create_user
function, we are using a
local dictionary to avoid side effects. This function replaces the calls to
the database with a local dictionary to avoid side effects.
Source code in examples/tests/v3/test_01_fastapi_provider.py
mock_post_request_to_create_user() -> None
¶
Mock the database for the post request to create a user.
While the FAKE_DB
is a dictionary in this example, one should imagine that
this is a real database. In this instance, we are replacing the calls to the
database with a local dictionary to avoid side effects; thereby eliminating
the need to stand up a real database for the tests.
The added benefit of using this approach is that the mock can subsequently
be inspected to ensure that the correct calls were made to the database. For
example, asserting that the correct user ID was retrieved from the database.
These checks are performed as part of the teardown
action. This action can
also be used to reset the mock, or in the case were a real database is used,
to clean up any side effects.
Source code in examples/tests/v3/test_01_fastapi_provider.py
mock_user_doesnt_exist() -> None
¶
Mock the database for the user doesn't exist state.
Source code in examples/tests/v3/test_01_fastapi_provider.py
mock_user_exists() -> None
¶
Mock the database for the user exists state.
You may notice that the return value here differs from the consumer's expected response. This is because the consumer's expected response is guided by what the consumer uses.
By using consumer-driven contracts and testing the provider against the consumer's contract, we can ensure that the provider is what the consumer needs. This allows the provider to safely evolve their API (by both adding and removing fields) without fear of breaking the interactions with the consumers.
Source code in examples/tests/v3/test_01_fastapi_provider.py
provider_state_handler(state: str, action: str, _parameters: dict[str, Any] | None) -> None
¶
Handler for the provider state callback.
For Pact to be able to correctly test compliance with the contract, the internal state of the provider needs to be set up correctly. For example, if the consumer expects a user to exist in the database, the provider needs to have a user with the given ID in the database.
Naïvely, this can be achieved by setting up the database with the correct
data for the test, but this can be slow and error-prone, and requires
standing up additional infrastructure. The alternative showcased here is to
mock the relevant calls to the database so as to avoid any side effects. The
unittest.mock
library is used to achieve this as part of the setup
action.
The added benefit of using this approach is that the mock can subsequently
be inspected to ensure that the correct calls were made to the database. For
example, asserting that the correct user ID was retrieved from the database.
These checks are performed as part of the teardown
action. This action can
also be used to reset the mock, or in the case were a real database is used,
to clean up any side effects.
PARAMETER | DESCRIPTION |
---|---|
action
|
One of
TYPE:
|
state
|
The name of the state to set up or tear down.
TYPE:
|
RETURNS | DESCRIPTION |
---|---|
None
|
A dictionary containing the result of the action. |
Source code in examples/tests/v3/test_01_fastapi_provider.py
server() -> Generator[str, None, None]
¶
test_provider(server: str) -> None
¶
Test the FastAPI provider with Pact.
This function performs all of the provider testing. It runs as follows:
- The FastAPI server is started in a separate process. A small wait time is added to allow the server to start up before the tests begin.
-
The Verifier is created and configured.
-
The
set_info
method tells Pact the names of provider to be tested. Pact will automatically discover all the consumers that have contracts with this provider.The
url
parameter is used to specify the base URL of the provider against which the tests will be run. 2. Theadd_source
method adds the directory where the pact files are stored. In a more typical setup, this would in fact be a Pact Broker URL. 3. Theset_state
method defines the endpoint on the provider that will be called to set the provider into the correct state before the tests begin. At present, this is the only way to set the provider into the correct state; however, future version of Pact Python intend to provide a more Pythonic way to do this.
-
-
The
verify
method is called to run the tests. This will run all the tests defined in the pact files and verify that the provider responds correctly to each request. More specifically, for each interaction, it will perform the following steps:- The provider state(s) are by sending a POST request to the
provider's
_pact/callback
endpoint. - Pact impersonates the consumer by sending a request to the provider.
- The provider handles the request and sends a response back to Pact.
- Pact validates the response against the expected response defined in the pact file.
- If
teardown
is set toTrue
forset_state
, Pact will send ateardown
action to the provider to reset the provider state(s).
- The provider state(s) are by sending a POST request to the
provider's
Pact will output the results of the tests to the console and verify that the provider is compliant with the contract. If the provider is not compliant, the tests will fail and the output will show which interactions failed and why.
Source code in examples/tests/v3/test_01_fastapi_provider.py
verify_mock_delete_request_to_delete_user() -> None
¶
Source code in examples/tests/v3/test_01_fastapi_provider.py
verify_mock_post_request_to_create_user() -> None
¶
Source code in examples/tests/v3/test_01_fastapi_provider.py
verify_user_doesnt_exist_mock() -> None
¶
Verify the mock calls for the 'user doesn't exist' state.
This function checks that the mock for FAKE_DB.get
was called, verifies
that it returned None
, and ensures that it was called with an integer
argument. It then resets the mock for future tests.
Source code in examples/tests/v3/test_01_fastapi_provider.py
verify_user_exists_mock() -> None
¶
Verify the mock calls for the 'user exists' state.
This function checks that the mock for FAKE_DB.get
was called, verifies
that it returned the expected user data, and ensures that it was called with
the integer argument 1
. It then resets the mock for future tests.