test_ok2/doc/en/setup.txt

213 lines
7.2 KiB
Plaintext

.. _xunitsetup:
.. _setup:
.. _`setup functions`:
.. _`@pytest.setup`:
``@setup`` functions or: xunit on steroids
========================================================
.. versionadded:: 2.3
.. _`funcargs`: funcargs.html
.. _`test parametrization`: funcargs.html#parametrizing-tests
.. _`unittest plugin`: plugin/unittest.html
.. _`xUnit`: http://en.wikipedia.org/wiki/XUnit
Python, Java and many other languages support a so called xUnit_ style
of resource setup. This typically involves the call of a ``setup``
("fixture") method before running a test function and ``teardown`` after
it has finished. Unlike :ref:`injected resources <resources>` setup
functions work indirectly by causing global side effects or
setting test case attributes which test methods can then access.
pytest originally introduced in 2005 a scope-specific model of detecting
setup and teardown functions on a per-module, class or function basis.
The Python unittest package and nose have subsequently incorporated them.
This model remains supported by pytest as :ref:`old-style xunit`.
Moreover, pytest-2.3 introduces a new ``pytest.setup()`` decorator
to mark functions as setup functions which allow to implement everything
you can do with the old-style and much more. Specifically setup functions:
- can receive :ref:`resources through funcargs <resources>`,
- fully interoperate with parametrized resources,
- can be defined in a plugin or :ref:`conftest.py <conftest.py>` file and get called
on a per-session, per-module, per-class or per-function basis,
- can access the :ref:`request <request>` for which the setup is called,
- can precisely control teardown by registering one or multiple
teardown functions as soon as they have performed some actions
which need undoing, eliminating the no need for a separate
"teardown" decorator.
- allow to separate different setup concerns even if they
happen to work in the same scope
All of these features are now demonstrated by little examples.
.. _`new_setup`:
.. _`@pytest.setup`:
basic per-function setup
-------------------------------
.. regendoc:wipe
Suppose you want to have a clean directory with a single
file entry for each test function in a module and have
the test execute with this directory as current working dir::
# content of test_funcdir.py
import pytest
import os
@pytest.setup()
def mydir(tmpdir):
tmpdir.join("myfile").write("example content")
old = tmpdir.chdir()
def test_function1():
assert os.path.exists("myfile")
f = open("anotherfile", "w")
f.write("")
f.close()
def test_function2():
assert os.path.exists("myfile")
assert not os.path.exists("anotherfile")
Our ``mydir`` setup function is executed on a per-function basis,
the default scope used by the ``pytest.setup`` decorator.
It accesses the ``tmpdir`` resource which provides a new empty
directory path object. The ``test_function2`` here checks that
it executes with a fresh directory and that it
does not see the previously created ``anotherfile``. We can
thus expect two passing tests::
$ py.test -v
=========================== test session starts ============================
platform linux2 -- Python 2.7.3 -- pytest-2.3.0.dev7 -- /home/hpk/venv/1/bin/python
cachedir: /home/hpk/tmp/doc-exec-410/.cache
plugins: xdist, bugzilla, cache, oejskit, cli, pep8, cov
collecting ... collected 2 items
test_funcdir.py:9: test_function1 PASSED
test_funcdir.py:15: test_function2 PASSED
========================= 2 passed in 0.26 seconds =========================
per-function setup, for every function of a project
------------------------------------------------------------
If you want to define a setup per-function but want to apply
it to every function in your project you don't need to duplicate
the setup-definition into each test module. Instead you can put
it into a ``conftest.py`` file into the root of your project::
# content of conftest.py
import pytest
import os
@pytest.setup()
def cleandir(tmpdir):
old = tmpdir.chdir()
The ``cleandir`` setup function will be called for every test function
below the directory tree where ``conftest.py`` resides. In this
case it just uses the builtin ``tmpdir`` resource to change to the
empty directory ahead of running a test.
test modules accessing a global resource
-------------------------------------------------------
.. note::
Relying on `global state is considered bad programming practise <http://en.wikipedia.org/wiki/Global_variable>`_ but when you work with an application
that relies on it you often have no choice.
If you want test modules to access a global resource,
you can stick the resource to the module globals in
a per-module setup function. We use a :ref:`resource factory
<@pytest.factory>` to create our global resource::
# content of conftest.py
import pytest
class GlobalResource:
def __init__(self):
pass
@pytest.factory(scope="session")
def globresource():
return GlobalResource()
@pytest.setup(scope="module")
def setresource(request, globresource):
request.module.globresource = globresource
Now any test module can access ``globresource`` as a module global::
# content of test_glob.py
def test_1():
print ("test_1 %s" % globresource)
def test_2():
print ("test_2 %s" % globresource)
Let's run this module without output-capturing::
$ py.test -qs test_glob.py
collecting ... collected 2 items
..
2 passed in 0.02 seconds
test_1 <conftest.GlobalResource instance at 0x13197e8>
test_2 <conftest.GlobalResource instance at 0x13197e8>
The two tests see the same global ``globresource`` object.
Parametrizing the global resource
+++++++++++++++++++++++++++++++++++++++++++++++++
We extend the previous example and add parametrization to the globresource
factory and also add a finalizer::
# content of conftest.py
import pytest
class GlobalResource:
def __init__(self, param):
self.param = param
@pytest.factory(scope="session", params=[1,2])
def globresource(request):
g = GlobalResource(request.param)
def fin():
print "finalizing", g
request.addfinalizer(fin)
return g
@pytest.setup(scope="module")
def setresource(request, globresource):
request.module.globresource = globresource
And then re-run our test module::
$ py.test -qs test_glob.py
collecting ... collected 4 items
....
4 passed in 0.02 seconds
test_1 <conftest.GlobalResource instance at 0x1922e18>
test_2 <conftest.GlobalResource instance at 0x1922e18>
finalizing <conftest.GlobalResource instance at 0x1922e18>
test_1 <conftest.GlobalResource instance at 0x1925518>
test_2 <conftest.GlobalResource instance at 0x1925518>
finalizing <conftest.GlobalResource instance at 0x1925518>
We are now running the two tests twice with two different global resource
instances. Note that the tests are ordered such that only
one instance is active at any given time: the finalizer of
the first globresource instance is called before the second
instance is created and sent to the setup functions.