get to a workable state for cached_setup() and docs, move msot related code to SetupState class
--HG-- branch : trunk
This commit is contained in:
parent
767dcc69f3
commit
4035fa6326
|
@ -120,21 +120,54 @@ to access test configuration and test context:
|
||||||
|
|
||||||
``request.param``: if exists was passed by a `parametrizing test generator`_
|
``request.param``: if exists was passed by a `parametrizing test generator`_
|
||||||
|
|
||||||
|
perform scoped setup and teardown
|
||||||
|
---------------------------------------------
|
||||||
|
|
||||||
|
.. sourcecode:: python
|
||||||
|
|
||||||
|
def cached_setup(setup, teardown=None, scope="module", keyextra=None):
|
||||||
|
""" setup and return value of calling setup(), cache results and
|
||||||
|
optionally teardown the value by calling ``teardown(value)``. The scope
|
||||||
|
determines the key for cashing the setup value. Specify ``keyextra``
|
||||||
|
to add to the cash-key.
|
||||||
|
scope == 'function': when test function run finishes.
|
||||||
|
scope == 'module': when tests in a different module are run
|
||||||
|
scope == 'run': when the test run has been finished.
|
||||||
|
"""
|
||||||
|
|
||||||
|
example for providing a value that is to be setup only once during a test run:
|
||||||
|
|
||||||
|
.. sourcecode:: python
|
||||||
|
|
||||||
|
def pytest_funcarg__db(request):
|
||||||
|
return request.cached_setup(
|
||||||
|
lambda: ExpensiveSetup(request.config.option.db),
|
||||||
|
lambda val: val.close(),
|
||||||
|
scope="run"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
cleanup after test function execution
|
cleanup after test function execution
|
||||||
---------------------------------------------
|
---------------------------------------------
|
||||||
|
|
||||||
Request objects allow to **register a finalizer method** which is
|
.. sourcecode:: python
|
||||||
called after a test function has finished running.
|
|
||||||
This is useful for tearing down or cleaning up
|
def addfinalizer(func, scope="function"):
|
||||||
test state related to a function argument. Here is a basic
|
""" register calling a a finalizer function.
|
||||||
example for providing a ``myfile`` object that will be
|
scope == 'function': when test function run finishes.
|
||||||
closed upon test function finish:
|
scope == 'module': when tests in a different module are run
|
||||||
|
scope == 'run': when all tests have been run.
|
||||||
|
"""
|
||||||
|
|
||||||
|
Calling ``request.addfinalizer()`` is useful for scheduling teardown
|
||||||
|
functions. The given scope determines when the teardown function
|
||||||
|
will be called. Here is a basic example for providing a ``myfile``
|
||||||
|
object that is to be closed when the test function finishes.
|
||||||
|
|
||||||
.. sourcecode:: python
|
.. sourcecode:: python
|
||||||
|
|
||||||
def pytest_funcarg__myfile(self, request):
|
def pytest_funcarg__myfile(self, request):
|
||||||
# ... create and open a "myfile" object ...
|
# ... create and open a unique per-function "myfile" object ...
|
||||||
request.addfinalizer(lambda: myfile.close())
|
request.addfinalizer(lambda: myfile.close())
|
||||||
return myfile
|
return myfile
|
||||||
|
|
||||||
|
|
|
@ -89,14 +89,36 @@ class FuncargRequest:
|
||||||
attrname=self._argprefix + str(argname)
|
attrname=self._argprefix + str(argname)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def cached_setup(self, setup, teardown=None, scope="module", extrakey=None):
|
||||||
|
if not hasattr(self.config, '_setupcache'):
|
||||||
|
self.config._setupcache = {}
|
||||||
|
cachekey = (self._getscopeitem(scope), extrakey)
|
||||||
|
cache = self.config._setupcache
|
||||||
|
try:
|
||||||
|
val = cache[cachekey]
|
||||||
|
except KeyError:
|
||||||
|
val = setup()
|
||||||
|
cache[cachekey] = val
|
||||||
|
if teardown is not None:
|
||||||
|
self.addfinalizer(lambda: teardown(val), scope=scope)
|
||||||
|
return val
|
||||||
|
|
||||||
def call_next_provider(self):
|
def call_next_provider(self):
|
||||||
if not self._provider:
|
if not self._provider:
|
||||||
raise self.Error("no provider methods left")
|
raise self.Error("no provider methods left")
|
||||||
next_provider = self._provider.pop()
|
next_provider = self._provider.pop()
|
||||||
return next_provider(request=self)
|
return next_provider(request=self)
|
||||||
|
|
||||||
def addfinalizer(self, finalizer):
|
def _getscopeitem(self, scope):
|
||||||
self._pyfuncitem.addfinalizer(finalizer)
|
if scope == "function":
|
||||||
|
return self._pyfuncitem
|
||||||
|
elif scope == "module":
|
||||||
|
return self._pyfuncitem._getparent(py.test.collect.Module)
|
||||||
|
raise ValueError("unknown finalization scope %r" %(scope,))
|
||||||
|
|
||||||
|
def addfinalizer(self, finalizer, scope="function"):
|
||||||
|
colitem = self._getscopeitem(scope)
|
||||||
|
self.config._setupstate.addfinalizer(finalizer=finalizer, colitem=colitem)
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return "<FuncargRequest %r for %r>" %(self.argname, self._pyfuncitem)
|
return "<FuncargRequest %r for %r>" %(self.argname, self._pyfuncitem)
|
||||||
|
|
|
@ -328,7 +328,6 @@ class Function(FunctionMixin, py.test.collect.Item):
|
||||||
def __init__(self, name, parent=None, config=None, args=(),
|
def __init__(self, name, parent=None, config=None, args=(),
|
||||||
callspec=None, callobj=_dummy):
|
callspec=None, callobj=_dummy):
|
||||||
super(Function, self).__init__(name, parent, config=config)
|
super(Function, self).__init__(name, parent, config=config)
|
||||||
self._finalizers = []
|
|
||||||
self._args = args
|
self._args = args
|
||||||
if args:
|
if args:
|
||||||
assert not callspec, "yielded functions (deprecated) cannot have funcargs"
|
assert not callspec, "yielded functions (deprecated) cannot have funcargs"
|
||||||
|
@ -339,15 +338,8 @@ class Function(FunctionMixin, py.test.collect.Item):
|
||||||
if callobj is not _dummy:
|
if callobj is not _dummy:
|
||||||
self._obj = callobj
|
self._obj = callobj
|
||||||
|
|
||||||
def addfinalizer(self, func):
|
#def addfinalizer(self, func):
|
||||||
self._finalizers.append(func)
|
# self.config._setupstate.ddfinalizer(func, colitem=self)
|
||||||
|
|
||||||
def teardown(self):
|
|
||||||
finalizers = self._finalizers
|
|
||||||
while finalizers:
|
|
||||||
call = finalizers.pop()
|
|
||||||
call()
|
|
||||||
super(Function, self).teardown()
|
|
||||||
|
|
||||||
def readkeywords(self):
|
def readkeywords(self):
|
||||||
d = super(Function, self).readkeywords()
|
d = super(Function, self).readkeywords()
|
||||||
|
@ -370,7 +362,10 @@ class Function(FunctionMixin, py.test.collect.Item):
|
||||||
return (self.name == other.name and
|
return (self.name == other.name and
|
||||||
self._args == other._args and
|
self._args == other._args and
|
||||||
self.parent == other.parent and
|
self.parent == other.parent and
|
||||||
self.obj == other.obj)
|
self.obj == other.obj and
|
||||||
|
getattr(self, '_requestparam', None) ==
|
||||||
|
getattr(other, '_requestparam', None)
|
||||||
|
)
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
pass
|
pass
|
||||||
return False
|
return False
|
||||||
|
|
|
@ -172,16 +172,32 @@ class SetupState(object):
|
||||||
""" shared state for setting up/tearing down test items or collectors. """
|
""" shared state for setting up/tearing down test items or collectors. """
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.stack = []
|
self.stack = []
|
||||||
|
self._finalizers = {}
|
||||||
|
|
||||||
|
def addfinalizer(self, finalizer, colitem):
|
||||||
|
assert callable(finalizer)
|
||||||
|
#assert colitem in self.stack
|
||||||
|
self._finalizers.setdefault(colitem, []).append(finalizer)
|
||||||
|
|
||||||
|
def _teardown(self, colitem):
|
||||||
|
finalizers = self._finalizers.pop(colitem, None)
|
||||||
|
while finalizers:
|
||||||
|
fin = finalizers.pop()
|
||||||
|
fin()
|
||||||
|
colitem.teardown()
|
||||||
|
for colitem in self._finalizers:
|
||||||
|
assert colitem in self.stack
|
||||||
|
|
||||||
def teardown_all(self):
|
def teardown_all(self):
|
||||||
while self.stack:
|
while self.stack:
|
||||||
col = self.stack.pop()
|
col = self.stack.pop()
|
||||||
col.teardown()
|
self._teardown(col)
|
||||||
|
assert not self._finalizers
|
||||||
|
|
||||||
def teardown_exact(self, item):
|
def teardown_exact(self, item):
|
||||||
if self.stack and self.stack[-1] == item:
|
if self.stack and self.stack[-1] == item:
|
||||||
col = self.stack.pop()
|
col = self.stack.pop()
|
||||||
col.teardown()
|
self._teardown(col)
|
||||||
|
|
||||||
def prepare(self, colitem):
|
def prepare(self, colitem):
|
||||||
""" setup objects along the collector chain to the test-method
|
""" setup objects along the collector chain to the test-method
|
||||||
|
@ -191,7 +207,7 @@ class SetupState(object):
|
||||||
if self.stack == needed_collectors[:len(self.stack)]:
|
if self.stack == needed_collectors[:len(self.stack)]:
|
||||||
break
|
break
|
||||||
col = self.stack.pop()
|
col = self.stack.pop()
|
||||||
col.teardown()
|
self._teardown(col)
|
||||||
for col in needed_collectors[len(self.stack):]:
|
for col in needed_collectors[len(self.stack):]:
|
||||||
col.setup()
|
col.setup()
|
||||||
self.stack.append(col)
|
self.stack.append(col)
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import py
|
import py
|
||||||
from py.__.test.conftesthandle import Conftest
|
from py.__.test.conftesthandle import Conftest
|
||||||
|
|
||||||
def pytest_generate_tests(metafunc, generator):
|
def pytest_generate_tests(metafunc):
|
||||||
if "basedir" in metafunc.funcargnames:
|
if "basedir" in metafunc.funcargnames:
|
||||||
metafunc.addcall(param="global")
|
metafunc.addcall(param="global")
|
||||||
metafunc.addcall(param="inpackage")
|
metafunc.addcall(param="inpackage")
|
||||||
|
@ -15,8 +15,7 @@ def pytest_funcarg__basedir(request):
|
||||||
d.ensure("adir/__init__.py")
|
d.ensure("adir/__init__.py")
|
||||||
d.ensure("adir/b/__init__.py")
|
d.ensure("adir/b/__init__.py")
|
||||||
return d
|
return d
|
||||||
return request.cached_setup(perclass=basedirmaker)
|
return request.cached_setup(lambda: basedirmaker(request), extrakey=request.param)
|
||||||
return request.cached_setup(perclass=basedirmaker)
|
|
||||||
|
|
||||||
class TestConftestValueAccessGlobal:
|
class TestConftestValueAccessGlobal:
|
||||||
def test_basic_init(self, basedir):
|
def test_basic_init(self, basedir):
|
||||||
|
|
|
@ -121,9 +121,73 @@ class TestRequest:
|
||||||
def test_func(something): pass
|
def test_func(something): pass
|
||||||
""")
|
""")
|
||||||
req = funcargs.FuncargRequest(item, "something")
|
req = funcargs.FuncargRequest(item, "something")
|
||||||
|
py.test.raises(ValueError, "req.addfinalizer(None, scope='xyz')")
|
||||||
l = [1]
|
l = [1]
|
||||||
req.addfinalizer(l.pop)
|
req.addfinalizer(l.pop)
|
||||||
item.teardown()
|
req.config._setupstate._teardown(item)
|
||||||
|
assert not l
|
||||||
|
|
||||||
|
def test_request_cachedsetup(self, testdir):
|
||||||
|
item1,item2 = testdir.getitems("""
|
||||||
|
class TestClass:
|
||||||
|
def test_func1(self, something):
|
||||||
|
pass
|
||||||
|
def test_func2(self, something):
|
||||||
|
pass
|
||||||
|
""")
|
||||||
|
req1 = funcargs.FuncargRequest(item1, "something")
|
||||||
|
l = ["hello"]
|
||||||
|
def setup():
|
||||||
|
return l.pop()
|
||||||
|
ret1 = req1.cached_setup(setup)
|
||||||
|
assert ret1 == "hello"
|
||||||
|
ret1b = req1.cached_setup(setup)
|
||||||
|
assert ret1 == ret1b
|
||||||
|
req2 = funcargs.FuncargRequest(item2, "something")
|
||||||
|
ret2 = req2.cached_setup(setup)
|
||||||
|
assert ret2 == ret1
|
||||||
|
|
||||||
|
def test_request_cachedsetup_extrakey(self, testdir):
|
||||||
|
item1 = testdir.getitem("def test_func(): pass")
|
||||||
|
req1 = funcargs.FuncargRequest(item1, "something")
|
||||||
|
l = ["hello", "world"]
|
||||||
|
def setup():
|
||||||
|
return l.pop()
|
||||||
|
ret1 = req1.cached_setup(setup, extrakey=1)
|
||||||
|
ret2 = req1.cached_setup(setup, extrakey=2)
|
||||||
|
assert ret2 == "hello"
|
||||||
|
assert ret1 == "world"
|
||||||
|
ret1b = req1.cached_setup(setup, extrakey=1)
|
||||||
|
ret2b = req1.cached_setup(setup, extrakey=2)
|
||||||
|
assert ret1 == ret1b
|
||||||
|
assert ret2 == ret2b
|
||||||
|
|
||||||
|
def test_request_cached_setup_functional(self, testdir):
|
||||||
|
testdir.makepyfile(test_0="""
|
||||||
|
l = []
|
||||||
|
def pytest_funcarg__something(request):
|
||||||
|
val = request.cached_setup(setup, teardown)
|
||||||
|
return val
|
||||||
|
def setup(mycache=[1]):
|
||||||
|
l.append(mycache.pop())
|
||||||
|
return l
|
||||||
|
def teardown(something):
|
||||||
|
l.remove(something[0])
|
||||||
|
l.append(2)
|
||||||
|
def test_list_once(something):
|
||||||
|
assert something == [1]
|
||||||
|
def test_list_twice(something):
|
||||||
|
assert something == [1]
|
||||||
|
""")
|
||||||
|
testdir.makepyfile(test_1="""
|
||||||
|
import test_0 # should have run already
|
||||||
|
def test_check_test0_has_teardown_correct():
|
||||||
|
assert test_0.l == [2]
|
||||||
|
""")
|
||||||
|
result = testdir.runpytest("-v")
|
||||||
|
result.stdout.fnmatch_lines([
|
||||||
|
"*3 passed*"
|
||||||
|
])
|
||||||
|
|
||||||
def test_request_getmodulepath(self, testdir):
|
def test_request_getmodulepath(self, testdir):
|
||||||
modcol = testdir.getmodulecol("def test_somefunc(): pass")
|
modcol = testdir.getmodulecol("def test_somefunc(): pass")
|
||||||
|
|
|
@ -242,6 +242,19 @@ class TestFunction:
|
||||||
assert f1 == f1_b
|
assert f1 == f1_b
|
||||||
assert not f1 != f1_b
|
assert not f1 != f1_b
|
||||||
|
|
||||||
|
class callspec1:
|
||||||
|
param = 1
|
||||||
|
funcargs = {}
|
||||||
|
class callspec2:
|
||||||
|
param = 2
|
||||||
|
funcargs = {}
|
||||||
|
f5 = py.test.collect.Function(name="name", config=config,
|
||||||
|
callspec=callspec1, callobj=isinstance)
|
||||||
|
f5b = py.test.collect.Function(name="name", config=config,
|
||||||
|
callspec=callspec2, callobj=isinstance)
|
||||||
|
assert f5 != f5b
|
||||||
|
assert not (f5 == f5b)
|
||||||
|
|
||||||
class TestSorting:
|
class TestSorting:
|
||||||
def test_check_equality_and_cmp_basic(self, testdir):
|
def test_check_equality_and_cmp_basic(self, testdir):
|
||||||
modcol = testdir.getmodulecol("""
|
modcol = testdir.getmodulecol("""
|
||||||
|
|
|
@ -2,6 +2,15 @@
|
||||||
from py.__.test.config import SetupState
|
from py.__.test.config import SetupState
|
||||||
|
|
||||||
class TestSetupState:
|
class TestSetupState:
|
||||||
|
def test_setup(self, testdir):
|
||||||
|
ss = SetupState()
|
||||||
|
item = testdir.getitem("def test_func(): pass")
|
||||||
|
l = [1]
|
||||||
|
ss.addfinalizer(l.pop, colitem=item)
|
||||||
|
ss._teardown(item)
|
||||||
|
assert not l
|
||||||
|
|
||||||
|
class TestSetupStateFunctional:
|
||||||
disabled = True
|
disabled = True
|
||||||
def test_setup_ok(self, testdir):
|
def test_setup_ok(self, testdir):
|
||||||
item = testdir.getitem("""
|
item = testdir.getitem("""
|
||||||
|
@ -90,3 +99,4 @@ class TestSetupState:
|
||||||
assert event.excinfo
|
assert event.excinfo
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue