python接口自动化—mock模块的基本使用


一、mock简介

mock原是python的第三方库

python3以后mock模块已经整合到了unittest测试框架中,不用再单独安装

Mock这个词在英语中有模拟的意思,因此我们可以猜测出这个库的主要功能是模拟一些东西

准确的说,Mock是Python中一个用于支持单元测试的库,它的主要功能是使用mock对象替代掉指定的Python对象,以达到模拟对象的行为

既然mock已经被整合到了unittest单元测试框架中,可想而知mock的目的就是为了让我们更好的进行测试

二、mock作用

1. 解决依赖问题:当我们测试一个接口或者功能模块的时候,如果这个接口或者功能模块依赖其他接口或其他模块,那么如果所依赖的接口或功能模块未开发完毕,那么我们就可以

使用mock模拟被依赖接口,完成目标接口的测试

2. 单元测试:如果某个功能未开发完成,我们又要进行测试用例的代码编写,我们也可以先模拟这个功能进行测试

3. 模拟复杂业务的接口:实际工作中如果我们在测试一个接口功能时,如果这个接口依赖一个非常复杂的接口业务,那么我们完全可以使用mock来模拟这个复杂的业务接口,其实

这个和解决接口依赖是一样的原理

4.前后端联调:如果你是一个前端页面开发,现在需要开发一个功能:根据后台返回的状态展示不同的页面,那么你就需要调用后台的接口,但是后台接口还未开发完成,是不是你

就停止这部分工作呢?答案是否定的,你完全可以借助mock来模拟后台这个接口返回你想要的数据

三、mock安装

python 3 的mock模块已经被整合到了unittest框架中,所以你使用的时候只需要在文件开头from unittest import mock 导入即可

如果你使用的是python2 那么你需要执行pip install mock安装后再 import mock即可

四、mock实例

一个未开发完成的功能如何测试?

假如们现在有一个实现两个数相加的功能需要编写测试用例,但是由于开发进度缓慢,只搭两个简单的框架,并没有内部实现

"""

------------------------------------

@Time : 2020/7/21 14:09

@Auth : 梵音

@File : ClassFunc.py

@IDE  : PyCharm

@Motto: Real warriors,dare to face the bleak warning,dare to face the incisive error!

@QQ  : 829792258@qq.com

@GROUP: 829792258

------------------------------------

"""

import unittest

from unittest import mock

class SubClass(object):

    def add(self, a, b):

        """两个数相加"""

        pass

class TestSub(unittest.TestCase):

  """测试两个数相加用例"""

    def test_sub(self):

        sub = SubClass()  # 初始化被测函数类实例

        sub.add = mock.Mock(return_value=10)  # mock add方法 返回10

        result = sub.add(5, 5)  # 调用被测函数

        self.assertEqual(result, 10)  # 断言实际结果和预期结果

if __name__ == '__main__':

    unittest.main()

测试结果

----------------------------------------------------------------------

Ran 1 test in 0.000s

OK

Process finished with exit code

测试结果显示,测试用例执行已经通过

实际上mock模拟add方法的原理是 使用相同的对象方法接收mock的对象(使用sub.add接收),那么当mock对象被调用时(sub.add())就会返回return_value参数对应的数据

你可以做一个实验,把用例中的add改成别的名字也一样可以测试通过

我们用例编写完了,而且开发既然也把功能开发完了,既然真实的功能已经可以测试了,那么我们怎么在上面用例的基础上直接测试真实功能呢?

完整的功能如何测试?

我们把用例的代码稍做修改

"""

------------------------------------

@Time : 2020/7/21 14:09

@Auth : 梵音

@File : ClassFunc.py

@IDE  : PyCharm

@Motto: Real warriors,dare to face the bleak warning,dare to face the incisive error!

@QQ  : 829792258@qq.com

@GROUP: 829792258

------------------------------------

"""

import unittest

from unittest import mock

