模拟方法的返回值不起作用

tim*_*78h 7 mocking python-3.x python-mock

在测试该create_response方法时,我似乎无法模拟该方法的返回值get_external_response

/foo/响应

from abc import ABCMeta, abstractmethod


def create_response(url, type):
    query = create_query(url, type)
    external_response = get_external_response(query)  <-- what I want to mock
    return external_response

def create_query(url, type):
    cc = MyFactory
    return cc.get_concrete_class(url, type)

def get_external_response(cc):
    return cc.make_query()


class MyAbstractClass(metaclass=ABCMeta):
    def __init__(self, url, type):
        self.url = url
        self.type = type
        self.query = self.make_query()

    @abstractmethod
    def make_query(self):
        pass

class MyFactory:
    @staticmethod
    def get_concrete_class(url, type):
        if type == 'A':
            return MyClass(url, type)
        else:
            print("not valid type")

class MyClass(MyAbstractClass):
    def __init__(self, url, type):
        super().__init__(url, type)

    def make_query(self):
        return self.url + self.type


if __name__ == '__main__':
    result = create_response('www.stackoverflow.com', 'A')
    print(result)
Run Code Online (Sandbox Code Playgroud)

如果我运行上面的代码,我会得到预期的www.stackoverflow.comA.

但是如果尝试模拟 的返回值get_external_response,它似乎没有做任何事情:它仍然返回www.stackoverflow.comA并且下面的断言失败。

/foo/test_response

from foo.response import create_response
import pytest
from unittest import mock


def test_create_response():
    mock_external_response = mock.Mock()
    mock_external_response.create_flask_response.return_value = 'www'

    result = create_response('www.stackoverflow.com', 'A')
    assert result == 'www'
Run Code Online (Sandbox Code Playgroud)

我不明白为什么没有设置返回值,因为当create_response调用 时,它最终会达到调用 的点create_flask_response,如果我没有记错的话,www鉴于我已经模拟了它,应该返回它。

有人可以解释我做错了什么吗?

小智 4

我注意到您正在函数内部创建一个 Mock 对象,但实际上并未使用该 Mock。看来您需要修补用于使用 Mock 的函数。

/foo/test_response

@mock.patch('foo.response.get_external_response')
def test_create_response(mock_get_external_reponse):
    mock_get_external_response.return_value = 'www'         # This is a Mock object, but will be used as a patch over the actual function where it is used

    result = create_response('www.stackoverflow.com', 'A')
    assert result == 'www'
Run Code Online (Sandbox Code Playgroud)

为了方便起见,快速链接到相关文档部分:

模拟和修补简介: https://docs.python.org/3/library/unittest.mock.html#quick-guide

专门在这里修补: https://docs.python.org/3/library/unittest.mock.html#unittest.mock.patch