assert
¶
Learning outcomes
- Understand the purpose of
assert
- Understand that Python has a debug and release mode
- Understand that
assert
is superior to documentation - Understand that
assert
has no reduction in run-time speed in release mode - Understand that
assert
statements should have no side effects - Understand that
assert
can be used to signal stubs assert
liberally
For teachers
Prior:
- Somewhere in you code, you know something, e.g.
x
is positive, and the rest of the code needs that to be true to be able to function. Should, and if yes, how, this knowledge be written down? - What synonymn does 'to assert' have in English?
- Have you ever used asserts? In R, the equivalent is
testthat::expect_
Lesson plans:
gantt
title Lesson plan pair programming 1
dateFormat X
axisFormat %s
Introduction : intro, 0, 5s
Theory 1: theory_1, after intro, 5s
Exercise 1: crit, exercise_1, after theory_1, 15s
Feedback 1: feedback_1, after exercise_1, 5s
Theory 2: theory_2, after feedback_1, 5s
Exercise 2: crit, exercise_2, after theory_2, 10s
Exercise 2 after the break: crit, after exercise_2, 5s
Why?¶
While writing code, you make assumptions.
Writing down these assumptions as assert
statements
will help you structure your thoughts and reduce
the time you spend debugging.
About assumptions¶
Assert liberally to document internal assumptions and invariants
[Sutter & Alexandrescu, 2004]
, chapter 68.
While coding, we have assumptions. Take, for example, this code:
Here, we assume that n_elements
is non-zero,
else we cannot calculate an average.
We can express that with an assert
statement:
Prefer R?
In R, the same story hold for the testthat::expect_true
(as part of the Tidyverse) function.
This assert
will terminate the program if the assertion is false.
Assert liberally to document assumptions
[Sutter & Alexandrescu, 2004][Stroustrup, 1997][McConnell, 2004a]
[Liberty, 2001][Lakos, 1996][Stroustrup, 2013a]
,
even if you think something should never occur [McConnell, 2004b]
.
Assertions in debug and release mode¶
Python scripts can be run in debug mode or release mode:
Do not assume that assert
is always evaluated [Stroustrup, 2013b]
:
when running a script in debug mode or release mode,
assert
has different behavior:
- in debug mode: the
assert
s are active - in release mode: the
assert
s are ignored
This means that there is no drawback in speed when running your script in release mode.
It also mean that assert
statements:
- must not do anything (e.g. modifying a value)
[Turner, 2024]
- cannot be used to talk to the user of your code
Parameter | Debug mode | Release mode |
---|---|---|
Purpose of assert |
To checking yourself | None |
How to run script | python my_script.py |
python -O my_script.py |
Behavior of assert |
Active | Ignored |
Runtime speed | Lower | Unaffected |
Using assertions as a stub¶
assert
can be used as a stub,
to signal that work that needs to be done.
For example, here is a function that aligns any amount of DNA sequences:
def align(dna_sequences):
"""Align the DNA sequences"""
assert len(dna_sequences) == 2 # TODO
# the actual code
The developer (you!) has indicated clearly that the function,
for now, expects to take two DNA sequences.
The # TODO
is signalling clearly that this will be changed in the
future.
Use assert
to specifify assumption on the input of a function
[Stroustrup, 1997][McConnell, 2004a]
.
Note
Specifify: When you have to further define what you are talking about to further explain yourself.
Using assertions to check a function's return value¶
Lastly, assert
can be used to quick check
to check if a function return a not-too-crazy value.
For example, here is a function that aligns two DNA sequences:
def align_two_dna_sequences(dna_sequences):
"""Align the DNA sequences"""
# ....
results = ["AAAA", "AAC-"] # Should be result of calculation
assert len(results[1]) == len(results[2])
return results
Here the assert
checks if the resulting DNA sequences are equally long,
as they should. The resulting DNA sequences may have incorrect content,
which likely is to be checked someplace else. But checking for the
sequences to be of equal length may help as a first quick
test to find bugs.
Use assert
to specifify assumption on the output of a function
[Stroustrup, 1997][McConnell, 2004a]
.
Exercises¶
Exercise 1: use assert
¶
- Write a script called
my_assert.py
with the following content:
- Run the script, using
python my_assert.py
- What is the error message? Why?
Answer
This is the error message:
$ python my_assert.py
Traceback (most recent call last):
File "/home/sven/my_assert.py", line 1, in <module>
assert 1 == 2
AssertionError
It fails because one is, indeed, not equal to two.
- Run the script, now using
python -O my_assert.py
- What is the error message? Why?
Answer
There is no error message here.
There is no error message, as the Python script was run in
release mode: in release mode, all assert
s are removed
Exercise 2: use assert
in the learners' project¶
There is likely to be some code in the learners project that assumptions that are either invisible or written down as comments.
What if not?
Then it is time to develop new code :-) .
Assign yourself an issue and write it.
Put assert
whenever you assume something to be true
Search for one occasion where this happens
and put an assert
statements there.
If you find a bug that way, create an issue to have someone fix that bug.
References¶
[Sutter & Alexandrescu, 2004]
Herb Sutter, Andrei Alexandrescu. C++ coding standards: 101 rules, guidelines, and best practices. 2004. ISBN: 0-32-111358-6. Chapter 68: 'Assert liberally to document internal assumptions and invariants'[Stroustrup, 1997]
Bjarne Stroustrup. The C++ Programming Language (3rd edition). 1997. ISBN: 0-201-88954-4. Advice 24.5.18: 'Explicitly express preconditions, postconditions, and other assertions as assertions'[McConnell, 2004a]
Steve McConnell. Code Complete (2nd edition). 2004. ISBN: -735619670. Chapter 8.2 'Assertions', paragraph 'Guidelines for using asserts': 'Use assertions to document and verify preconditions and postconditions'[McConnell, 2004b]
Steve McConnell. Code Complete (2nd edition). 2004. ISBN: -735619670. Chapter 8.2 'Assertions', paragraph 'Guidelines for using asserts': 'Use assertions for conditions that should never occur'.[Liberty, 2001]
[5] Jesse Liberty. Sams teach yourself C++ in 24 hours. ISBN: 0-672-32224-2. Hour 24, chapter 'assert()': 'Use assert freely'[Lakos, 1996]
John Lakos. Large-Scale C++ Software Design. 1996. ISBN: 0-201-63362-0. Chapter 2.6: 'The use of assert statements can help to document the assumptions you make when implementing your code[Stroustrup, 2013a]
Bjarne Stroustrup. The C++ Programming Language (4th edition). 2013. ISBN: 978-0-321-56384-2. Chapter 30.5. Advice. page 884: '[13] Use static_assert() and assert() extensively'[Stroustrup, 2013b]
Bjarne Stroustrup. The C++ Programming Language (4th edition). 2013. ISBN: 978-0-321-56384-2. Chapter 30.5. Advice. page 884: '[14] Do not assume that assert() is always evaluated'[Turner, 2024]
Jason Turner, cppbestpractices: 'Never Put Code with Side Effects Inside an assert()' here