Chainmock - Mocking library for Python and Pytest.
Documentation
Chainmock is a mocking Library for Python and pytest. Under the hood it uses Python standard library mocks providing an alternative syntax to create mocks and assertions. Chainmock also comes with some additional features to make testing faster and more straightforward. The syntax works especially well with pytest fixtures.
Installation
Install with pip:
pip install chainmock
Features
Chainmock supports all the same features that Python standard library unittest
mocks support and adds some convenient extra functionality.
- Mocking: Create mocks and assert call counts and arguments or replace
return values.
- Spying: Spying proxies the calls to the original function or method.
With spying you can assert call counts and arguments without mocking.
- Stubs: Easily create stub objects that can be used in tests as fake data
or to replace real objects.
- Async support: Chainmock supports mocking and spying async functions and
methods. Most of the time it also recognizes automatically when async mocking
should be used so it is not any harder than mocking sync code.
- Fully type annotated: The whole codebase is fully type annotated so
Chainmock works well with static analysis tools and editor autocomplete.
- Works with Python 3.9+ and PyPy3.
- Supports
pytest
, unittest
, and doctest
test runners.
Examples
The entrypoint to Chainmock is the mocker
function. Import the mocker
function as follows:
from chainmock import mocker
Mocking
To mock you just give the object that you want to mock to the mocker
function.
After this you can start mocking individual attributes and methods as follows:
mocker(Teapot).mock("add_tea").called_once()
mocker(Teapot).mock("brew").return_value("mocked").called_twice()
mocker(Teapot).mock("add_tea").all_calls_with("green").call_count_at_most(2)
Spying
Spying is not any harder than mocking. You just need to call the spy
method
instead of the mock
method. After spying a callable, it works just like before
spying and you can start making assertions on it.
mocker(Teapot).spy("add_tea").called()
mocker(Teapot).spy("add_tea").any_call_with("green").call_count_at_most(2)
Stubs
To create a stub object, just call mocker
function without any arguments.
stub = mocker().mock("my_method").return_value("it works!").self()
assert stub.my_method() == "it works!"
stub = mocker(my_property=10)
assert stub.my_property == 10
For more details and examples, see the API reference.
Similar projects
If chainmock is not what you need, check out also these cool projects:
- flexmock: Chainmock's API is heavily
inspired by flexmock. Flexmock doesn't use standard library unittest and it
has fully custom mocking implementation. Compared to flexmock, chainmock has
more familiar API if you have been using standard library unittest and
Chainmock also supports async mocking and partial argument matching.
- pytest-mock: Similar to
chainmock, pytest-mock is a wrapper for standard library unittest. However,
pytest-mock doesn't provide any extra functionality as it just exposes
unittest mocks directly to the user.
Contributing
Do you like this project and want to help? If you need ideas, check out the open issues and feel free to open a new pull request. Bug reports and feature requests are also very welcome.