我希望模拟一个符合以下要求的课程:
__init__()
方法autospec=True
,因此该类的API将严格检查调用简化的班级样本:
class MyClass():
id = 0
def __init__(self, x=0.0, y=1.0):
self.x = x
self.y = y
self.id = MyClass._id
MyClass.id +=1
def calc_x_times_y(self):
return self.x*self.y
def calc_x_div_y(self, raise_if_y_not_zero=True):
try:
return self.x/self.y
except ZeroDivisionError:
if raise_if_y_not_zero:
raise ZeroDivisionError
else:
return float('nan')
就属性而言,我需要使mock对象表现为原始对象:
x,y
属性
但模拟方法调用应该被mock拦截,并且其调用签名已经过验证最好的方法是什么?
修改
我已经尝试过几种方法,包括继承Mock
类,使用attach_mock()
和mock_add_spec()
,但总是遇到一些死胡同。
我正在使用标准mock库。
答案 0 :(得分:2)
由于没有答案,我会发布对我有用的内容(不一定是最好的方法,但是这里有):
我创建了一个模拟工厂,它创建了一个Mock()
对象,使用here描述的语法设置其id
属性,并返回对象:
class MyClassMockFactory():
_id = 0
def get_mock_object(self, *args,**kwargs):
mock = Mock(MyClass, autospec = True)
self._attach_mock_property(mock , 'x', kwargs['x'])
self._attach_mock_property(mock , 'y', kwargs['y'])
self._attach_mock_property(mock , 'id', MyClassMockFactory._id)
MyClassMockFactory._id += 1
return mock
def _attach_mock_property(self, mock_object, name, value):
p = PropertyMock(return_value=value)
setattr(type(mock_object), name, p)
现在,我可以为我的测试修补MyClass()
构造函数:
class TestMyClass(TestCase):
mock_factory = MyClassMockFactory()
@patch('MyClass',side_effect=mock_factory.get_mock_object)
test_my_class(self,*args):
obj0 = MyClass()
obj1 = MyClass(1.0,2.2)
obj0.calc_x_times_y()
# Assertions
obj0.calc_x_times_y.assert_called_once_with()
self.assertEqaul(obj0.id, 0)
self.assertEqaul(obj1.id, 1)
答案 1 :(得分:0)
很抱歉找到一篇旧帖子,但是让你能够完全按照自己的意愿去做的事情就是修补calc_x_times_y
和calc_x_div_y
并在那里设置autospec=True
,反对嘲笑整个班级的创作。
类似的东西:
@patch('MyClass.calc_x_times_y')
@patch('MyClass.calc_x_div_y')
test_foo(patched_div, patched_times):
my_class = MyClass() #using real class to define attributes
# ...rest of test