class SubClass(object):

    def add(self, a, b):

        """两个数相加"""

        return a + b

class TestSub(unittest.TestCase):

    """测试两个数相加"""

    def test_sub(self):

        sub = SubClass()  # 初始化被测函数类实例

        sub.add = mock.Mock(return_value=10, side_effect=sub.add)  # 传递side_effect关键字参数, 会覆盖return_value参数值, 使用真实的add方法测试

        result = sub.add(5, 11)  # 真正的调用被测函数

        self.assertEqual(result, 16)  # 断言实际结果和预期结果

if __name__ == '__main__':

    unittest.main()

side_effect参数

代码中我们给Mock方法添加了另一个关键字参数side_effect = sub.add, 这个参数和return_value 正好相反,当传递这个参数的时候return_value 参数就会失效

而side_effect生效,这里我给的参数值是sub.add 相当于add方法的地址,那么当调用add方法时就会真实的使用add方法,也就达到了我们测试实际的add 方法。

你也可以理解为当传递了side_effect参数且值为被测方法地址时,mock就不会起作用

side_effect接收的是一个可迭代序列,当传递多个值时,那么每次调用mock时会返回不同的值

mock_obj = mock.Mock(side_effect= [1,2,3])

print(mock_obj())

print(mock_obj())

print(mock_obj())

print(mock_obj())

输出

Traceback (most recent call last):

1

  File "D:/MyThreading/mymock.py", line 37, in <module>

2

    print(mock_obj())

3

  File "C:\Python36\lib\unittest\mock.py", line 939, in __call__

    return _mock_self._mock_call(*args, **kwargs)

  File "C:\Python36\lib\unittest\mock.py", line 998, in _mock_call

    result = next(effect)

StopIteration

Process finished with exit code 1

当所有值被取完后就会报错(这个地方有点类似生成器的原理)

存在依赖关系的功能如何测试?

假设有这样一个场景:我们要测试一个支付接口但是这个支付接口又依赖一个第三方支付接口,那么第三方支付接口我们暂时没有权限使用,那么我们该如何测试我们自己这个接口呢?

看下面的实例

假设第三方接口和我们自己的支付接口如下

"""

------------------------------------

@Time :2020/7/21 15:09

@Auth : 梵音

@File : PayMent.py

@IDE  : PyCharm

@Motto: Real warriors,dare to face the bleak warning,dare to face the incisive error!

@QQ  : 829792258@qq.com

@GROUP: 829792258

------------------------------------

"""

import requests

class PayApi(object):

    @staticmethod

    def auth(card, amount):

        """

        第三方支付接口

        :param card: 卡号

        :param amount: 支付金额

        :return:

        """

        pay_url = "http://www.zhifubao.com"  # 第三方支付接口地址

        data = {"card": card, "amount": amount}

        response = requests.post(pay_url, data=data)  # 请求第三方支付接口

        return response  # 返回状态码

    def pay(self, user_id, card, amount):

        """

        我们自己的支付接口

        :param user_id: 用户id

        :param card: 卡号

        :param amount: 支付金额

        :return:

        """

     # 调用第三方支付接口

        response = self.auth(card, amount)

        try:

            if response['status_code'] == '200':

                print('用户{}支付金额{}成功'.format(user_id, amount))

                return '支付成功'

            elif response['status_code'] == '500':

                print('用户{}支付失败, 金额不变'.format(user_id))

                return '支付失败'

            else:

                return '未知错误'

        except Exception:

            return "Error, 服务器异常!"

if __name__ == '__main__':

    pass

很明显第三方支付接口是无法访问的,因为接口的地址是我DIY的,为了模拟实际中我们无法使用的第三方支付接口

编写测试用例

"""

------------------------------------

@Time : 2020/7/21 15:22

@Auth : 梵音

@File : testpay.py

@IDE  : PyCharm

@Motto: Real warriors,dare to face the bleak warning,dare to face the incisive error!

@QQ  : 829792258@qq.com

@GROUP: 829792258

------------------------------------

"""

