issue351: Add ability to specify parametrize ids as a callable, to generate custom test ids. + tests, docs

--HG--
branch : issue351
This commit is contained in:
Brianna Laugher 2014-04-17 15:08:49 -04:00
parent c47835f5ec
commit 4e35c00ab0
3 changed files with 159 additions and 15 deletions

View File

@ -771,9 +771,14 @@ class Metafunc(FuncargnamesCompatAttr):
function so that it can perform more expensive setups during the function so that it can perform more expensive setups during the
setup phase of a test rather than at collection time. setup phase of a test rather than at collection time.
:arg ids: list of string ids each corresponding to the argvalues so :arg ids: list of string ids, or a callable.
that they are part of the test id. If no ids are provided they will If strings, each is corresponding to the argvalues so that they are
be generated automatically from the argvalues. part of the test id.
If callable, it should take one argument (a single argvalue) and return
a string or return None. If None, the automatically generated id for that
argument will be used.
If no ids are provided they will be generated automatically from
the argvalues.
:arg scope: if specified it denotes the scope of the parameters. :arg scope: if specified it denotes the scope of the parameters.
The scope is used for grouping tests by parameter instances. The scope is used for grouping tests by parameter instances.
@ -813,11 +818,15 @@ class Metafunc(FuncargnamesCompatAttr):
raise ValueError("%r uses no fixture %r" %( raise ValueError("%r uses no fixture %r" %(
self.function, arg)) self.function, arg))
valtype = indirect and "params" or "funcargs" valtype = indirect and "params" or "funcargs"
idfn = None
if callable(ids):
idfn = ids
ids = None
if ids and len(ids) != len(argvalues): if ids and len(ids) != len(argvalues):
raise ValueError('%d tests specified with %d ids' %( raise ValueError('%d tests specified with %d ids' %(
len(argvalues), len(ids))) len(argvalues), len(ids)))
if not ids: if not ids:
ids = idmaker(argnames, argvalues) ids = idmaker(argnames, argvalues, idfn)
newcalls = [] newcalls = []
for callspec in self._calls or [CallSpec2(self)]: for callspec in self._calls or [CallSpec2(self)]:
for param_index, valset in enumerate(argvalues): for param_index, valset in enumerate(argvalues):
@ -865,17 +874,31 @@ class Metafunc(FuncargnamesCompatAttr):
cs.setall(funcargs, id, param) cs.setall(funcargs, id, param)
self._calls.append(cs) self._calls.append(cs)
def idmaker(argnames, argvalues):
idlist = [] def _idval(val, argname, idx, idfn):
for valindex, valset in enumerate(argvalues): if idfn:
this_id = [] try:
for nameindex, val in enumerate(valset): s = idfn(val)
if not isinstance(val, (float, int, str, bool, NoneType)): if s:
this_id.append(str(argnames[nameindex])+str(valindex)) return s
else: except Exception:
this_id.append(str(val)) pass
idlist.append("-".join(this_id)) if isinstance(val, (float, int, str, bool, NoneType)):
return idlist return str(val)
return str(argname)+str(idx)
def _idvalset(idx, valset, argnames, idfn):
this_id = [_idval(val, argname, idx, idfn)
for val, argname in zip(valset, argnames)]
return "-".join(this_id)
def idmaker(argnames, argvalues, idfn=None):
ids = [_idvalset(valindex, valset, argnames, idfn)
for valindex, valset in enumerate(argvalues)]
if len(set(ids)) < len(ids):
# the user may have provided a bad idfn which means the ids are not unique
ids = ["{}".format(i) + testid for i, testid in enumerate(ids)]
return ids
def showfixtures(config): def showfixtures(config):
from _pytest.main import wrap_session from _pytest.main import wrap_session

View File

