2011-05-27 10:49:47 +00:00
|
|
|
==========
|
|
|
|
Unit tests
|
|
|
|
==========
|
|
|
|
|
2015-02-18 19:19:21 -08:00
|
|
|
.. highlight:: console
|
|
|
|
|
2011-05-27 10:49:47 +00:00
|
|
|
Django comes with a test suite of its own, in the ``tests`` directory of the
|
2011-07-19 13:16:09 +00:00
|
|
|
code base. It's our policy to make sure all tests pass at all times.
|
2011-05-27 10:49:47 +00:00
|
|
|
|
|
|
|
We appreciate any and all contributions to the test suite!
|
|
|
|
|
|
|
|
The Django tests all use the testing infrastructure that ships with Django for
|
2013-12-31 06:24:11 -05:00
|
|
|
testing applications. See :doc:`/topics/testing/overview` for an explanation of
|
|
|
|
how to write new tests.
|
2011-05-27 10:49:47 +00:00
|
|
|
|
|
|
|
.. _running-unit-tests:
|
|
|
|
|
|
|
|
Running the unit tests
|
2016-01-03 12:56:22 +02:00
|
|
|
======================
|
2011-05-27 10:49:47 +00:00
|
|
|
|
|
|
|
Quickstart
|
2016-01-03 12:56:22 +02:00
|
|
|
----------
|
2011-05-27 10:49:47 +00:00
|
|
|
|
2015-06-15 09:43:35 -04:00
|
|
|
If you are on Python 2, you'll first need to install a backport of the
|
|
|
|
``unittest.mock`` module that's available in Python 3. See
|
2014-11-29 11:45:06 -05:00
|
|
|
:ref:`running-unit-tests-dependencies` for details on installing `mock`_ and
|
|
|
|
the other optional test dependencies.
|
|
|
|
|
2011-05-27 10:49:47 +00:00
|
|
|
Running the tests requires a Django settings module that defines the
|
2013-02-28 10:00:38 +02:00
|
|
|
databases to use. To make it easy to get started, Django provides and uses a
|
2015-02-18 19:19:21 -08:00
|
|
|
sample settings module that uses the SQLite database. To run the tests::
|
2011-05-27 10:49:47 +00:00
|
|
|
|
2014-08-27 00:39:07 +01:00
|
|
|
$ git clone https://github.com/django/django.git django-repo
|
2013-12-25 14:54:14 -05:00
|
|
|
$ cd django-repo/tests
|
|
|
|
$ PYTHONPATH=..:$PYTHONPATH ./runtests.py
|
2013-02-28 10:00:38 +02:00
|
|
|
|
2015-03-22 11:19:22 +00:00
|
|
|
.. admonition:: Windows users
|
|
|
|
|
|
|
|
We recommend something like `Git Bash <https://msysgit.github.io/>`_ to run
|
|
|
|
the tests using the above approach.
|
|
|
|
|
2013-10-22 10:52:04 -04:00
|
|
|
You can avoid typing the ``PYTHONPATH`` bit each time by adding your Django
|
|
|
|
checkout to your ``PYTHONPATH`` or by installing the source checkout using pip.
|
|
|
|
See :ref:`installing-development-version`.
|
2013-02-28 10:00:38 +02:00
|
|
|
|
2014-03-16 14:32:55 +05:30
|
|
|
Having problems? See :ref:`troubleshooting-unit-tests` for some common issues.
|
|
|
|
|
2016-06-19 17:08:41 -04:00
|
|
|
Running tests using ``tox``
|
|
|
|
---------------------------
|
|
|
|
|
|
|
|
`Tox <http://tox.testrun.org/>`_ is a tool for running tests in different
|
|
|
|
virtual environments. Django includes a basic ``tox.ini`` that automates some
|
|
|
|
checks that our build server performs on pull requests. To run the unit tests
|
|
|
|
and other checks (such as :ref:`import sorting <coding-style-imports>`, the
|
|
|
|
:ref:`documentation spelling checker <documentation-spelling-check>`, and
|
|
|
|
:ref:`code formatting <coding-style-python>`), install and run the ``tox``
|
|
|
|
command from any place in the Django source tree::
|
|
|
|
|
|
|
|
$ pip install tox
|
|
|
|
$ tox
|
|
|
|
|
|
|
|
By default, ``tox`` runs the test suite with the bundled test settings file for
|
|
|
|
SQLite, ``flake8``, ``isort``, and the documentation spelling checker. In
|
|
|
|
addition to the system dependencies noted elsewhere in this documentation,
|
|
|
|
the commands ``python2`` and ``python3`` must be on your path and linked to
|
|
|
|
the appropriate versions of Python. A list of default environments can be seen
|
|
|
|
as follows::
|
|
|
|
|
|
|
|
$ tox -l
|
|
|
|
py3
|
|
|
|
flake8
|
|
|
|
docs
|
|
|
|
isort
|
|
|
|
|
|
|
|
Testing other Python versions and database backends
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
In addition to the default environments, ``tox`` supports running unit tests
|
|
|
|
for other versions of Python and other database backends. Since Django's test
|
|
|
|
suite doesn't bundle a settings file for database backends other than SQLite,
|
|
|
|
however, you must :ref:`create and provide your own test settings
|
|
|
|
<running-unit-tests-settings>`. For example, to run the tests on Python 3.5
|
|
|
|
using PostgreSQL::
|
|
|
|
|
|
|
|
$ tox -e py35-postgres -- --settings=my_postgres_settings
|
|
|
|
|
|
|
|
This command sets up a Python 3.5 virtual environment, installs Django's
|
|
|
|
test suite dependencies (including those for PostgreSQL), and calls
|
|
|
|
``runtests.py`` with the supplied arguments (in this case,
|
|
|
|
``--settings=my_postgres_settings``).
|
|
|
|
|
|
|
|
The remainder of this documentation shows commands for running tests without
|
|
|
|
``tox``, however, any option passed to ``runtests.py`` can also be passed to
|
|
|
|
``tox`` by prefixing the argument list with ``--``, as above.
|
|
|
|
|
|
|
|
Tox also respects the ``DJANGO_SETTINGS_MODULE`` environment variable, if set.
|
|
|
|
For example, the following is equivalent to the command above::
|
|
|
|
|
|
|
|
$ DJANGO_SETTINGS_MODULE=my_postgres_settings tox -e py35-postgres
|
|
|
|
|
|
|
|
Running the JavaScript tests
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
Django includes a set of :ref:`JavaScript unit tests <javascript-tests>` for
|
|
|
|
functions in certain contrib apps. The JavaScript tests aren't run by default
|
|
|
|
using ``tox`` because they require `Node.js` to be installed and aren't
|
|
|
|
necessary for the majority of patches. To run the JavaScript tests using
|
|
|
|
``tox``::
|
|
|
|
|
|
|
|
$ tox -e javascript
|
|
|
|
|
|
|
|
This command runs ``npm install`` to ensure test requirements are up to
|
|
|
|
date and then runs ``npm test``.
|
|
|
|
|
2012-10-30 20:09:49 -04:00
|
|
|
.. _running-unit-tests-settings:
|
|
|
|
|
2011-05-27 10:49:47 +00:00
|
|
|
Using another ``settings`` module
|
2016-01-03 12:56:22 +02:00
|
|
|
---------------------------------
|
2011-05-27 10:49:47 +00:00
|
|
|
|
2016-04-06 01:22:34 +05:30
|
|
|
The included settings module (``tests/test_sqlite.py``) allows you to run the
|
|
|
|
test suite using SQLite. If you want to run the tests using a different
|
|
|
|
database, you'll need to define your own settings file. Some tests, such as
|
|
|
|
those for ``contrib.postgres``, are specific to a particular database backend
|
|
|
|
and will be skipped if run with a different backend.
|
2011-05-27 10:49:47 +00:00
|
|
|
|
2013-05-20 13:45:32 -04:00
|
|
|
To run the tests with different settings, ensure that the module is on your
|
|
|
|
``PYTHONPATH`` and pass the module with ``--settings``.
|
2011-05-27 10:49:47 +00:00
|
|
|
|
2013-05-20 13:45:32 -04:00
|
|
|
The :setting:`DATABASES` setting in any test settings module needs to define
|
2011-05-27 10:49:47 +00:00
|
|
|
two databases:
|
|
|
|
|
2011-10-14 00:12:01 +00:00
|
|
|
* A ``default`` database. This database should use the backend that
|
2015-09-05 18:57:05 +02:00
|
|
|
you want to use for primary testing.
|
2011-05-27 10:49:47 +00:00
|
|
|
|
2016-03-05 08:39:56 -05:00
|
|
|
* A database with the alias ``other``. The ``other`` database is used to test
|
|
|
|
that queries can be directed to different databases. This database should use
|
|
|
|
the same backend as the ``default``, and it must have a different name.
|
2011-05-27 10:49:47 +00:00
|
|
|
|
|
|
|
If you're using a backend that isn't SQLite, you will need to provide other
|
|
|
|
details for each database:
|
|
|
|
|
2013-09-25 18:20:33 +02:00
|
|
|
* The :setting:`USER` option needs to specify an existing user account
|
2013-11-09 09:41:03 +01:00
|
|
|
for the database. That user needs permission to execute ``CREATE DATABASE``
|
|
|
|
so that the test database can be created.
|
2011-05-27 10:49:47 +00:00
|
|
|
|
2011-10-14 00:12:01 +00:00
|
|
|
* The :setting:`PASSWORD` option needs to provide the password for
|
|
|
|
the :setting:`USER` that has been specified.
|
2011-05-27 10:49:47 +00:00
|
|
|
|
2013-11-09 09:41:03 +01:00
|
|
|
Test databases get their names by prepending ``test_`` to the value of the
|
|
|
|
:setting:`NAME` settings for the databases defined in :setting:`DATABASES`.
|
|
|
|
These test databases are deleted when the tests are finished.
|
|
|
|
|
2011-05-27 10:49:47 +00:00
|
|
|
You will also need to ensure that your database uses UTF-8 as the default
|
|
|
|
character set. If your database server doesn't use UTF-8 as a default charset,
|
2015-09-05 18:57:05 +02:00
|
|
|
you will need to include a value for :setting:`CHARSET <TEST_CHARSET>` in the
|
|
|
|
test settings dictionary for the applicable database.
|
2011-05-27 10:49:47 +00:00
|
|
|
|
2014-11-06 21:33:43 +01:00
|
|
|
.. _runtests-specifying-labels:
|
|
|
|
|
2011-05-27 10:49:47 +00:00
|
|
|
Running only some of the tests
|
2016-01-03 12:56:22 +02:00
|
|
|
------------------------------
|
2011-05-27 10:49:47 +00:00
|
|
|
|
|
|
|
Django's entire test suite takes a while to run, and running every single test
|
|
|
|
could be redundant if, say, you just added a test to Django that you want to
|
|
|
|
run quickly without running everything else. You can run a subset of the unit
|
|
|
|
tests by appending the names of the test modules to ``runtests.py`` on the
|
|
|
|
command line.
|
|
|
|
|
|
|
|
For example, if you'd like to run tests only for generic relations and
|
2015-02-18 19:19:21 -08:00
|
|
|
internationalization, type::
|
2011-05-27 10:49:47 +00:00
|
|
|
|
2013-12-25 14:54:14 -05:00
|
|
|
$ ./runtests.py --settings=path.to.settings generic_relations i18n
|
2011-05-27 10:49:47 +00:00
|
|
|
|
2013-02-26 15:00:16 +01:00
|
|
|
How do you find out the names of individual tests? Look in ``tests/`` — each
|
2015-02-10 12:03:47 -05:00
|
|
|
directory name there is the name of a test.
|
2011-05-27 10:49:47 +00:00
|
|
|
|
|
|
|
If you just want to run a particular class of tests, you can specify a list of
|
|
|
|
paths to individual test classes. For example, to run the ``TranslationTests``
|
2015-02-18 19:19:21 -08:00
|
|
|
of the ``i18n`` module, type::
|
2011-05-27 10:49:47 +00:00
|
|
|
|
2013-12-25 14:54:14 -05:00
|
|
|
$ ./runtests.py --settings=path.to.settings i18n.tests.TranslationTests
|
2011-05-27 10:49:47 +00:00
|
|
|
|
2015-02-18 19:19:21 -08:00
|
|
|
Going beyond that, you can specify an individual test method like this::
|
2011-05-27 10:49:47 +00:00
|
|
|
|
2013-12-25 14:54:14 -05:00
|
|
|
$ ./runtests.py --settings=path.to.settings i18n.tests.TranslationTests.test_lazy_objects
|
2011-05-27 10:49:47 +00:00
|
|
|
|
Fixed #2879 -- Added support for the integration with Selenium and other in-browser testing frameworks. Also added the first Selenium tests for `contrib.admin`. Many thanks to everyone for their contributions and feedback: Mikeal Rogers, Dirk Datzert, mir, Simon G., Almad, Russell Keith-Magee, Denis Golomazov, devin, robertrv, andrewbadr, Idan Gazit, voidspace, Tom Christie, hjwp2, Adam Nelson, Jannis Leidel, Anssi Kääriäinen, Preston Holmes, Bruno Renié and Jacob Kaplan-Moss.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@17241 bcc190cf-cafb-0310-a4f2-bffc1f526a37
2011-12-22 08:33:58 +00:00
|
|
|
Running the Selenium tests
|
2016-01-03 12:56:22 +02:00
|
|
|
--------------------------
|
Fixed #2879 -- Added support for the integration with Selenium and other in-browser testing frameworks. Also added the first Selenium tests for `contrib.admin`. Many thanks to everyone for their contributions and feedback: Mikeal Rogers, Dirk Datzert, mir, Simon G., Almad, Russell Keith-Magee, Denis Golomazov, devin, robertrv, andrewbadr, Idan Gazit, voidspace, Tom Christie, hjwp2, Adam Nelson, Jannis Leidel, Anssi Kääriäinen, Preston Holmes, Bruno Renié and Jacob Kaplan-Moss.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@17241 bcc190cf-cafb-0310-a4f2-bffc1f526a37
2011-12-22 08:33:58 +00:00
|
|
|
|
2016-02-07 07:54:36 +05:30
|
|
|
Some tests require Selenium and a Web browser. To run these tests, you must
|
|
|
|
install the selenium_ package and run the tests with the
|
|
|
|
``--selenium=<BROWSERS>`` option. For example, if you have Firefox and Google
|
|
|
|
Chrome installed::
|
2013-02-23 17:10:48 +01:00
|
|
|
|
2016-02-07 07:54:36 +05:30
|
|
|
$ ./runtests.py --selenium=firefox,chrome
|
|
|
|
|
|
|
|
See the `selenium.webdriver`_ package for the list of available browsers.
|
2013-02-23 17:10:48 +01:00
|
|
|
|
2016-02-19 13:56:13 -05:00
|
|
|
Specifying ``--selenium`` automatically sets ``--tags=selenium`` to run only
|
|
|
|
the tests that require selenium.
|
|
|
|
|
2016-02-07 07:54:36 +05:30
|
|
|
.. _selenium.webdriver: https://github.com/SeleniumHQ/selenium/tree/master/py/selenium/webdriver
|
|
|
|
|
2012-10-30 20:09:49 -04:00
|
|
|
.. _running-unit-tests-dependencies:
|
|
|
|
|
2011-05-27 10:49:47 +00:00
|
|
|
Running all the tests
|
2016-01-03 12:56:22 +02:00
|
|
|
---------------------
|
2011-05-27 10:49:47 +00:00
|
|
|
|
|
|
|
If you want to run the full suite of tests, you'll need to install a number of
|
|
|
|
dependencies:
|
|
|
|
|
2016-04-22 13:26:41 +02:00
|
|
|
* argon2-cffi_ 16.1.0+
|
2013-07-10 12:01:17 -04:00
|
|
|
* bcrypt_
|
|
|
|
* docutils_
|
2015-11-16 22:41:46 +01:00
|
|
|
* enum34_ (Python 2 only)
|
2015-07-29 16:21:03 -05:00
|
|
|
* geoip2_
|
2015-07-03 08:20:53 -04:00
|
|
|
* jinja2_ 2.7+
|
2013-12-02 17:21:48 +00:00
|
|
|
* numpy_
|
2013-07-01 13:58:04 -04:00
|
|
|
* Pillow_
|
2011-10-14 00:12:01 +00:00
|
|
|
* PyYAML_
|
2013-07-01 13:58:04 -04:00
|
|
|
* pytz_
|
2011-10-14 00:12:01 +00:00
|
|
|
* setuptools_
|
|
|
|
* memcached_, plus a :ref:`supported Python binding <memcached>`
|
2015-07-03 08:24:58 -04:00
|
|
|
* mock_ (for Python 2)
|
2011-10-14 00:12:01 +00:00
|
|
|
* gettext_ (:ref:`gettext_on_windows`)
|
2013-07-01 13:58:04 -04:00
|
|
|
* selenium_
|
2014-04-26 10:22:48 +02:00
|
|
|
* sqlparse_
|
2013-07-01 13:58:04 -04:00
|
|
|
|
|
|
|
You can find these dependencies in `pip requirements files`_ inside the
|
|
|
|
``tests/requirements`` directory of the Django source tree and install them
|
2015-02-18 19:19:21 -08:00
|
|
|
like so::
|
2013-12-25 14:54:14 -05:00
|
|
|
|
2014-11-29 11:45:06 -05:00
|
|
|
$ pip install -r tests/requirements/py3.txt # Python 2: py2.txt
|
2013-07-01 13:58:04 -04:00
|
|
|
|
2016-07-30 02:03:48 +02:00
|
|
|
If you encounter an error during the installation, your system might be missing
|
|
|
|
a dependency for one or more of the Python packages. Consult the failing
|
|
|
|
package's documentation or search the Web with the error message that you
|
|
|
|
encounter.
|
|
|
|
|
2013-07-01 13:58:04 -04:00
|
|
|
You can also install the database adapter(s) of your choice using
|
|
|
|
``oracle.txt``, ``mysql.txt``, or ``postgres.txt``.
|
2011-05-27 10:49:47 +00:00
|
|
|
|
|
|
|
If you want to test the memcached cache backend, you'll also need to define
|
|
|
|
a :setting:`CACHES` setting that points at your memcached instance.
|
|
|
|
|
2013-07-01 13:58:04 -04:00
|
|
|
To run the GeoDjango tests, you will need to :doc:`setup a spatial database
|
|
|
|
and install the Geospatial libraries</ref/contrib/gis/install/index>`.
|
|
|
|
|
2011-05-27 10:49:47 +00:00
|
|
|
Each of these dependencies is optional. If you're missing any of them, the
|
|
|
|
associated tests will be skipped.
|
|
|
|
|
2015-12-26 13:14:07 +01:00
|
|
|
.. _argon2-cffi: https://pypi.python.org/pypi/argon2_cffi
|
2013-07-10 12:01:17 -04:00
|
|
|
.. _bcrypt: https://pypi.python.org/pypi/bcrypt
|
|
|
|
.. _docutils: https://pypi.python.org/pypi/docutils
|
2015-11-16 22:41:46 +01:00
|
|
|
.. _enum34: https://pypi.python.org/pypi/enum34
|
2015-07-29 16:21:03 -05:00
|
|
|
.. _geoip2: https://pypi.python.org/pypi/geoip2
|
2015-07-03 08:20:53 -04:00
|
|
|
.. _jinja2: https://pypi.python.org/pypi/jinja2
|
2013-12-02 17:21:48 +00:00
|
|
|
.. _numpy: https://pypi.python.org/pypi/numpy
|
2013-07-01 13:58:04 -04:00
|
|
|
.. _Pillow: https://pypi.python.org/pypi/Pillow/
|
2011-05-27 10:49:47 +00:00
|
|
|
.. _PyYAML: http://pyyaml.org/wiki/PyYAML
|
2013-07-01 13:58:04 -04:00
|
|
|
.. _pytz: https://pypi.python.org/pypi/pytz/
|
2013-12-08 18:39:26 +01:00
|
|
|
.. _setuptools: https://pypi.python.org/pypi/setuptools/
|
2012-06-28 10:49:07 +02:00
|
|
|
.. _memcached: http://memcached.org/
|
2014-11-29 11:45:06 -05:00
|
|
|
.. _mock: https://pypi.python.org/pypi/mock
|
2015-11-29 08:29:46 -08:00
|
|
|
.. _gettext: https://www.gnu.org/software/gettext/manual/gettext.html
|
2013-12-08 18:39:26 +01:00
|
|
|
.. _selenium: https://pypi.python.org/pypi/selenium
|
2014-04-26 10:22:48 +02:00
|
|
|
.. _sqlparse: https://pypi.python.org/pypi/sqlparse
|
2015-08-08 13:56:37 +02:00
|
|
|
.. _pip requirements files: https://pip.pypa.io/en/latest/user_guide.html#requirements-files
|
2011-05-27 10:49:47 +00:00
|
|
|
|
2012-10-11 06:11:52 -04:00
|
|
|
Code coverage
|
2016-01-03 12:56:22 +02:00
|
|
|
-------------
|
2012-10-11 06:11:52 -04:00
|
|
|
|
|
|
|
Contributors are encouraged to run coverage on the test suite to identify areas
|
|
|
|
that need additional tests. The coverage tool installation and use is described
|
|
|
|
in :ref:`testing code coverage<topics-testing-code-coverage>`.
|
|
|
|
|
2015-09-16 02:32:59 +01:00
|
|
|
Coverage should be run in a single process to obtain accurate statistics. To
|
|
|
|
run coverage on the Django test suite using the standard test settings::
|
2013-12-25 14:54:14 -05:00
|
|
|
|
2015-09-16 02:32:59 +01:00
|
|
|
$ coverage run ./runtests.py --settings=test_sqlite --parallel=1
|
2012-10-11 06:11:52 -04:00
|
|
|
|
2015-02-18 19:19:21 -08:00
|
|
|
After running coverage, generate the html report by running::
|
2012-10-11 06:11:52 -04:00
|
|
|
|
2013-12-25 14:54:14 -05:00
|
|
|
$ coverage html
|
2012-10-11 06:11:52 -04:00
|
|
|
|
|
|
|
When running coverage for the Django tests, the included ``.coveragerc``
|
|
|
|
settings file defines ``coverage_html`` as the output directory for the report
|
|
|
|
and also excludes several directories not relevant to the results
|
|
|
|
(test code or external code included in Django).
|
|
|
|
|
2011-07-19 13:16:09 +00:00
|
|
|
.. _contrib-apps:
|
|
|
|
|
2011-05-27 10:49:47 +00:00
|
|
|
Contrib apps
|
2016-01-03 12:56:22 +02:00
|
|
|
============
|
2011-05-27 10:49:47 +00:00
|
|
|
|
2015-02-10 12:03:47 -05:00
|
|
|
Tests for contrib apps can be found in the ``tests/`` directory, typically
|
|
|
|
under ``<app_name>_tests``. For example, tests for ``contrib.auth`` are located
|
|
|
|
in ``tests/auth_tests``.
|
2014-03-16 14:32:55 +05:30
|
|
|
|
|
|
|
.. _troubleshooting-unit-tests:
|
|
|
|
|
|
|
|
Troubleshooting
|
2016-01-03 12:56:22 +02:00
|
|
|
===============
|
2014-03-16 14:32:55 +05:30
|
|
|
|
2014-11-06 21:33:43 +01:00
|
|
|
Many test failures with ``UnicodeEncodeError``
|
2016-01-03 12:56:22 +02:00
|
|
|
----------------------------------------------
|
2014-03-16 14:32:55 +05:30
|
|
|
|
|
|
|
If the ``locales`` package is not installed, some tests will fail with a
|
|
|
|
``UnicodeEncodeError``.
|
|
|
|
|
2015-02-18 19:19:21 -08:00
|
|
|
You can resolve this on Debian-based systems, for example, by running::
|
2014-03-16 14:32:55 +05:30
|
|
|
|
|
|
|
$ apt-get install locales
|
|
|
|
$ dpkg-reconfigure locales
|
2014-11-06 21:33:43 +01:00
|
|
|
|
|
|
|
Tests that only fail in combination
|
2016-01-03 12:56:22 +02:00
|
|
|
-----------------------------------
|
2014-11-06 21:33:43 +01:00
|
|
|
|
|
|
|
In case a test passes when run in isolation but fails within the whole suite,
|
|
|
|
we have some tools to help analyze the problem.
|
|
|
|
|
2014-11-22 17:54:42 +01:00
|
|
|
The ``--bisect`` option of ``runtests.py`` will run the failing test while
|
|
|
|
halving the test set it is run together with on each iteration, often making
|
|
|
|
it possible to identify a small number of tests that may be related to the
|
|
|
|
failure.
|
2014-11-06 21:33:43 +01:00
|
|
|
|
|
|
|
For example, suppose that the failing test that works on its own is
|
2015-02-18 19:19:21 -08:00
|
|
|
``ModelTest.test_eq``, then using::
|
2014-11-06 21:33:43 +01:00
|
|
|
|
2015-01-10 22:52:59 +00:00
|
|
|
$ ./runtests.py --bisect basic.tests.ModelTest.test_eq
|
2014-11-06 21:33:43 +01:00
|
|
|
|
|
|
|
will try to determine a test that interferes with the given one. First, the
|
|
|
|
test is run with the first half of the test suite. If a failure occurs, the
|
|
|
|
first half of the test suite is split in two groups and each group is then run
|
|
|
|
with the specified test. If there is no failure with the first half of the test
|
|
|
|
suite, the second half of the test suite is run with the specified test and
|
|
|
|
split appropriately as described earlier. The process repeats until the set of
|
|
|
|
failing tests is minimized.
|
|
|
|
|
|
|
|
The ``--pair`` option runs the given test alongside every other test from the
|
|
|
|
suite, letting you check if another test has side-effects that cause the
|
2015-02-18 19:19:21 -08:00
|
|
|
failure. So::
|
2014-11-06 21:33:43 +01:00
|
|
|
|
2015-01-10 22:52:59 +00:00
|
|
|
$ ./runtests.py --pair basic.tests.ModelTest.test_eq
|
2014-11-06 21:33:43 +01:00
|
|
|
|
|
|
|
will pair ``test_eq`` with every test label.
|
|
|
|
|
|
|
|
With both ``--bisect`` and ``--pair``, if you already suspect which cases
|
|
|
|
might be responsible for the failure, you may limit tests to be cross-analyzed
|
|
|
|
by :ref:`specifying further test labels <runtests-specifying-labels>` after
|
2015-02-18 19:19:21 -08:00
|
|
|
the first one::
|
2014-11-06 21:33:43 +01:00
|
|
|
|
2015-01-10 22:52:59 +00:00
|
|
|
$ ./runtests.py --pair basic.tests.ModelTest.test_eq queries transactions
|
2014-11-22 17:59:05 +01:00
|
|
|
|
|
|
|
You can also try running any set of tests in reverse using the ``--reverse``
|
|
|
|
option in order to verify that executing tests in a different order does not
|
2015-02-18 19:19:21 -08:00
|
|
|
cause any trouble::
|
2014-11-22 17:59:05 +01:00
|
|
|
|
2015-01-10 22:52:59 +00:00
|
|
|
$ ./runtests.py basic --reverse
|
|
|
|
|
2015-11-16 09:12:26 -05:00
|
|
|
Seeing the SQL queries run during a test
|
2016-01-03 12:56:22 +02:00
|
|
|
----------------------------------------
|
2015-11-16 09:12:26 -05:00
|
|
|
|
2015-01-10 22:52:59 +00:00
|
|
|
If you wish to examine the SQL being run in failing tests, you can turn on
|
|
|
|
:ref:`SQL logging <django-db-logger>` using the ``--debug-sql`` option. If you
|
2015-02-18 19:19:21 -08:00
|
|
|
combine this with ``--verbosity=2``, all SQL queries will be output::
|
2015-01-10 22:52:59 +00:00
|
|
|
|
|
|
|
$ ./runtests.py basic --debug-sql
|
2014-11-22 17:59:05 +01:00
|
|
|
|
2015-11-16 09:12:26 -05:00
|
|
|
Seeing the full traceback of a test failure
|
2016-01-03 12:56:22 +02:00
|
|
|
-------------------------------------------
|
2015-11-16 09:12:26 -05:00
|
|
|
|
|
|
|
By default tests are run in parallel with one process per core. When the tests
|
|
|
|
are run in parallel, however, you'll only see a truncated traceback for any
|
|
|
|
test failures. You can adjust this behavior with the ``--parallel`` option::
|
2015-09-06 11:12:08 +02:00
|
|
|
|
|
|
|
$ ./runtests.py basic --parallel=1
|
|
|
|
|
|
|
|
You can also use the ``DJANGO_TEST_PROCESSES`` environment variable for this
|
|
|
|
purpose.
|
|
|
|
|
2015-11-17 00:33:18 -05:00
|
|
|
Tips for writing tests
|
|
|
|
----------------------
|
|
|
|
|
|
|
|
.. highlight:: python
|
|
|
|
|
|
|
|
Isolating model registration
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
To avoid polluting the global :attr:`~django.apps.apps` registry and prevent
|
|
|
|
unnecessary table creation, models defined in a test method should be bound to
|
|
|
|
a temporary ``Apps`` instance::
|
|
|
|
|
|
|
|
from django.apps.registry import Apps
|
|
|
|
from django.db import models
|
|
|
|
from django.test import SimpleTestCase
|
|
|
|
|
|
|
|
class TestModelDefinition(SimpleTestCase):
|
|
|
|
def test_model_definition(self):
|
|
|
|
test_apps = Apps(['app_label'])
|
|
|
|
|
|
|
|
class TestModel(models.Model):
|
|
|
|
class Meta:
|
|
|
|
apps = test_apps
|
|
|
|
...
|
|
|
|
|
|
|
|
.. function:: django.test.utils.isolate_apps(*app_labels, attr_name=None, kwarg_name=None)
|
|
|
|
|
|
|
|
.. versionadded:: 1.10
|
|
|
|
|
|
|
|
Since this pattern involves a lot of boilerplate, Django provides the
|
|
|
|
:func:`~django.test.utils.isolate_apps` decorator. It's used like this::
|
|
|
|
|
|
|
|
from django.db import models
|
|
|
|
from django.test import SimpleTestCase
|
|
|
|
from django.test.utils import isolate_apps
|
|
|
|
|
|
|
|
class TestModelDefinition(SimpleTestCase):
|
|
|
|
@isolate_apps('app_label')
|
|
|
|
def test_model_definition(self):
|
|
|
|
class TestModel(models.Model):
|
|
|
|
pass
|
|
|
|
...
|
|
|
|
|
|
|
|
.. admonition:: Setting ``app_label``
|
|
|
|
|
|
|
|
Models defined in a test method with no explicit
|
|
|
|
:attr:`~django.db.models.Options.app_label` are automatically assigned the
|
|
|
|
label of the app in which their test class is located.
|
|
|
|
|
|
|
|
In order to make sure the models defined within the context of
|
|
|
|
:func:`~django.test.utils.isolate_apps` instances are correctly
|
|
|
|
installed, you should pass the set of targeted ``app_label`` as arguments:
|
|
|
|
|
|
|
|
.. snippet::
|
|
|
|
:filename: tests/app_label/tests.py
|
|
|
|
|
|
|
|
from django.db import models
|
|
|
|
from django.test import SimpleTestCase
|
|
|
|
from django.test.utils import isolate_apps
|
|
|
|
|
|
|
|
class TestModelDefinition(SimpleTestCase):
|
|
|
|
@isolate_apps('app_label', 'other_app_label')
|
|
|
|
def test_model_definition(self):
|
|
|
|
# This model automatically receives app_label='app_label'
|
|
|
|
class TestModel(models.Model):
|
|
|
|
pass
|
|
|
|
|
|
|
|
class OtherAppModel(models.Model):
|
|
|
|
class Meta:
|
|
|
|
app_label = 'other_app_label'
|
|
|
|
...
|
|
|
|
|
|
|
|
The decorator can also be applied to classes::
|
|
|
|
|
|
|
|
from django.db import models
|
|
|
|
from django.test import SimpleTestCase
|
|
|
|
from django.test.utils import isolate_apps
|
|
|
|
|
|
|
|
@isolate_apps('app_label')
|
|
|
|
class TestModelDefinition(SimpleTestCase):
|
|
|
|
def test_model_definition(self):
|
|
|
|
class TestModel(models.Model):
|
|
|
|
pass
|
|
|
|
...
|
|
|
|
|
|
|
|
The temporary ``Apps`` instance used to isolate model registration can be
|
|
|
|
retrieved as an attribute when used as a class decorator by using the
|
|
|
|
``attr_name`` parameter::
|
|
|
|
|
|
|
|
from django.db import models
|
|
|
|
from django.test import SimpleTestCase
|
|
|
|
from django.test.utils import isolate_apps
|
|
|
|
|
|
|
|
@isolate_apps('app_label', attr_name='apps')
|
|
|
|
class TestModelDefinition(SimpleTestCase):
|
|
|
|
def test_model_definition(self):
|
|
|
|
class TestModel(models.Model):
|
|
|
|
pass
|
|
|
|
self.assertIs(self.apps.get_model('app_label', 'TestModel'), TestModel)
|
|
|
|
|
|
|
|
Or as an argument on the test method when used as a method decorator by using
|
|
|
|
the ``kwarg_name`` parameter::
|
|
|
|
|
|
|
|
from django.db import models
|
|
|
|
from django.test import SimpleTestCase
|
|
|
|
from django.test.utils import isolate_apps
|
|
|
|
|
|
|
|
class TestModelDefinition(SimpleTestCase):
|
|
|
|
@isolate_apps('app_label', kwarg_name='apps')
|
|
|
|
def test_model_definition(self, apps):
|
|
|
|
class TestModel(models.Model):
|
|
|
|
pass
|
|
|
|
self.assertIs(apps.get_model('app_label', 'TestModel'), TestModel)
|