pytest-mock:模拟一个抽象 class

pytest-mock: Mock an abstract class

我正在尝试使以下代码工作:

from pytest_mock import mocker

class TestClass(mocker):
  def setup_method(self):
    self.some_mock = SomeAbstractClass()
    self.testsubject = ClassThatIsBeingTested(self.some_mock)

  def test_1(self):
    mocker.patch(self.some_mock, 'some_function', return_value=5)
    assert 5 == self.testsubject.function_that_internally_uses_the_mock()

但是我得到一个 TypeError 试图实例化一个抽象 class。

如何模拟 SomeAbstractClass

您可以使用 Python 2.7) 的 unittest.mock module (also provided as the mock 包来创建模拟实例,而不是尝试修补真实实例:

import sys

if sys.version_info[0] == 2:
    from mock import Mock
else:
    from unittest.mock import Mock

# ...

some_mock = Mock(spec=SomeAbstractClass)
some_mock.some_function.return_value = 5

some_mock.some_function()
# => 5

some_mock.non_existent_function()
# => AttributeError: Mock object has no attribute 'non_existent_function'

这适用于创建任何 class 的模拟实例——而不仅仅是抽象实例。

使用 pytest-mockunittest.mock 您可以使用 mocker.patch.multiple 方法覆盖 __abstractmethods__ 属性。通过这样做,您将能够创建抽象 class 的实例并测试它的 non-abstract 方法。

使用 pytest 和 pytest-mock 的示例:

def test_something(mocker):
    mocker.patch.multiple(ExampleClass, __abstractmethods__=set())
    instance = ExampleClass()
    ...