我正在尝试运行此测试:self.assertRaises(AttributeError, branch[0].childrennodes)
,branch[0
]没有属性childrennodes
,所以它应该抛出AttributeError
, assertRaises
应该捕获,但是当我运行测试时,测试失败,因为它正在抛出AttributeError
。
Traceback (most recent call last):
File "/home/tttt/../tttt/tests.py", line 504, in test_get_categories_branch
self.assertRaises(AttributeError, branch[0].children_nodes)
AttributeError: 'Category' object has no attribute 'children_nodes'
有什么想法吗?
答案 0 :(得分:52)
当测试运行时,在调用self.assertRaises之前,Python需要找到所有方法参数的值。在这样做时,它会评估branch[0].children_nodes
,这会引发AttributeError。由于我们尚未调用assertRaises,因此未捕获此异常,导致测试失败。
解决方案是将branch[0].children_nodes
包装在函数或lambda中:
self.assertRaises(AttributeError, lambda: branch[0].children_nodes)
assertRaises也可以用作上下文管理器(从Python 2.7开始,或者在PyPI包'unittest2'中):
with self.assertRaises(AttributeError):
branch[0].children_nodes
# etc
这很好,因为它可以在测试过程中用于任意代码块,而不是仅仅为了定义它所适用的代码块而创建一个新函数。
如果需要,它可以让您访问引发的异常以进行进一步处理:
with self.assertRaises(AttributeError) as cm:
branch[0].children_nodes
self.assertEquals(cm.exception.special_attribute, 123)
答案 1 :(得分:20)
我认为因为断言加注只接受一个可调用的。它会评估调用是否会引发异常,而不是语句本身是否会发生异常。
self.assertRaises(AttributeError, getattr, branch[0], "childrennodes")
应该有用。
编辑:
正如THC4k正确地说它在收集时收集语句然后会出错,而不是在测试时。
这也是我喜欢鼻子的原因,它有一个装饰器(加注),对于这类测试来说是有用和清晰的。
@raises(AttributeError)
def test_1(self)
branch[0].childrennodes
答案 2 :(得分:2)
pytest也有类似的装饰者:
from pytest import raises
def test_raising():
with raises(AttributeError):
branch[0].childrennodes