Python unittest mock:是否可以在测试时模拟方法的默认参数的值?

nsf*_*n55 12 python mocking python-unittest

我有一个接受默认参数的方法:

def build_url(endpoint, host=settings.DEFAULT_HOST):
    return '{}{}'.format(host, endpoint)
Run Code Online (Sandbox Code Playgroud)

我有一个运行此方法的测试用例:

class BuildUrlTestCase(TestCase):
    def test_build_url(self):
        """ If host and endpoint are supplied result should be 'host/endpoint' """

        result = build_url('/end', 'host')
        expected = 'host/end'

        self.assertEqual(result,expected)

     @patch('myapp.settings')
     def test_build_url_with_default(self, mock_settings):
        """ If only endpoint is supplied should default to settings"""
        mock_settings.DEFAULT_HOST = 'domain'

        result = build_url('/end')
        expected = 'domain/end'

        self.assertEqual(result,expected)
Run Code Online (Sandbox Code Playgroud)

如果我删除调试点build_url并检查此属性settings.DEFAULT_HOST将返回模拟值.但是测试继续失败,断言指示host从我的实际值中分配值settings.py.我知道这是因为host关键字参数是在导入时设置的,我的模拟不被考虑.

调试器

(Pdb) settings
<MagicMock name='settings' id='85761744'>                                                                                                                                                                                               
(Pdb) settings.DEFAULT_HOST
'domain'
(Pdb) host
'host-from-settings.com'                                                                                                                                                 
Run Code Online (Sandbox Code Playgroud)

有没有办法在测试时覆盖此值,以便我可以使用模拟settings对象运行默认路径?

che*_*ner 12

函数func_defaults定义时,函数将其参数默认值存储在属性中,因此您可以对其进行修补.就像是

def test_build_url(self):
    """ If only endpoint is supplied should default to settings"""

    # Use `func_defaults` in Python2.x and `__defaults__` in Python3.x.
    with patch.object(build_url, 'func_defaults', ('domain',)):
      result = build_url('/end')
      expected = 'domain/end'

    self.assertEqual(result,expected)
Run Code Online (Sandbox Code Playgroud)

我使用patch.object上下文管理器而不是装饰器来避免将不必要的补丁对象作为参数传递给test_build_url.


And*_*rew 8

另一种方法是:使用 functools.partial 提供您想要的“默认”参数。从技术上讲,这与覆盖它们不同。被调用者看到一个显式的参数,但调用者不必提供它。大多数时候这已经足够接近了,并且在上下文管理器退出后它会做正确的事情:

# mymodule.py
def myfunction(arg=17):
    return arg

# test_mymodule.py
from functools import partial
from mock import patch

import mymodule

class TestMyModule(TestCase):
    def test_myfunc(self):
        patched = partial(mymodule.myfunction, arg=23)
        with patch('mymodule.myfunction', patched):
            self.assertEqual(23, mymodule.myfunction())  # Passes; default overridden
        self.assertEqual(17, mymodule.myfunction()) # Also passes; original default restored
Run Code Online (Sandbox Code Playgroud)

我在测试时使用它来覆盖默认配置文件位置。感谢,我从 Danilo Bargen那里得到了这个想法


gae*_*fan 5

我对这个问题应用了另一个答案,但是在上下文管理器之后,修补的功能与以前不一样。

我的修补函数如下所示:

def f(foo=True):
    pass
Run Code Online (Sandbox Code Playgroud)

在我的测试中,我这样做了:

with patch.object(f, 'func_defaults', (False,)):
Run Code Online (Sandbox Code Playgroud)

f上下文管理器之后(不在)调用时,默认值完全消失而不是回到以前的值。f不带参数的调用给出了错误TypeError: f() takes exactly 1 argument (0 given)

相反,我只是在测试之前这样做了:

f.func_defaults = (False,)
Run Code Online (Sandbox Code Playgroud)

这在我的测试之后:

f.func_defaults = (True,)
Run Code Online (Sandbox Code Playgroud)