我在使用函数和属性修补类时遇到了一些麻烦。
我正在使用的项目结构如下:
project
|- src
| |- logic
| | |- sub_logic
| | | | __init__.py
| | | | cache.py
| | | | manager.py
| | | __init__.py
|- test
| | test.py
我的缓存文件看起来像这样
class Cache(object):
def __init__(self, val):
self._val = val
@property
def Val(self):
return self._val
def other_function(self):
return False
经理文件看起来像这样
from cache import Cache
class Manager(object):
def __init__(self):
self._cache = Cache(20)
def do_something(self):
if self._cache.Val != 20:
raise ValueError(u"Val is not 20")
return True
def do_something_else(self):
if self._cache.other_function():
raise ValueError(u"Something is True")
我试图做的测试如下:
from unittest import TestCase
from mock import PropertyMock, patch
from logic.sub_logic.manager import Manager
from logic.sub_logic.cache import Cache
class ManagerTestCase(TestCase):
def test_01_cache(self):
manager = Manager()
self.assertEqual(manager.do_something(), True)
@patch('logic.sub_logic.manager.Cache.Val', new_callable=PropertyMock)
def test_02_cache(self, property_mock):
property_mock.return_value = 20
manager = Manager()
self.assertEqual(manager.do_something(), True)
@patch('logic.sub_logic.manager.Cache', spec=Cache)
def test_03_cache(self, cache_mock):
cache_mock.other_function.return_value = True
manager = Manager()
with self.assertRaises(ValueError):
manager.do_something_else()
@patch('logic.sub_logic.manager.Cache', spec=Cache)
def test_04_cache(self, cache_mock):
cache_mock.other_function.return_value = True
cache_mock.Val = PropertyMock()
cache_mock.Val.return_value = 20
manager = Manager()
with self.assertRaises(ValueError):
manager.do_something_else()
self.assertEqual(manager.do_something(), True)
@patch('logic.sub_logic.manager.Cache.Val', new_callable=PropertyMock)
@patch('logic.sub_logic.manager.Cache', spec=Cache)
def test_05_cache(self, cache_mock, property_mock):
cache_mock.other_function.return_value = True
property_mock.return_value = 20
manager = Manager()
with self.assertRaises(ValueError):
manager.do_something_else()
self.assertEqual(manager.do_something(), True)
@patch('logic.sub_logic.manager.Cache', spec=Cache)
def test_06_cache(self, cache_mock):
cache_mock.other_function.return_value = True
cache_mock.Val = 20
manager = Manager()
with self.assertRaises(ValueError):
manager.do_something_else()
self.assertEqual(manager.do_something(), True)
问题是test_04_cache和test_05_cache无效。 在调试测试时,提供的mock参数就像我预期的那样。但是Manager会创建一个MagicMock,其中属性Val不是PropertyMock,而是MagicMock。
我在PyCharm Debugger中检查了test_06_cache,报告了以下内容:
cache_mock.Val = {int}20
manager._cache.Val = {MagicMock}<MagicMock name='Cache().Val' id='61044848'>
我错过了什么吗?或者是不可能的?
答案 0 :(得分:4)
使用时
@patch('logic.sub_logic.manager.Cache', spec=Cache)
生成的模拟用于类。然后,您的Manager
会在__init__
中通过调用该类来创建实例。因此,您应该在mock_cache()
(注意括号)上设置属性并返回值,这是将分配给manager._cache
的“实例”,而不是“{”mock_cache
。
请注意,由于管理员不知道缓存使用@property
,您只需设置:
mock_cache().Val = 20