|
|
@ -11,7 +11,7 @@ improve existing code and developers of untested applications tend to |
|
|
|
become pretty paranoid. If an application has automated tests, you can |
|
|
|
become pretty paranoid. If an application has automated tests, you can |
|
|
|
safely make changes and instantly know if anything breaks. |
|
|
|
safely make changes and instantly know if anything breaks. |
|
|
|
|
|
|
|
|
|
|
|
Flask provides a way to test your application by exposing the Werkzeug |
|
|
|
Flask provides a way to test your application by exposing the Werkzeug |
|
|
|
test :class:`~werkzeug.test.Client` and handling the context locals for you. |
|
|
|
test :class:`~werkzeug.test.Client` and handling the context locals for you. |
|
|
|
You can then use that with your favourite testing solution. In this documentation |
|
|
|
You can then use that with your favourite testing solution. In this documentation |
|
|
|
we will use the :mod:`unittest` package that comes pre-installed with Python. |
|
|
|
we will use the :mod:`unittest` package that comes pre-installed with Python. |
|
|
@ -19,17 +19,17 @@ we will use the :mod:`unittest` package that comes pre-installed with Python. |
|
|
|
The Application |
|
|
|
The Application |
|
|
|
--------------- |
|
|
|
--------------- |
|
|
|
|
|
|
|
|
|
|
|
First, we need an application to test; we will use the application from |
|
|
|
First, we need an application to test; we will use the application from |
|
|
|
the :ref:`tutorial`. If you don't have that application yet, get the |
|
|
|
the :ref:`tutorial`. If you don't have that application yet, get the |
|
|
|
sources from `the examples`_. |
|
|
|
sources from `the examples`_. |
|
|
|
|
|
|
|
|
|
|
|
.. _the examples: |
|
|
|
.. _the examples: |
|
|
|
http://github.com/mitsuhiko/flask/tree/master/examples/flaskr/ |
|
|
|
https://github.com/mitsuhiko/flask/tree/master/examples/flaskr/ |
|
|
|
|
|
|
|
|
|
|
|
The Testing Skeleton |
|
|
|
The Testing Skeleton |
|
|
|
-------------------- |
|
|
|
-------------------- |
|
|
|
|
|
|
|
|
|
|
|
In order to test the application, we add a second module |
|
|
|
In order to test the application, we add a second module |
|
|
|
(`flaskr_tests.py`) and create a unittest skeleton there:: |
|
|
|
(`flaskr_tests.py`) and create a unittest skeleton there:: |
|
|
|
|
|
|
|
|
|
|
|
import os |
|
|
|
import os |
|
|
@ -55,15 +55,15 @@ In order to test the application, we add a second module |
|
|
|
|
|
|
|
|
|
|
|
The code in the :meth:`~unittest.TestCase.setUp` method creates a new test |
|
|
|
The code in the :meth:`~unittest.TestCase.setUp` method creates a new test |
|
|
|
client and initializes a new database. This function is called before |
|
|
|
client and initializes a new database. This function is called before |
|
|
|
each individual test function is run. To delete the database after the |
|
|
|
each individual test function is run. To delete the database after the |
|
|
|
test, we close the file and remove it from the filesystem in the |
|
|
|
test, we close the file and remove it from the filesystem in the |
|
|
|
:meth:`~unittest.TestCase.tearDown` method. Additionally during setup the |
|
|
|
:meth:`~unittest.TestCase.tearDown` method. Additionally during setup the |
|
|
|
``TESTING`` config flag is activated. What it does is disabling the error |
|
|
|
``TESTING`` config flag is activated. What it does is disabling the error |
|
|
|
catching during request handling so that you get better error reports when |
|
|
|
catching during request handling so that you get better error reports when |
|
|
|
performing test requests against the application. |
|
|
|
performing test requests against the application. |
|
|
|
|
|
|
|
|
|
|
|
This test client will give us a simple interface to the application. We can |
|
|
|
This test client will give us a simple interface to the application. We can |
|
|
|
trigger test requests to the application, and the client will also keep track |
|
|
|
trigger test requests to the application, and the client will also keep track |
|
|
|
of cookies for us. |
|
|
|
of cookies for us. |
|
|
|
|
|
|
|
|
|
|
|
Because SQLite3 is filesystem-based we can easily use the tempfile module |
|
|
|
Because SQLite3 is filesystem-based we can easily use the tempfile module |
|
|
@ -89,8 +89,8 @@ with an exception. |
|
|
|
The First Test |
|
|
|
The First Test |
|
|
|
-------------- |
|
|
|
-------------- |
|
|
|
|
|
|
|
|
|
|
|
Now it's time to start testing the functionality of the application. |
|
|
|
Now it's time to start testing the functionality of the application. |
|
|
|
Let's check that the application shows "No entries here so far" if we |
|
|
|
Let's check that the application shows "No entries here so far" if we |
|
|
|
access the root of the application (``/``). To do this, we add a new |
|
|
|
access the root of the application (``/``). To do this, we add a new |
|
|
|
test method to our class, like this:: |
|
|
|
test method to our class, like this:: |
|
|
|
|
|
|
|
|
|
|
@ -109,13 +109,13 @@ test method to our class, like this:: |
|
|
|
rv = self.app.get('/') |
|
|
|
rv = self.app.get('/') |
|
|
|
assert 'No entries here so far' in rv.data |
|
|
|
assert 'No entries here so far' in rv.data |
|
|
|
|
|
|
|
|
|
|
|
Notice that our test functions begin with the word `test`; this allows |
|
|
|
Notice that our test functions begin with the word `test`; this allows |
|
|
|
:mod:`unittest` to automatically identify the method as a test to run. |
|
|
|
:mod:`unittest` to automatically identify the method as a test to run. |
|
|
|
|
|
|
|
|
|
|
|
By using `self.app.get` we can send an HTTP `GET` request to the application with |
|
|
|
By using `self.app.get` we can send an HTTP `GET` request to the application with |
|
|
|
the given path. The return value will be a :class:`~flask.Flask.response_class` object. |
|
|
|
the given path. The return value will be a :class:`~flask.Flask.response_class` object. |
|
|
|
We can now use the :attr:`~werkzeug.wrappers.BaseResponse.data` attribute to inspect |
|
|
|
We can now use the :attr:`~werkzeug.wrappers.BaseResponse.data` attribute to inspect |
|
|
|
the return value (as string) from the application. In this case, we ensure that |
|
|
|
the return value (as string) from the application. In this case, we ensure that |
|
|
|
``'No entries here so far'`` is part of the output. |
|
|
|
``'No entries here so far'`` is part of the output. |
|
|
|
|
|
|
|
|
|
|
|
Run it again and you should see one passing test:: |
|
|
|
Run it again and you should see one passing test:: |
|
|
@ -132,8 +132,8 @@ Logging In and Out |
|
|
|
|
|
|
|
|
|
|
|
The majority of the functionality of our application is only available for |
|
|
|
The majority of the functionality of our application is only available for |
|
|
|
the administrative user, so we need a way to log our test client in and out |
|
|
|
the administrative user, so we need a way to log our test client in and out |
|
|
|
of the application. To do this, we fire some requests to the login and logout |
|
|
|
of the application. To do this, we fire some requests to the login and logout |
|
|
|
pages with the required form data (username and password). And because the |
|
|
|
pages with the required form data (username and password). And because the |
|
|
|
login and logout pages redirect, we tell the client to `follow_redirects`. |
|
|
|
login and logout pages redirect, we tell the client to `follow_redirects`. |
|
|
|
|
|
|
|
|
|
|
|
Add the following two methods to your `FlaskrTestCase` class:: |
|
|
|
Add the following two methods to your `FlaskrTestCase` class:: |
|
|
@ -194,7 +194,7 @@ suite. |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
.. _MiniTwit Example: |
|
|
|
.. _MiniTwit Example: |
|
|
|
http://github.com/mitsuhiko/flask/tree/master/examples/minitwit/ |
|
|
|
https://github.com/mitsuhiko/flask/tree/master/examples/minitwit/ |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Other Testing Tricks |
|
|
|
Other Testing Tricks |
|
|
|