Python Mock之初探

Mock是什么

Mock这个词在英语中是模拟的意思,因此我们可以猜测出这个库的主要功能是模拟一些东西。准确的说,Mock是Python中一个用于支持单元测试的库,它的主要功能是使用mock对象替代掉指定的Python对象,以达到模拟对象的行为。

试想如下应用场景:
假设你开发的项目叫a,里面包含了一个模块b,模块b中的一个函数c(也就是a.b.c)在工作的时候需要调用发送请求给特定的服务器来得到一个JSON返回值,然后根据这个返回值来做处理。如果要为a.b.c函数写一个单元测试,该如何做?

首先想到的简单的方法是搭建一个服务器用来跟用户交互,不过这可能存在各种各样的现实问题,比如搭建服务器的成本(如时间/金钱等)很高,又或者测试服务器无法返回所有可能的值或者需要大量的工作才能达到这个目的。那么此时就可以利用mock替换掉测试服务器与用户的交互过程,这样返回值也会由mock对象来决定,而不需要测试服务器的参与。

Mock的安装及导入

# python 3.3以前的版本中,需要额外安装mock模块
pip install mock
import mock
# 从python 3.3开始,mock被集成到unittest模块中,可以直接import
from unit test import mock

Mock对象

基本用法

Mock对象是mock模块中最重要的概念。Mock对象就是mock模块中的一个类的实例,这个类的实例可以用来替换其他的Python对象,来达到模拟的效果。Mock类的定义如下:

class Mock(spec=None, side_effect=None, return_value=DEFAULT, wraps=None, name=None, spec_set=None, **kwargs)

mock一个对象的一般步骤如下:

  1. 确定替换对象,是一个函数/类/类的实例;
  2. 建立mock对象(Mock实例化),并且设置这个对象的行为;
  3. 用建立的mock对象代替步骤1确定要替换的对象;
  4. 编写UT;

举个栗子:有一个简单的客户端实现,用来访问一个url,当访问正常时,需要返回状态码200,否则返回状态码404,客户端实现代码如下:

#!/usr/bin/env python3
# -*- coding: utf-8 -*-
# client.py

import request
def send_request(url):
    r = request.get(url)
    return r.status_code

def visit_google():
    return send_request('www.google.com')

外部模块访问谷歌,下面我们使用mock对象在UT中分别测试访问正常和不正常的情况。

#!/usr/bin/env python3
# -*- coding: utf-8 -*-

import client, mock, unittest
class TestClient(unittest.TestCase):

    def test_success_request(self):
        success_send = mock.Mock(return_value='200')
        client.send_request = success_request
        self.assertEqual(client.visit_google(), '200')

 def test_fail_request(self):
        fail_send = mock.Mock(return_value='404')
        client.send_request = fail_request
        self.assertEqual(client.visit_google(), '404')

mock用法拓展

class Mock参数

下面仅列出几个常用重要的参数:

  • name: 这个是用来命名一个mock对象,只是起到标识作用,当你print一个mock对象的时候,可以看到它的name。
  • return_value: mock对象被调用返回的值,默认值是一个mock对象。
  • side_effect: 这个参数指向一个可调用对象,一般就是函数。当mock对象被调用时,如果该函数返回值不是DEFAULT时,那么以该函数的返回值作为mock对象调用的返回值。

其他的参数可参考官方文档: http://www.voidspace.org.uk/python/mock/mock.html

mock对象的自动创建

当访问一个mock对象中不存在的属性时,mock会自动建立一个子mock对象,并且把正在访问的属性指向它,这个功能对于实现多级属性的mock很方便。

对方法调用进行检查

mock对象有一些方法可以用来检查该对象是否被调用过、被调用时的参数如何、被调用了几次等。实现这些功能可以调用mock对象的方法,具体的可以查看mock官方文档

patch和patch.object

patch

语法

from mock import patch
patch(target, new=DEFAULT, spec=None, create=False, spec_set=None, autospec=None, new_callable=None, **kwargs)

具体参数的讲解可参见patch官方文档

运用

一般地patch有三种用法:函数装饰器,类装饰器,上下文管理器,到这初学者可能会一脸懵逼,话不多说直接上代码。

先写一个公共代码文件common.py方便上面所说三种用法的调用。

class something(object):
    def do_something():
        return 'Done'

测试patch三种用法的代码test_patch.py如下:

#!/usr/bin/env python3
# -*- coding: utf-8 -*-

import common
from mock import patch

def new_func():
    print('welcome')

# 函数装饰器
@patch('common.something.do_something')
def test_patch(do_sth):
    do_sth.return_value = 'Not Done'
    print(common.something.do_something())

if __name__ == '__main__':
    patcher = patch('common.something', first='one', second='two') #类装饰器
    mock_thing = patcher.start()
    print(mock_thing.first)
    print(mock_thing.second)
    patcher.stop()

    test_patch()

    # contextmanager
    with patch('common.something.do_something') as mock_func: 
        mock_func.side_effect = [1,2,3]
        print(common.something.do_something())
        print(common.something.do_something())
        print(common.something.do_something())

    patcher = patch('__main__.new_func') # 函数装饰器
    patcher.start()
    new_func()
    patcher.stop()
    new_func()

#输出
#one
#two
#Not Done
#1
#2
#3
#welcome

patch.object

语法定义

#patch the named member (attribute) on an object (target) with a mock object
patch.object(target, attribute, new=DEFAULT, spec=None, create=False, spec_set=None, autospec=None, new_callable=None, **kwargs)

运用

从上面的语法定义不难看出,patch.objec跟patch的用法基本上是一样的,只是使用格式略微不同,举个栗子,使用patch.object来替换上面的test_patch。

# -*- coding: utf-8 -*-

import common
from mock import patch

@patch.object(common.something, 'do_something')
def test_patch(do_sth):
    do_sth.return_value = 'Not Done'
    print(common.something.do_something())

总的来说,patch和patch.object通过重新定义函数或类的行为来掩藏原来函数或类的行为,在一些应用场景比如UT测试会有神奇的效果。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 概要 64学时 3.5学分 章节安排 电子商务网站概况 HTML5+CSS3 JavaScript Node 电子...
    阿啊阿吖丁阅读 9,342评论 0 3
  • 〇、前言 本文共108张图,流量党请慎重! 历时1个半月,我把自己学习Python基础知识的框架详细梳理了一遍。 ...
    Raxxie阅读 19,084评论 17 410
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,993评论 19 139
  • 朗读训练
    夏天0608阅读 260评论 0 0
  • 我们每天有二十四个小时, 每天工 作八个小时, 用来养活自己和家人。 每天睡眠和休息八个小时, 用来保障身体的健康...
    再凑热闹阅读 136评论 0 0