No magic, intuitive assertion library with descriptive error messages. Works with Python's assert statement and is inspired by pytest support for assertions and grappa-py/grappa descriptive error messages.
Currently supports only Python 3.6 or above.
- No special assertion methods. Uses the default assert statement.
- No magic. Assertion statements are not modified and the default AssertionError class is not overridden.
- High performance. No extra code is executed if the assertion does not fail unless the assertion has side effects.
- No external dependencies.
- Simple and clean API.
- Compatible with most Python test frameworks.
Use error for a single assert statement
from testflows.asserts import error
assert 1 == 1, error()
or use errors context manager to wrap multiple assert statements
from testflows.asserts import errors
with errors():
assert 1 == 1
assert 2 == 2
and if you don't want to abort when an assertion fails and would like to keep going then the errors context manager supports soft assertions through it's error method.
from testflows.asserts import errors
with errors() as soft:
with soft.error():
assert 1 == 2
assert 2 == 2
When an assertion fails a descriptive error message is produced. For example
from testflows.asserts import error assert 1 == 2, error()
produces the following output
AssertionError: Oops! Assertion failed The following assertion was not satisfied assert 1 == 2, error() Assertion values assert 1 == 2, error() ^ is = False assert 1 == 2, error() ^ is False Where File 't.py', line 3 in '<module>' 0| 1| from testflows.asserts import error 2| 3|> assert 1 == 2, error()
The asserts module works similarly to the old implementation of pytest assertions. If the assertion fails, the assert statement is reinterpreted to produce a detailed error message.
Therefore, if the assertion statement has a side effect it might not work as expected when an assertion fails.
In the pytest framework, this problem is solved by rewriting the original assertion. The asserts module solves this problem by explicitly using values context manager to store the values of the expression that has a side effect.
$ ./build; ./install
where
$ ./build
creates a pip installable package in ./dist, for example
$ ls dist/
testflows.asserts-4.1.190811.155018.tar.gz
and
$ ./install
uses sudo pip install command to perform the system-wide installation.
If assertion has side effects then values context manager can be used to address this problem.
The example below demonstrates the problem.
from testflows.asserts import error
buf = [1]
assert buf.append(2) and buf, error()
In the code above, the assertion fails and the buf list is modified twice. Once when the assertion fails and once when the assertion is reinterpreted when error() method is evaluated.
The error message that is produced shows the problem
The following assertion was not satisfied
assert buf.append(2) and buf, error()
Assertion values
assert buf.append(2) and buf, error()
^ is [1, 2, 2]
assert buf.append(2) and buf, error()
^ is = <built-in method append of list object at 0x7f13d1c41248>
assert buf.append(2) and buf, error()
^ is = None
assert buf.append(2) and buf, error()
^ is [1, 2, 2]
assert buf.append(2) and buf, error()
^ is = None
assert buf.append(2) and buf, error()
^ is False
Where
File 't.py', line 4 in '<module>'
1| from testflows.asserts import error
2|
3| buf = [1]
4|> assert buf.append(2) and buf, error()
specifically, the lines below show that value of buf is [1,2,2] instead of the desired value of [1,2]
Assertion values
assert buf.append(2) and buf, error()
^ is [1, 2, 2]
In order to work around this problem, values context manager can be used as follows
from testflows.asserts import values, error
buf = [1]
with values() as that:
assert that(buf.append(2)) and buf, error()
and it will produce the error message
The following assertion was not satisfied
assert that(buf.append(2)) and buf, error()
Assertion values
assert that(buf.append(2)) and buf, error()
^ is = None
assert that(buf.append(2)) and buf, error()
^ is [1, 2]
assert that(buf.append(2)) and buf, error()
^ is = None
assert that(buf.append(2)) and buf, error()
^ is False
Where
File 't.py', line 5 in '<module>'
1| from testflows.asserts import values, error
2|
3| buf = [1]
4| with values() as that:
5|> assert that(buf.append(2)) and buf, error()
the lines below show that the buf list has the expected value of [1,2]
assert that(buf.append(2)) and buf, error()
^ is [1, 2]
this is because the expression passed to that is not reinterpreted and only the result of the expression is stored and used during the generation of the error message.
The explicit use of values context manager provides a simple solution without any need to rewrite the original assertion statement.