A pytest-plugin for updating doctest outputs
Project description
pytest-accept
pytest-accept is a pytest plugin for automatically updating doctest outputs. It runs doctests, observes the generated outputs, and writes them to the doctests' documented outputs.
It's designed for a couple of use cases:
- People who work with doctests and don't enjoy manually copying generated outputs from the pytest error log and pasting them into their doctests' documented outputs. pytest-accept does the copying & pasting for you.
- People who generally find writing tests a bit annoying, and prefer to develop by "running the code and seeing whether it works". This library aims to make testing a joyful part of that development loop.
pytest-accept is decoupled from the doctests it works with — it can be used with existing doctests, and the doctests it edits are no different from normal doctests.
Jesse, what the?
Here's an example of pytest-accept does: given a file like
add.py
containing an incorrect documented output:
def add(x, y):
"""
Adds two values.
>>> add(1, 1)
3
>>> add("ab", "c")
'bac'
"""
return x + y
...running doctests using pytest and passing --accept
replaces the existing
incorrect values with correct values:
pytest --doctest-modules examples/add.py --accept
diff --git a/examples/add.py b/examples/add.py
index 10a71fd..c2c945f 100644
--- a/examples/add.py
+++ b/examples/add.py
@@ -3,10 +3,10 @@ def add(x, y):
Adds two values.
>>> add(1, 1)
- 3
+ 2
>>> add("ab", "c")
- 'bac'
+ 'abc'
"""
return x + y
This style of testing is fairly well-developed in some languages, although still doesn't receive the attention I think it deserves, and historically hasn't had good support in python.
Confusingly, it's referred to "snapshot testing" or "regression testing" or "expect testing" or "literate testing" or "acceptance testing". The best explanation I've seen on this testing style is from Ron Minsky in a Jane Street Blogpost. @matklad also has an excellent summary in his blog post How to Test.
Installation
pip install pytest-accept
What about normal tests?
A previous effort in assert_plugin.py
attempted to do this for assert
statements, and the file contains some notes
on the effort. The biggest problem is pytest stops on the first assert
failure
in each test, which is very limiting. (Whereas pytest can be configured to
continue on doctest failures, which this library takes advantage of.)
It's probably possible to change pytest's behavior here, but it's a significant effort on the pytest codebase.
Some alternatives:
- Use an existing library like pytest-regtest, which offers file snapshot testing (i.e. not inline).
- We could write a specific function / fixture, like
accept(result, "abc")
, similar to frameworks like rust's excellent insta (which I developed some features for), or ocaml's ppx_expect.- But this has the disadvantage of coupling the test to the plugin: it's not
possible to run tests independently of the plugin, or use the plugin on
general
assert
tests. And one of the great elegances of pytest is its deferral to a normalassert
statement.
- But this has the disadvantage of coupling the test to the plugin: it's not
possible to run tests independently of the plugin, or use the plugin on
general
- Some of this testing feels like writing a notebook and testing that. pytest-notebook fully implements this.
Anything else?
Not really! Some things to watch out for:
- It'll replace the file at the end of a test. So — to the extent there are
useful changes to the file between the start and and the end of a test — it'll
overwrite them. Passing
--accept-copy
will cause the plugin to instead create a file named{file}.py.new
.- TODO: Should we disable the plugin on
--pdb
as one way of long-running tests? - It will overwrite the existing values, though these aren't generally useful — they're designed to match the results of the code.
- TODO: Should we disable the plugin on
- This is early, and there are probably some small bugs. Let me know and I'll attempt to fix them.
- It currently doesn't affect the printing of test results; the doctests will
still print as failures.
- TODO: A future version could print something about them being fixed.
- Python's doctest library is imperfect:
- It can't handle indents, and probably other things. (We do handle blank lines though, and TODO: check whether the output we paste is valid doctest output).
- The syntax for
.*
is an ellipsis...
, which is also the syntax for continuing a code line, so it can't be at the start of a line. - The syntax for all the directives is arguably less than aesthetically pleasing.
- It reports line numbers incorrectly in some cases — two docstring lines
separated with continuation character
\
is counted as one, meaning this library will not have access to the correct line number for doctest inputs and outputs.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
File details
Details for the file pytest-accept-0.1.4.tar.gz
.
File metadata
- Download URL: pytest-accept-0.1.4.tar.gz
- Upload date:
- Size: 13.1 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/1.1.6 CPython/3.8.10 Linux/5.4.0-1047-azure
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | f964b81b696efe27a49ab1a63a7719bfb21903072ca9eccbff4d60d206c82434 |
|
MD5 | 4c12cb78285fc1399d817fa0d5c0b2a8 |
|
BLAKE2b-256 | 92f118981d42496bd41917c8985ec6261aa0ca5e56d566650b476164ea94b7f0 |
File details
Details for the file pytest_accept-0.1.4-py3-none-any.whl
.
File metadata
- Download URL: pytest_accept-0.1.4-py3-none-any.whl
- Upload date:
- Size: 13.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/1.1.6 CPython/3.8.10 Linux/5.4.0-1047-azure
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 2ddb25cb239ccae98d033f6b21ec2ab40d01c29ff00411f127f02b5a6c50facf |
|
MD5 | 0d4ad4f4a170989bedd948387d1633a9 |
|
BLAKE2b-256 | 6f2b69d6f28a7f6f766bffe5c1824ac208448bf82fc360133c9512f38cd49647 |