Python返回MagicMock对象而不是return_value

Meh*_*omi 57 python python-unittest magicmock

我有一个python文件a.py,其中包含两个类AB.

class A(object):
    def method_a(self):
        return "Class A method a"

class B(object):
    def method_b(self):
        a = A()
        print a.method_a()
Run Code Online (Sandbox Code Playgroud)

我想通过嘲笑method_b在课堂上B进行单元测试A.以下是testa.py用于此目的的文件内容:

import unittest
import mock
import a


class TestB(unittest.TestCase):

    @mock.patch('a.A')
    def test_method_b(self, mock_a):
        mock_a.method_a.return_value = 'Mocked A'
        b = a.B()
        b.method_b()


if __name__ == '__main__':
    unittest.main()
Run Code Online (Sandbox Code Playgroud)

我希望得到Mocked A输出.但我得到的是:

<MagicMock name='A().method_a()' id='4326621392'>
Run Code Online (Sandbox Code Playgroud)

我哪里做错了?

jon*_*rpe 60

当你@mock.patch('a.A'),你正在替换A测试代码中的类mock_a.

B.method_b你接盘a = A(),这是现在a = mock_a()-即areturn_valuemock_a.由于您没有指定此值,因此它是常规值MagicMock; 这也没有配置,因此MagicMock在调用方法时会得到默认响应(又是另一个).

相反,你要配置return_valuemock_a有适当的方法,你可以做两种:

mock_a().method_a.return_value = 'Mocked A' 
    # ^ note parentheses
Run Code Online (Sandbox Code Playgroud)

或者,或许更明确:

mock_a.return_value.method_a.return_value = 'Mocked A'
Run Code Online (Sandbox Code Playgroud)

您的代码将在这种情况下工作a = A(分配类,而不是创建实例),因为那样a.method_a()会触发您的mock方法.

  • 惊人的。你让我今天一整天都感觉很好。 (11认同)
  • 第一个解决方案“mock_a().method_a.return_value”以这样的方式调用“mock_a”:即使被测试的代码从未调用“mock_a”,“mock_a.assert_used()”也会变为 true。第二个解决方案“mock_a.return_value.method_a.return_value”不会以这种方式调用“mock_a”(“mock_a.assert_used()”仍然为 false)。因此,我更喜欢第二种解决方案。 (4认同)
  • 这个解释比任何 python 文档都要好得多。谢谢你! (3认同)
  • 如果你问它工作的原因......`mock_a()`返回一个单例 (2认同)
  • 我已经设置了“return_value”并且“mock”无论如何都会返回。这些模拟并不容易正确设置 (2认同)