@ -68,6 +68,81 @@ let's run the full monty::
As expected when running the full range of ``param1`` values As expected when running the full range of ``param1`` values
we'll get an error on the last one. we'll get an error on the last one.
Different options for test IDs
------------------------------------
pytest will build a string that is the test ID for each set of values in a
parametrized test. These IDs can be used with "-k" to select specific cases
to run, and they will also identify the specific case when one is failing.
Running pytest with --collect-only will show the generated IDs.
Numbers, strings, booleans and None will have their usual string representation
used in the test ID. For other objects, pytest will make a string based on
the argument name::
# contents of test_time.py
from datetime import datetime, timedelta
testdata = [(datetime(2001, 12, 12), datetime(2001, 12, 11), timedelta(1)),
(datetime(2001, 12, 11), datetime(2001, 12, 12), timedelta(-1)),
]
@pytest.mark.parametrize("a,b,expected", testdata)
def test_timedistance_v0(a, b, expected):
diff = a - b
assert diff == expected
@pytest.mark.parametrize("a,b,expected", testdata, ids=["forward", "backward"])
def test_timedistance_v1(a, b, expected):
diff = a - b
assert diff == expected
def idfn(val):
if isinstance(val, (datetime,)):
# note this wouldn't show any hours/minutes/seconds
return val.strftime('%Y%m%d')
@pytest.mark.parametrize("a,b,expected", testdata, ids=idfn)
def test_timedistance_v2(a, b, expected):
diff = a - b
assert diff == expected
In ``test_timedistance_v0``, we let pytest generate the test IDs.
In ``test_timedistance_v1``, we specified ``ids`` as a list of strings which were
used as the test IDs. These are succinct, but can be a pain to maintain.
In ``test_timedistance_v2``, we specified ``ids`` as a function that can generate a
string representation to make part of the test ID. So our ``datetime`` values use the
label generated by ``idfn``, but because we didn't generate a label for ``timedelta``
objects, they are still using the default pytest representation::
$ py.test test_time.py --collect-only
============================ test session starts =============================
platform linux2 -- Python 2.7.3 -- py-1.4.20 -- pytest-2.6.0.dev1
plugins: cache
collected 6 items
<Module 'test_time.py'>
<Function 'test_timedistance_v0[a0-b0-expected0]'>
<Function 'test_timedistance_v0[a1-b1-expected1]'>
<Function 'test_timedistance_v1[forward]'>
<Function 'test_timedistance_v1[backward]'>
<Function 'test_timedistance_v2[20011212-20011211-expected0]'>
<Function 'test_timedistance_v2[20011211-20011212-expected1]'>
============================== in 0.04 seconds ===============================
A quick port of "testscenarios" A quick port of "testscenarios"
------------------------------------ ------------------------------------

View File

@ -151,6 +151,52 @@ class TestMetafunc:
"a6-b6", "a6-b6",
"a7-b7"] "a7-b7"]
@pytest.mark.issue351
def test_idmaker_idfn(self):
from _pytest.python import idmaker
def ids(val):
if isinstance(val, Exception):
return repr(val)
result = idmaker(("a", "b"), [(10.0, IndexError()),
(20, KeyError()),
("three", [1, 2, 3]),
], idfn=ids)
assert result == ["10.0-IndexError()",
"20-KeyError()",
"three-b2",
]
@pytest.mark.issue351
def test_idmaker_idfn_unique_names(self):
from _pytest.python import idmaker
def ids(val):
return 'a'
result = idmaker(("a", "b"), [(10.0, IndexError()),
(20, KeyError()),
("three", [1, 2, 3]),
], idfn=ids)
assert result == ["0a-a",
"1a-a",
"2a-a",
]
@pytest.mark.issue351
def test_idmaker_idfn_exception(self):
from _pytest.python import idmaker
def ids(val):
raise Exception("bad code")
result = idmaker(("a", "b"), [(10.0, IndexError()),
(20, KeyError()),
("three", [1, 2, 3]),
], idfn=ids)
assert result == ["10.0-b0",
"20-b1",
"three-b2",
]
def test_addcall_and_parametrize(self): def test_addcall_and_parametrize(self):
def func(x, y): pass def func(x, y): pass
metafunc = self.Metafunc(func) metafunc = self.Metafunc(func)