On testing#

When we write code, most of the time, we make mistakes. These mistakes can be hard to see.

Most untrained programmers write code, try it a few times at the interactive prompt, get the answers they expect, and then assume the code is OK.

Long experience shows that this is rarely true:

If it’s not tested, it’s broken

  • The code may give the right answer for some inputs and the wrong answer for others that you did not test;

  • The code may not work on another system or configuration.

The main way to reduce these problems is to write tests.

Writing tests#

For example, let’s say we had a module called rdmodule, like this:

%%file rdmodule.py
def rem_div(arg1, arg2):
    """ Take `arg1` modulo 2, divide by `arg2`
    """
    arg1 == arg1 % 2  # Remainder of dividing by 2.
    return arg1 / arg2
Writing rdmodule.py

We call it rdmodule because it contains the rem_div function.

Interactively, we might try a few numbers:

import rdmodule

# Expecting (1 % 2) / 4 = 0.25
rdmodule.rem_div(1, 4)
0.25
# Expecting (0 % 2) / 3 = 0
rdmodule.rem_div(0, 3)
0.0

That looks right so far. But, if we had explored further, we would have found there’s a problem:

# Expecting (3 % 2) / 3 = 0.3333
rdmodule.rem_div(3, 2)
1.5

Oops, that was not what we wanted. Can you see the problem?

Keep looking for problems#

What we should have done, was write a range of tests for this function, to check it was working as we expect it too. We could make a test using assert. One way of doing that is to put some tests into a function called — say — test_rem_div, like this:

%%file rdmodule.py
def rem_div(arg1, arg2):
    """ Take `arg1` modulo 2, divide by `arg2`
    """
    arg1 == arg1 % 2  # Remainder of dividing by 2.
    return arg1 / arg2


def test_rem_div():
    # Expecting (1 % 2) / 4 = 0.25
    assert rem_div(1, 4) == 1 / 4
    # Expecting (0 % 2) / 3 = 0
    assert rem_div(0, 3) == 0
    # Expecting (3 % 2 / 3 = 0.3333
    assert rem_div(3, 3) == 1 / 3
Overwriting rdmodule.py

Of course we will have to Changing the module, reloading to get the new version of the module:

import importlib

importlib.reload(rdmodule)
<module 'rdmodule' from '/home/runner/work/nipraxis-textbook/nipraxis-textbook/rdmodule.py'>

Then we can run the tests like this:

rdmodule.test_rem_div()
---------------------------------------------------------------------------
AssertionError                            Traceback (most recent call last)
Cell In[7], line 1
----> 1 rdmodule.test_rem_div()

File ~/work/nipraxis-textbook/nipraxis-textbook/rdmodule.py:14, in test_rem_div()
     12 assert rem_div(0, 3) == 0
     13 # Expecting (3 % 2 / 3 = 0.3333
---> 14 assert rem_div(3, 3) == 1 / 3

AssertionError: 

Indeed this reveals we have a problem we need to fix. We will do that soon.

Before we fix the problem, let us save ourselves the reload step, and the step of running the test_rem_div function by hand, by using Pytest.

Pytest has a command line script, pytest which will look for functions that start with the name test_ in .py files, and then run them.

Here we are using the Bash shell terminal available in Linux and macOS to run the command as if from the command line:

%%bash
python3 -m pytest rdmodule.py
============================= test session starts ==============================
platform linux -- Python 3.10.12, pytest-7.4.0, pluggy-1.2.0
rootdir: /home/runner/work/nipraxis-textbook/nipraxis-textbook
collected 1 item

rdmodule.py F                                                            [100%]

=================================== FAILURES ===================================
_________________________________ test_rem_div _________________________________

    def test_rem_div():
        # Expecting (1 % 2) / 4 = 0.25
        assert rem_div(1, 4) == 1 / 4
        # Expecting (0 % 2) / 3 = 0
        assert rem_div(0, 3) == 0
        # Expecting (3 % 2 / 3 = 0.3333
>       assert rem_div(3, 3) == 1 / 3
E       assert 1.0 == (1 / 3)
E        +  where 1.0 = rem_div(3, 3)

rdmodule.py:14: AssertionError
=========================== short test summary info ============================
FAILED rdmodule.py::test_rem_div - assert 1.0 == (1 / 3)
 +  where 1.0 = rem_div(3, 3)
============================== 1 failed in 0.07s ===============================
---------------------------------------------------------------------------
CalledProcessError                        Traceback (most recent call last)
Cell In[8], line 1
----> 1 get_ipython().run_cell_magic('bash', '', 'python3 -m pytest rdmodule.py\n')

