从上下文管理器创建的对象模拟功能

时间:2019-06-06 13:35:09

标签: python-2.7 mocking python-mock

我要尝试为其编写以下模块的模块。

import myModuleWithCtxMgr

def myFunc(arg1):
    with myModuleWithCtxMgr.ctxMgr() as ctxMgr:
        result = ctxMgr.someFunc()

    if result:
        return True, result
    return False, None

我正在进行的单元测试看起来像这样。

import mock
import unittest
import myModule as myModule

class MyUnitTests(unittest.TestCase):

    @mock.patch("myModuleWithCtxMgr.ctxMgr")
    def testMyFunc(self, mockFunc):
        mockReturn = mock.MagicMock()
        mockReturn.someFunc = mock.Mock(return_value="val")
        mockFunc.return_value = mockReturn
        result = myModule.myFunc("arg")

        self.assertEqual(result, (True, "val"))

测试失败,因为result [0] = magicMock()而不是我配置的返回值。

我尝试了一些不同的测试变体,但似乎无法模拟ctxMgr.someFunc()的返回值。有人知道我该怎么做吗?

谢谢!

1 个答案:

答案 0 :(得分:1)

错误提示:

First differing element 1:
<MagicMock name='ctxMgr().__enter__().someFunc()' id='139943278730000'>
'val'

- (True, <MagicMock name='ctxMgr().__enter__().someFunc()' id='139943278730000'>)
+ (True, 'val')

错误包含模拟名称,该名称准确地显示了您需要模拟的内容。请注意,__enter__对应于Context Manager protocol

这对我有用:

class MyUnitTests(unittest.TestCase):

    @mock.patch("myModuleWithCtxMgr.ctxMgr")
    def testMyFunc(self, mockCtxMgr):
        mockCtxMgr().__enter__().someFunc.return_value = "val"
        result = myModule.myFunc("arg")

        self.assertEqual(result, (True, "val"))

请注意,每个实例都是一个单独的MagicMock实例,您可以对其进行配置:

  • mockCtxMgr
  • mockCtxMgr()
  • mockCtxMgr().__enter__
  • mockCtxMgr().__enter__()
  • mockCtxMgr().__enter__().someFunc

MagicMocks是懒惰的,但是具有标识,因此您可以通过这种方式对其进行配置,并且可以正常工作。