Python模块pytestmock如何在一个类调用另一个类实例时测试方法调用参数?

2024-04-26 20:48:35 发布

您现在位置:Python中文网/ 问答频道 /正文

几天前,我开始使用pytest学习单元测试,并对pytest模拟插件(https://github.com/pytest-dev/pytest-mock)感兴趣

我已经能够编写相当多的单元测试来测试我的代码,但现在我想测试我的部分代码如何与其他对象交互。假设我有类B,这是我感兴趣测试的类,它有一个方法将调用类a中的方法,我想断言,当类B调用类a时,方法调用是使用预期参数进行的

我总结了一些能够完成任务的黑客示例(见下面的链接),但正如您所见,这可能不是正确的方法。所以我的问题是,如何通过使用pythonmock或pytestmock模块正确地处理这个问题?不基于pytest模拟插件的答案也很受欢迎

在下面的代码中,我不喜欢断言,因为我更喜欢使用现有的pytest mock“assert_called_once_with(…)”-方法,但我就是无法让它工作。mock对象中提供了所有信息,但我不明白在这种情况下如何正确使用pytest mock API

def test_how_class_b_interact_with_class_a(mocker):
    class A(object):
        def further_process_nbr(self, nbr):
            pass # don't care for the sake of this example

    class B(object):
        def __init__(self):
            self.a = A()

        def process_nbr(self, nbr):
            nbr += 2 # do something fancy with number and then have it further processed by object a
            return self.a.further_process_nbr(nbr)

    nbr = 4
    expected = 6

    # Prepare
    mock_a = mocker.patch.object(A, 'further_process_nbr', autospec=True)

    # Exercise
    B().process_nbr(nbr)

    # Assert
    assert mock_a.call_args_list[0].args[1] == expected # This is working, but not exactly a nice way
    # mock_a.assert_called_once_with(expected) something like this is basically what I would like to do

Tags: 方法代码selfobjectpytestdefwithassert
1条回答
网友
1楼 · 发布于 2024-04-26 20:48:35

实际上,您看到的结果是正确的,具体如下:

assert (<a.A object at 0x000001628C6878B0>, 2) == (2,)

在元组的第一个元素中看到的是self参数,因为您在类中模拟了该方法,所以它也接收self

要检查assert_调用了_once_,您可以使用mocker.ANY,它将接受任何内容:

mock_a.assert_called_once_with(mocker.ANY, nbr)

相关问题 更多 >