File /opt/hostedtoolcache/Python/3.10.12/x64/lib/python3.10/site-packages/IPython/core/interactiveshell.py:2478, in InteractiveShell.run_cell_magic(self, magic_name, line, cell)
   2476 with self.builtin_trap:
   2477     args = (magic_arg_s, cell)
-> 2478     result = fn(*args, **kwargs)
   2480 # The code below prevents the output from being displayed
   2481 # when using magics with decodator @output_can_be_silenced
   2482 # when the last Python token in the expression is a ';'.
   2483 if getattr(fn, magic.MAGIC_OUTPUT_CAN_BE_SILENCED, False):

File /opt/hostedtoolcache/Python/3.10.12/x64/lib/python3.10/site-packages/IPython/core/magics/script.py:154, in ScriptMagics._make_script_magic.<locals>.named_script_magic(line, cell)
    152 else:
    153     line = script
--> 154 return self.shebang(line, cell)

File /opt/hostedtoolcache/Python/3.10.12/x64/lib/python3.10/site-packages/IPython/core/magics/script.py:314, in ScriptMagics.shebang(self, line, cell)
    309 if args.raise_error and p.returncode != 0:
    310     # If we get here and p.returncode is still None, we must have
    311     # killed it but not yet seen its return code. We don't wait for it,
    312     # in case it's stuck in uninterruptible sleep. -9 = SIGKILL
    313     rc = p.returncode or -9
--> 314     raise CalledProcessError(rc, cell)

CalledProcessError: Command 'b'python3 -m pytest rdmodule.py\n'' returned non-zero exit status 1.

If you get No module named pytest, you may need to install it. Check the Pytest web pages for instructions.

Notice that Pytest has found the test_rem_div function and run it, finding our error. Notice too that Pytest gives us lots of information about the test that failed, and the tests that it has run.

Finally, we fix the function:

%%file rdmodule.py
def rem_div(arg1, arg2):
    """ Take `arg1` modulo 2, divide by `arg2`
    """
    # Notice the single =
    arg1 = arg1 % 2  # Remainder of dividing by 2.
    return arg1 / arg2


def test_rem_div():
    # Expecting (1 % 2) / 4 = 0.25
    assert rem_div(1, 4) == 1 / 4
    # Expecting (0 % 2) / 3 = 0
    assert rem_div(0, 3) == 0
    # Expecting (3 % 2) / 3 = 0.3333
    assert rem_div(3, 3) == 1 / 3
Overwriting rdmodule.py

We confirm that the tests pass.

%%bash
python3 -m pytest rdmodule.py
============================= test session starts ==============================
platform linux -- Python 3.10.12, pytest-7.4.0, pluggy-1.2.0
rootdir: /home/runner/work/nipraxis-textbook/nipraxis-textbook
collected 1 item

rdmodule.py .                                                            [100%]

============================== 1 passed in 0.01s ===============================

Test modules#

It can get cluttered to have the test_ functions in the same module as the code. To reduce clutter, we often write the tests out as a separate file module, named after the module it is testing. In this case the file would be test_rdmodule.py, like this:

%%file rdmodule.py
def rem_div(arg1, arg2):
    """ Take `arg1` modulo 2, divide by `arg2`
    """
    # Notice the single =
    arg1 = arg1 % 2  # Remainder of dividing by 2.
    return arg1 / arg2
Overwriting rdmodule.py
%%file test_rdmodule.py

# Import the function we are testing.
from rdmodule import rem_div


def test_rem_div():
    # Expecting (1 % 2) / 4 = 0.25
    assert rem_div(1, 4) == 1 / 4
    # Expecting (0 % 2) / 3 = 0
    assert rem_div(0, 3) == 0
    # Expecting (3 % 2) / 3 = 0.3333
    assert rem_div(3, 3) == 1 / 3
Writing test_rdmodule.py
%%bash
python3 -m pytest test_rdmodule.py
============================= test session starts ==============================
platform linux -- Python 3.10.12, pytest-7.4.0, pluggy-1.2.0
rootdir: /home/runner/work/nipraxis-textbook/nipraxis-textbook
collected 1 item

test_rdmodule.py .                                                       [100%]

============================== 1 passed in 0.01s ===============================

Luckily we thought to test this case. Now we have tested it, we have fixed it. We can keep testing it every time we edit the code, to make sure we haven’t broken anything. This turns out to be very important in assuring yourself that your code still does what you think it does.

The testing habit#

Testing is a habit. Once you have got into that habit, you will find it hard to break, because you will find lots of problems in your code that you did not suspect. With time, you will start to feel uncomfortable if you are using code without tests, because you know that there’s a big risk that it is wrong. Once that discomfort sets in, you are well on your way to become a programmer who can keep learning.