mock模块怎么在python项目中使用
这篇文章给大家介绍mock模块怎么在python项目中使用,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。
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:2019/6/2614:09 @Auth:linux超 @File:ClassFunc.py @IDE:PyCharm @Motto:Realwarriors,daretofacethebleakwarning,daretofacetheincisiveerror! @QQ:28174043@qq.com @GROUP:878565760 ------------------------------------ """ importunittest fromunittestimportmock classSubClass(object): defadd(self,a,b): """两个数相加""" pass classTestSub(unittest.TestCase): """测试两个数相加用例""" deftest_sub(self): sub=SubClass()#初始化被测函数类实例 sub.add=mock.Mock(return_value=10)#mockadd方法返回10 result=sub.add(5,5)#调用被测函数 self.assertEqual(result,10)#断言实际结果和预期结果 if__name__=='__main__': unittest.main()
测试结果
. ---------------------------------------------------------------------- Ran1testin0.000s OK Processfinishedwithexitcode
测试结果显示,测试用例执行已经通过
实际上mock模拟add方法的原理是 使用相同的对象方法接收mock的对象(使用sub.add接收),那么当mock对象被调用时(sub.add())就会返回return_value参数对应的数据
这样一来,表面看起来就是模拟了add方法(这里只是我个人理解,不对请忽略)
你可以做一个实验,把用例中的add改成别的名字也一样可以测试通过
ok,继续
我们用例编写完了,而且开发既然也把功能开发完了(要骂街吗?),既然真实的功能已经可以测试了,那么我们怎么在上面用例的基础上直接测试真实功能呢?
完整的功能如何测试?
我们把用例的代码稍做修改
""" ------------------------------------ @Time:2019/6/2614:09 @Auth:linux超 @File:ClassFunc.py @IDE:PyCharm @Motto:Realwarriors,daretofacethebleakwarning,daretofacetheincisiveerror! @QQ:28174043@qq.com @GROUP:878565760 ------------------------------------ """ importunittest fromunittestimportmock classSubClass(object): defadd(self,a,b): """两个数相加""" returna+b classTestSub(unittest.TestCase): """测试两个数相加""" deftest_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(mostrecentcalllast): 1 File"D:/MyThreading/mymock.py",line37,in<module> 2 print(mock_obj()) 3 File"C:\Python36\lib\unittest\mock.py",line939,in__call__ return_mock_self._mock_call(*args,**kwargs) File"C:\Python36\lib\unittest\mock.py",line998,in_mock_call result=next(effect) StopIteration Processfinishedwithexitcode1
当所有值被取完后就会报错(这个地方有点类似生成器的原理)
存在依赖关系的功能如何测试?
假设有这样一个场景:我们要测试一个支付接口但是这个支付接口又依赖一个第三方支付接口,那么第三方支付接口我们暂时没有权限使用,那么我们该如何测试我们自己这个接口呢?
看下面的实例
假设第三方接口和我们自己的支付接口如下
""" ------------------------------------ @Time:2019/6/2615:09 @Auth:linux超 @File:PayMent.py @IDE:PyCharm @Motto:Realwarriors,daretofacethebleakwarning,daretofacetheincisiveerror! @QQ:28174043@qq.com @GROUP:878565760 ------------------------------------ """ importrequests classPayApi(object): @staticmethod defauth(card,amount): """ 第三方支付接口 :paramcard:卡号 :paramamount:支付金额 :return: """ pay_url="http://www.zhifubao.com"#第三方支付接口地址 data={"card":card,"amount":amount} response=requests.post(pay_url,data=data)#请求第三方支付接口 returnresponse#返回状态码 defpay(self,user_id,card,amount): """ 我们自己的支付接口 :paramuser_id:用户id :paramcard:卡号 :paramamount:支付金额 :return: """ #调用第三方支付接口 response=self.auth(card,amount) try: ifresponse['status_code']=='200': print('用户{}支付金额{}成功'.format(user_id,amount)) return'支付成功' elifresponse['status_code']=='500': print('用户{}支付失败,金额不变'.format(user_id)) return'支付失败' else: return'未知错误' exceptException: return"Error,服务器异常!" if__name__=='__main__': pass
很明显第三方支付接口是无法访问的,因为接口的地址是我DIY的,为了模拟实际中我们无法使用的第三方支付接口
编写测试用例
""" ------------------------------------ @Time:2019/6/2615:22 @Auth:linux超 @File:testpay.py @IDE:PyCharm @Motto:Realwarriors,daretofacethebleakwarning,daretofacetheincisiveerror! @QQ:28174043@qq.com @GROUP:878565760 ------------------------------------ """ importunittest fromunittestimportmock frompayment.PayMentimportPayApi classTestPayApi(unittest.TestCase): deftest_success(self): pay=PayApi() pay.auth=mock.Mock(return_value={'status_code':'200'}) status=pay.pay('1000','12345','10000') self.assertEqual(status,'支付成功') deftest_fail(self): pay=PayApi() pay.auth=mock.Mock(return_value={'status_code':'500'}) status=pay.pay('1000','12345','10000') self.assertEqual(status,'支付失败') deftest_error(self): pay=PayApi() pay.auth=mock.Mock(return_value={'status_code':'300'}) status=pay.pay('1000','12345','10000') self.assertEqual(status,'未知错误') deftest_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成功 ---------------------------------------------------------------------- Ran4testsin0.001s OK Processfinishedwithexitcode0
从执行结果可以看出,即使第三方支付接口无法使用,但是我们自己的支付接口仍然测试通过了
也许有人会问,第三方支付都不能用,我们的测试结果是否是有效的呢?
通常我们在测试一个模块的时候,我们是可以认为其他模块的功能是正常的,只针对目标模块进行测试是没有任何问题的,所以说测试结果也是正确的
其实上述代码还可以使用另一种方式来写
mock对象的方法
""" ------------------------------------ @Time:2019/6/2615:22 @Auth:linux超 @File:testpay.py @IDE:PyCharm @Motto:Realwarriors,daretofacethebleakwarning,daretofacetheincisiveerror! @QQ:28174043@qq.com @GROUP:878565760 ------------------------------------ """ importunittest fromunittestimportmock fromunittest.mockimportpatch frompayment.PayMentimportPayApi classTestPayApi(unittest.TestCase): defsetUp(self):self.pay=PayApi() @patch.object(PayApi,'auth') deftest_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') deftest_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') deftest_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') deftest_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模块怎么在python项目中使用就分享到这里了,希望以上内容可以对大家有一定的帮助,可以学到更多知识。如果觉得文章不错,可以把它分享出去让更多的人看到。
推荐阅读
-
Python中怎么动态声明变量赋值
这篇文章将为大家详细讲解有关Python中怎么动态声明变量赋值,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文...
-
python中变量的存储原理是什么
-
Python中怎么引用传递变量赋值
这篇文章将为大家详细讲解有关Python中怎么引用传递变量赋值,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文...
-
python中怎么获取程序执行文件路径
python中怎么获取程序执行文件路径,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的...
-
Python中如何获取文件系统的使用率
Python中如何获取文件系统的使用率,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴...
-
Python中怎么获取文件的创建和修改时间
这篇文章将为大家详细讲解有关Python中怎么获取文件的创建和修改时间,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读...
-
python中怎么获取依赖包
今天就跟大家聊聊有关python中怎么获取依赖包,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据...
-
python怎么实现批量文件加密功能
-
python中怎么实现threading线程同步
小编给大家分享一下python中怎么实现threading线程同步,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!...
-
python下thread模块创建线程的方法
本篇内容介绍了“python下thread模块创建线程的方法”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来...