具有函数和属性的 Python 模拟类

Kle*_*nsE 4 python unit-testing mocking python-2.7

我在使用函数和属性修补类时遇到一些问题。

我正在使用的项目结构如下:

project
|- src
|  |- logic
|  |  |- sub_logic
|  |  |  | __init__.py
|  |  |  | cache.py
|  |  |  | manager.py
|  |  | __init__.py  
|- test
|  | test.py  
Run Code Online (Sandbox Code Playgroud)

我的缓存文件看起来像这样

class Cache(object):
    def __init__(self, val):
        self._val = val

    @property
    def Val(self):
        return self._val

    def other_function(self):
        return False
Run Code Online (Sandbox Code Playgroud)

管理器文件看起来像这样

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")
Run Code Online (Sandbox Code Playgroud)

我尝试进行的测试如下:

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)
Run Code Online (Sandbox Code Playgroud)

问题是 test_04_cache 和 test_05_cache 不起作用。调试测试时,提供的模拟参数符合我的预期。但是 Manager 创建了一个 MagicMock,其中属性 Val 不是 PropertyMock,而是 MagicMock。

我在 PyCharm 调试器中检查了 test_06_cache,报告如下:

  • cache_mock.Val = {int}20
  • manager._cache.Val = {MagicMock}<MagicMock name='Cache().Val' id='61044848'>

我错过了什么吗?还是不可能?

jon*_*rpe 6

当你使用

@patch('logic.sub_logic.manager.Cache', spec=Cache)
Run Code Online (Sandbox Code Playgroud)

生成的模拟是针对班级的。然后,您通过调用该类Manager来创建一个实例,在. 因此,您应该在(注意括号)上设置属性和返回值,这是将分配给 的“实例” ,而不是在“类”上。__init__mock_cache()manager._cachemock_cache

请注意,由于管理器不知道缓存正在使用 a @property,因此您可以设置:

mock_cache().Val = 20
Run Code Online (Sandbox Code Playgroud)