20

我试图模拟一个类的属性(@property 装饰器)并遇到了这种不正确的行为:

 >>> from mock import MagicMock, PropertyMock
 >>> m = MagicMock()
 >>> type(m).p = PropertyMock(side_effect=AttributeError)
 >>> m.p
 <MagicMock name='mock.p' id='63150736'>

正确的行为是这样的:

 >>> from mock import MagicMock, PropertyMock
 >>> m = MagicMock()
 >>> type(m).p = PropertyMock(side_effect=ValueError)
 >>> m.p
Traceback (most recent call last)
[...]
ValueError

我无法理解为什么设置不同的例外会给我不同的结果。这两种情况的预期结果是应该引发异常!因此, In[4] 行应该引发AttributeError. 它不是。

有人愿意开导我吗?

附录:我要检查的属性会进行一些巧妙的检查,以查看传递的值是否正常。如果所述值不正常,它会返回 AttributeError ,因为我知道这是 Python 中的正确异常。因此,我需要检查使用该属性的代码是否成功以及是否成功。因此,使用 MagicMock 来模拟属性并引发所述异常。一个简单的例子是:

@x.setter
def x(self, value):
    if value < 0:
         raise AttributeError("Value cannot be negative!")
    self._x = value
4

2 回答 2

19

我知道这个问题很老,但我刚刚遇到了同样的问题并发现了这个问题。此外,大约两年前提交的错误报告似乎没有引起任何关注,所以我想我会分享我找到的解决方案,以防万一其他人遇到这个问题。

因此,如上所述PropertyMock不适用于AttributeErrorset as a side_effect。解决方法是创建一个简单Mockspec属性设置为空list,如下所示:

>>> from mock import Mock
>>> m = Mock(spec=[])
>>> m.p
Traceback (most recent call last)
[...]
AttributeError

文档中所述:

规范:这可以是字符串列表,也可以是充当模拟对象规范的现有对象(类或实例)。如果您传入一个对象,则通过在该对象上调用 dir 来形成一个字符串列表(不包括不受支持的魔法属性和方法)。访问不在此列表中的任何属性都会引发 AttributeError。

于 2015-04-03T07:52:47.573 回答
4

呃,拿着电话。这是否涵盖您的用例?

>>> import mock
>>> m = mock.MagicMock()
>>> m.p
<MagicMock name='mock.p' id='139756843423248'>
>>> del m.p #!
>>> m.p
Traceback (most recent call last):
     File "<stdin>", line 1, in <module>
     File "/home/ahammel/bin/python/mock-1.0.1-py2.6.egg/mock.py", line 664, in __getattr__
         raise AttributeError(name)
     AttributeError: p

我在寻找完全不同的东西时在文档中偶然发现了这一点。

于 2013-11-27T17:41:32.080 回答