我一直在研究一种方法,可以让鼻子中的生成器生成测试,以获得针对正在测试的特定迭代定制的描述。只要我的生成器目标方法从不尝试self
从我的生成器类访问,我就有一些有用的东西。我看到我的所有生成器目标实例都有一个公共测试类实例,而鼻子正在为生成器的每个测试运行生成一个一次性的测试类实例。这导致setUp
在鼻子创建的每个测试实例上运行,但永远不会在生成器目标绑定到的实例上运行(当然,真正的问题是我看不到如何将鼻子创建的实例绑定到生成器目标)。这是我用来尝试解决这一切的代码(是的,我知道装饰器作为可调用类可能会更好,但是鼻子,至少我拥有的版本 1.2.1,明确检查测试是否是函数或方法,因此可调用的类根本不会运行):
import inspect
def labelable_yielded_case(case):
argspec = inspect.getargspec(case)
if argspec.defaults is not None:
defaults_list = [''] * (len(argspec.args) - len(argspec.defaults)) + argspec.defaults
else:
defaults_list = [''] * len(argspec.args)
argument_defaults_list = zip(argspec.args, defaults_list)
case_wrappers = []
def add_description(wrapper_id, argument_dict):
case_wrappers[wrapper_id].description = case.__doc__.format(**argument_dict)
def case_factory(*factory_args, **factory_kwargs):
def case_wrapper_wrapper():
wrapper_id = len(case_wrappers)
def case_wrapper(*args, **kwargs):
args = factory_args + args
argument_list = []
for argument in argument_defaults_list:
argument_list.append(list(argument))
for index, value in enumerate(args):
argument_list[index][1] = value
argument_dict = dict(argument_list)
argument_dict.update(factory_kwargs)
argument_dict.update(kwargs)
add_description(wrapper_id, argument_dict)
return case(*args, **kwargs)
case_wrappers.append(case_wrapper)
case_wrapper.__name__ = case.__name__
return case_wrapper
return case_wrapper_wrapper()
return case_factory
class TestTest(object):
def __init__(self):
self.data = None
def setUp(self):
print 'setup', self
self.data = (1,2,3)
def test_all(self):
for index, value in enumerate((1,2,3)):
yield self.validate_equality(), index, value
def test_all_again(self):
for index, value in enumerate((1,2,3)):
yield self.validate_equality_again, index, value
@labelable_yielded_case
def validate_equality(self, index, value):
'''element {index} equals {value}'''
print 'test', self
assert self.data[index] == value, 'expected %d got %d' % (value, self.data[index])
def validate_equality_again(self, index, value):
print 'test', self
assert self.data[index] == value, 'expected %d got %d' % (value, self.data[index])
validate_equality_again.description = 'again'
当通过鼻子运行时,again
测试工作得很好,但是使用装饰生成器目标的测试集都失败了,因为self.data is None
(因为setUp
永远不会运行,因为TestTest
存储在闭包中的实例不是鼻子运行的实例)。我尝试让装饰器成为 的基类的实例成员,但是鼻子抛出了关于传递给未绑定的TestTest
参数太少(否)的错误。有什么办法可以使这项工作(没有砍头),还是我在不能让产量目标成为实例成员或没有为每个产量测试都贴上每个测试标签之间做出选择?self
labelable_yielded_case