import unittest

from unittest import mock

from payment.PayMent import PayApi

class TestPayApi(unittest.TestCase):

    def test_success(self):

        pay = PayApi()

        pay.auth = mock.Mock(return_value={'status_code':'200'})

        status = pay.pay('1000', '12345', '10000')

        self.assertEqual(status, '支付成功')

    def test_fail(self):

        pay = PayApi()

        pay.auth = mock.Mock(return_value={'status_code':'500'})

        status = pay.pay('1000', '12345', '10000')

        self.assertEqual(status, '支付失败')

    def test_error(self):

        pay = PayApi()

        pay.auth = mock.Mock(return_value={'status_code':'300'})

        status = pay.pay('1000', '12345', '10000')

        self.assertEqual(status, '未知错误')

    def test_exception(self):

        pay = PayApi()

        pay.auth = mock.Mock(return_value='200')

        status = pay.pay('1000', '12345', '10000')

        self.assertEqual(status, 'Error, 服务器异常!')

if __name__ == '__main__':

    unittest.main()

测试输出结果

....用户1000支付失败, 金额不变

用户1000支付金额10000成功

----------------------------------------------------------------------

Ran 4 tests in 0.001s

OK

Process finished with exit code 0

从执行结果可以看出,即使第三方支付接口无法使用,但是我们自己的支付接口仍然测试通过了

也许有人会问,第三方支付都不能用,我们的测试结果是否是有效的呢?

通常我们在测试一个模块的时候,我们是可以认为其他模块的功能是正常的,只针对目标模块进行测试是没有任何问题的,所以说测试结果也是正确的

其实上述代码还可以使用另一种方式来写

mock对象的方法

"""

------------------------------------

@Time : 2020/7/21 15:22

@Auth :梵音

@File : testpay.py

@IDE  : PyCharm

@Motto: Real warriors,dare to face the bleak warning,dare to face the incisive error!

@QQ  : 829792258@qq.com

@GROUP:829792258

------------------------------------

"""

import unittest

from unittest import mock

from unittest.mock import patch

from payment.PayMent import PayApi

class TestPayApi(unittest.TestCase):

    def setUp(self):

        self.pay = PayApi()

    @patch.object(PayApi, 'auth')

    def test_success(self, mock_auth):

        mock_auth.return_value = {'status_code':'200'}

        status = self.pay.pay('1000', '12345', '10000')

        self.assertEqual(status, '支付成功')

    @patch.object(PayApi, 'auth')

    def test_fail(self, mock_auth):

        mock_auth.return_value={'status_code':'500'}

        status = self.pay.pay('1000', '12345', '10000')

        self.assertEqual(status, '支付失败')

    @patch.object(PayApi, 'auth')

    def test_error(self, mock_auth):

        mock_auth.return_value={'status_code':'300'}

        status = self.pay.pay('1000', '12345', '10000')

        self.assertEqual(status, '未知错误')

    @patch.object(PayApi, 'auth')

    def test_exception(self, mock_auth):

        mock_auth.return_value='200'

        status = self.pay.pay('1000', '12345', '10000')

        self.assertEqual(status, 'Error, 服务器异常!')

if __name__ == '__main__':

    unittest.main()

还有mock一个普通函数,mock多个方法等,这里先不赘述,写法和上面实例差不多创建了一个测试交流群,如果对软件测试、接口测试、自动化测试、面试经验交流感兴趣可以加测试交流群:829792258,还会有同行一起技术交流

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,117评论 4 360
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 66,963评论 1 290
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 107,897评论 0 240
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,805评论 0 203
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,208评论 3 286
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,535评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,797评论 2 311
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,493评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,215评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,477评论 2 244
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,988评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,325评论 2 252
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,971评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,055评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,807评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,544评论 2 271
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,455评论 2 266