adding _id parameter to addcall, refinements

--HG--
branch : trunk
This commit is contained in:
holger krekel 2009-05-12 02:05:59 +02:00
parent c3f3dc653e
commit ebb0de4ff7
4 changed files with 45 additions and 17 deletions

View File

@ -41,7 +41,7 @@ Let's look at a simple example of using funcargs within a test module:
one can see the original provided value. one can see the original provided value.
generating test runs with multiple function argument values generating test functions call
---------------------------------------------------------------------- ----------------------------------------------------------------------
You can parametrize multiple runs of a test function by You can parametrize multiple runs of a test function by
@ -161,7 +161,12 @@ to generate tests with combinations of function argument values.
Calling ``funcspec.addcall(**funcargs)`` will add Calling ``funcspec.addcall(**funcargs)`` will add
a tests function call using the given dictionary a tests function call using the given dictionary
of function arguments. This addition of a call of function arguments. This addition of a call
happens during test collection. happens during test collection.
Note that you can provide a name for your test run
by passing a ``_id=...`` parameter. This name
will be shown during verbose and traceback
reporting.
Attributes of funcspec objects Attributes of funcspec objects
++++++++++++++++++++++++++++++++++++++++ ++++++++++++++++++++++++++++++++++++++++

View File

@ -25,6 +25,11 @@ def fillfuncargs(function):
except request.Error: except request.Error:
request._raiselookupfailed() request._raiselookupfailed()
class CallSpec:
def __init__(self, id, funcargs):
self.id = id
self.funcargs = funcargs
class FuncSpecs: class FuncSpecs:
def __init__(self, function, config=None, cls=None, module=None): def __init__(self, function, config=None, cls=None, module=None):
self.config = config self.config = config
@ -35,24 +40,31 @@ class FuncSpecs:
self.module = module self.module = module
self._calls = [] self._calls = []
def addcall(self, **kwargs): def addcall(self, _id=None, **kwargs):
for argname in kwargs: for argname in kwargs:
if argname[0] == "_":
raise TypeError("argument %r is not a valid keyword." % argname)
if argname not in self.funcargnames: if argname not in self.funcargnames:
raise ValueError("function %r has no funcarg %r" %( raise ValueError("function %r has no funcarg %r" %(
self.function, argname)) self.function, argname))
self._calls.append(kwargs) if _id is None:
_id = len(self._calls)
_id = str(_id)
#if _id in self._ids:
# raise ValueError("duplicate id %r" % _id)
self._calls.append(CallSpec(_id, kwargs))
class FunctionCollector(py.test.collect.Collector): class FunctionCollector(py.test.collect.Collector):
def __init__(self, name, parent, combinations): def __init__(self, name, parent, calls):
super(FunctionCollector, self).__init__(name, parent) super(FunctionCollector, self).__init__(name, parent)
self.combinations = combinations self.calls = calls
self.obj = getattr(self.parent.obj, name) self.obj = getattr(self.parent.obj, name)
def collect(self): def collect(self):
l = [] l = []
for i, funcargs in py.builtin.enumerate(self.combinations): for call in self.calls:
function = self.parent.Function(name="%s[%s]" %(self.name, i), function = self.parent.Function(name="%s[%s]" %(self.name, call.id),
parent=self, funcargs=funcargs, callobj=self.obj) parent=self, funcargs=call.funcargs, callobj=self.obj)
l.append(function) l.append(function)
return l return l

View File

@ -178,7 +178,7 @@ class PyCollectorMixin(PyobjMixin, py.test.collect.Collector):
if not funcspec._calls: if not funcspec._calls:
return self.Function(name, parent=self) return self.Function(name, parent=self)
return funcargs.FunctionCollector(name=name, return funcargs.FunctionCollector(name=name,
parent=self, combinations=funcspec._calls) parent=self, calls=funcspec._calls)
class Module(py.test.collect.File, PyCollectorMixin): class Module(py.test.collect.File, PyCollectorMixin):
def _getobj(self): def _getobj(self):

View File

@ -145,6 +145,17 @@ class TestFuncSpecs:
assert funcspec.function is func assert funcspec.function is func
assert funcspec.cls is None assert funcspec.cls is None
def test_addcall_with_id(self):
def func(arg1): pass
funcspec = funcargs.FuncSpecs(func)
py.test.raises(TypeError, """
funcspec.addcall(_xyz=10)
""")
funcspec.addcall(_id="hello", arg1=100)
call = funcspec._calls[0]
assert call.id == "hello"
assert call.funcargs == {'arg1': 100}
def test_addcall_basic(self): def test_addcall_basic(self):
def func(arg1): pass def func(arg1): pass
funcspec = funcargs.FuncSpecs(func) funcspec = funcargs.FuncSpecs(func)
@ -153,7 +164,7 @@ class TestFuncSpecs:
""") """)
funcspec.addcall(arg1=100) funcspec.addcall(arg1=100)
assert len(funcspec._calls) == 1 assert len(funcspec._calls) == 1
assert funcspec._calls[0] == {'arg1': 100} assert funcspec._calls[0].funcargs == {'arg1': 100}
def test_addcall_two(self): def test_addcall_two(self):
def func(arg1): pass def func(arg1): pass
@ -161,8 +172,8 @@ class TestFuncSpecs:
funcspec.addcall(arg1=100) funcspec.addcall(arg1=100)
funcspec.addcall(arg1=101) funcspec.addcall(arg1=101)
assert len(funcspec._calls) == 2 assert len(funcspec._calls) == 2
assert funcspec._calls[0] == {'arg1': 100} assert funcspec._calls[0].funcargs == {'arg1': 100}
assert funcspec._calls[1] == {'arg1': 101} assert funcspec._calls[1].funcargs == {'arg1': 101}
class TestGenfuncFunctional: class TestGenfuncFunctional:
def test_attributes(self, testdir): def test_attributes(self, testdir):
@ -232,11 +243,11 @@ class TestGenfuncFunctional:
class ConftestPlugin: class ConftestPlugin:
def pytest_genfunc(self, funcspec): def pytest_genfunc(self, funcspec):
assert "arg1" in funcspec.funcargnames assert "arg1" in funcspec.funcargnames
funcspec.addcall(arg1=1, arg2=2) funcspec.addcall(_id="world", arg1=1, arg2=2)
""") """)
p = testdir.makepyfile(""" p = testdir.makepyfile("""
def pytest_genfunc(funcspec): def pytest_genfunc(funcspec):
funcspec.addcall(arg1=10, arg2=10) funcspec.addcall(_id="hello", arg1=10, arg2=10)
class TestClass: class TestClass:
def test_myfunc(self, arg1, arg2): def test_myfunc(self, arg1, arg2):
@ -244,7 +255,7 @@ class TestGenfuncFunctional:
""") """)
result = testdir.runpytest("-v", p) result = testdir.runpytest("-v", p)
assert result.stdout.fnmatch_lines([ assert result.stdout.fnmatch_lines([
"*test_myfunc*0*PASS*", "*test_myfunc*hello*PASS*",
"*test_myfunc*1*FAIL*", "*test_myfunc*world*FAIL*",
"*1 failed, 1 passed*" "*1 failed, 1 passed*"
]) ])