Steve Piercy
2018-10-09 beeb8ee80c1d0823f965e5826ce3633972904dab
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
.. _qtut_unit_testing:
 
=============================
05: Unit Tests and ``pytest``
=============================
 
Provide unit testing for our project's Python code.
 
 
Background
==========
 
As the mantra says, "Untested code is broken code." The Python community has
had a long culture of writing test scripts which ensure that your code works
correctly as you write it and maintain it in the future. Pyramid has always had
a deep commitment to testing, with 100% test coverage from the earliest
pre-releases.
 
Python includes a :ref:`unit testing framework
<python:unittest-minimal-example>` in its standard library. Over the years a
number of Python projects, such as :ref:`pytest <pytest:features>`, have
extended this framework with alternative test runners that provide more
convenience and functionality. The Pyramid developers use ``pytest``, which
we'll use in this tutorial.
 
Don't worry, this tutorial won't be pedantic about "test-driven development"
(TDD). We'll do just enough to ensure that, in each step, we haven't majorly
broken the code. As you're writing your code, you might find this more
convenient than changing to your browser constantly and clicking reload.
 
We'll also leave discussion of `pytest-cov
<https://pytest-cov.readthedocs.io/en/latest/>`_ for another section.
 
 
Objectives
==========
 
- Write unit tests that ensure the quality of our code.
 
- Install a Python package (``pytest``) which helps in our testing.
 
 
Steps
=====
 
#.  First we copy the results of the previous step.
 
    .. code-block:: bash
 
        cd ..; cp -r debugtoolbar unit_testing; cd unit_testing
 
#.  Add ``pytest`` to our project's dependencies in ``setup.py`` as a :term:`Setuptools` "extra":
 
    .. literalinclude:: unit_testing/setup.py
        :language: python
        :linenos:
        :emphasize-lines: 15
 
#.  Install our project and its newly added dependency.
    Note that we use the extra specifier ``[dev]`` to install testing requirements and surround it with double quote marks.
 
    .. code-block:: bash
 
        $VENV/bin/pip install -e ".[dev]"
 
#.  Now we write a simple unit test in ``unit_testing/tutorial/tests.py``:
 
    .. literalinclude:: unit_testing/tutorial/tests.py
        :linenos:
 
#.  Now run the tests:
 
    .. code-block:: bash
 
 
        $VENV/bin/pytest tutorial/tests.py -q
        .
        1 passed in 0.14 seconds
 
 
Analysis
========
 
Our ``tests.py`` imports the Python standard unit testing framework. To make
writing Pyramid-oriented tests more convenient, Pyramid supplies some
``pyramid.testing`` helpers which we use in the test setup and teardown. Our
one test imports the view, makes a dummy request, and sees if the view returns
what we expect.
 
The ``tests.TutorialViewTests.test_hello_world`` test is a small example of a
unit test. First, we import the view inside each test. Why not import at the
top, like in normal Python code? Because imports can cause effects that break a
test. We'd like our tests to be in *units*, hence the name *unit* testing. Each
test should isolate itself to the correct degree.
 
Our test then makes a fake incoming web request, then calls our Pyramid view.
We test the HTTP status code on the response to make sure it matches our
expectations.
 
Note that our use of ``pyramid.testing.setUp()`` and
``pyramid.testing.tearDown()`` aren't actually necessary here; they are only
necessary when your test needs to make use of the ``config`` object (it's a
Configurator) to add stuff to the configuration state before calling the view.
 
Finally we've introduced the concept of :term:`Setuptools` extras.
These are optional or recommended features that may be installed with an "extras" specifier.
The specifier is the name of a key in a Python dictionary, and is surrounded by square brackets when invoked on the command line.
The value for the key is a Python list of dependencies.
 
 
Extra credit
============
 
#. Change the test to assert that the response status code should be ``404``
   (meaning, not found). Run ``pytest`` again. Read the error report and see
   if you can decipher what it is telling you.
 
#. As a more realistic example, put the ``tests.py`` back as you found it, and
   put an error in your view, such as a reference to a non-existing variable.
   Run the tests and see how this is more convenient than reloading your
   browser and going back to your code.
 
#. Finally, for the most realistic test, read about Pyramid ``Response``
   objects and see how to change the response code. Run the tests and see how
   testing confirms the "contract" that your code claims to support.
 
#. How could we add a unit test assertion to test the HTML value of the
   response body?
 
#. Why do we import the ``hello_world`` view function *inside* the
   ``test_hello_world`` method instead of at the top of the module?
 
.. seealso:: See also :ref:`testing_chapter` and `Setuptools Declaring "Extras" (optional features with their own dependencies) <https://setuptools.readthedocs.io/en/latest/setuptools.html#declaring-extras-optional-features-with-their-own-dependencies>`_.