unittest是Python标准库自带的单元测试框架,是Python版本的JUnit,关于unittest框架的使用,官方文档非常详细,网上也有不少好的教程,这里就不多说了。

本文主要分享在使用unittest的过程中,做的一些扩展尝试。先上一个例子。

import unittest

class TestLegion(unittest.TestCase):

def test_create_legion(self):

"""创建军团

:return:

"""

def test_bless(self):

""" 公会祈福

:return:

"""

def test_receive_bless_box(self):

""" 领取祈福宝箱

:return:

"""

def test_quit_legion(self):

"""退出军团

:return:

"""

这是一个标准的使用unittest进行测试的例子,写完后心里美滋滋,嗯,就按照这个顺序测就可以了。结果一运行。

什么鬼。执行的顺序乱了。第一个执行的测试用例并不是创建军团,而是公会祈福,此时玩家还没创建军团,进行公会祈福的话会直接报错,导致用例失败。

到这里有些同学会想说,为什么要让测试用例之间有所依赖呢?

的确,如果完全没依赖,测试用例的执行顺序是不需要关注的。但是这样对于用例的设计和实现,要求就高了许多。而对游戏来说,一个系统内的操作,是有很大的关联性的。以军团为例,军团内的每个操作都有一个前提,你需要加入一个军团。所以要实现用例之间的完全解耦,需要每个用例开始之前,检测玩家的军团状态。

如果可以控制测试用例的执行顺序,按照功能玩法流程一遍走下来,节省的代码量是非常可观的,阅读测试用例也会清晰许多。

如何控制unittest用例执行的顺序呢?

我们先看看,unittest是怎么样对用例进行排序的。在loader.py的loadTestsFromTestCase方法里边,调用了getTestCaseNames方法来获取测试用例的名称

def getTestCaseNames(self, testCaseClass):

"""Return a sorted sequence of method names found within testCaseClass

"""

def isTestMethod(attrname, testCaseClass=testCaseClass,

prefix=self.testMethodPrefix):

return attrname.startswith(prefix) and \

callable(getattr(testCaseClass, attrname))

testFnNames = list(filter(isTestMethod, dir(testCaseClass)))

if self.sortTestMethodsUsing:

testFnNames.sort(key=functools.cmp_to_key(self.sortTestMethodsUsing))

return testFnNames

可以看到,getTestCaseNames方法对测试用例的名称进行了排序

testFnNames.sort(key=functools.cmp_to_key(self.sortTestMethodsUsing))

看看排序方法

def three_way_cmp(x, y):

"""Return -1 if x < y, 0 if x == y and 1 if x > y"""

return (x > y) - (x < y)

根据排序规则,unittest执行测试用例,默认是根据ASCII码的顺序加载测试用例,数字与字母的顺序为:0-9,A-Z,a-z。

做个实验:

import functools

case_names = ["test_buy_goods", "test_Battle", "test_apply", "test_1_apply"]

def three_way_cmp(x, y):

"""Return -1 if x < y, 0 if x == y and 1 if x > y"""

return (x > y) - (x < y)

case_names.sort(key=functools.cmp_to_key(three_way_cmp))

print(case_names)

output:['test_1_apply', 'test_Battle', 'test_apply', 'test_buy_goods']

基于unittest的机制,如何控制用例执行顺序呢?查了一些网上的资料,主要介绍了两种方式:

方式1,通过TestSuite类的addTest方法,按顺序加载测试用例:

suite = unittest.TestSuite()

suite.addTest(TestLegion("test_create_legion"))

suite.addTest(TestLegion("test_bless"))

suite.addTest(TestLegion("test_receive_bless_box"))

suite.addTest(TestLegion("test_quit_legion"))

unittest.TextTestRunner(verbosity=3).run(suite)

方式2,通过修改函数名的方式:

class TestLegion(unittest.TestCase):

def test_1_create_legion(self):

"""创建军团

:return:

"""

def test_2_bless(self):

""" 公会祈福

:return:

"""

def test_3_receive_bless_box(self):

""" 领取祈福宝箱

:return:

"""

def test_4_quit_legion(self):

"""退出军团

:return:

"""

看起来都能满足需求,但是都不够好用,繁琐,代码不好维护。

那就造个轮子吧

于是开始了utx这个小项目,那么如何在不改动代码的情况下,让测试用例按照编写的顺序依次执行呢?

方案就是,在测试类初始化的时候,将测试方法按照编写的顺序,自动依次重命名为“test_1_create_legion”,“test_2_bless”,“test_3_receive_bless_box”等等,从而实现控制测试用例的执行。

这就需要控制类的创建行为,Python提供了一个非常强力的工具:元类,在元类的__new__方法中,我们可以获取类的全部成员函数,另外基于Python3.6的字典底层重构后,字典是有序的了,默认顺序和添加的顺序一致。所以我们拿到的测试用例,就和编写的顺序一致了。

接下来,就是按照顺序,依次改名了,定义一个全局的total_case_num变量,每次进行改名的时候,total_case_num递增+1,作为用例的id,加入到用例的名字当中。

@staticmethod

def modify_raw_func_name_to_sort_case(raw_func_name, raw_func):

case_id = Tool.general_case_id()

setattr(raw_func, CASE_ID_FLAG, case_id)

if setting.sort_case:

func_name = raw_func_name.replace("test_", "test_{:05d}_".format(case_id))

else:

func_name = raw_func_name

return func_name

接下来是定义自己的TestCase类,继承unittest.TestCase,使用上边定义的元类

class _TestCase(unittest.TestCase, metaclass=Meta):

def shortDescription(self):

"""覆盖父类的方法,获取函数的注释

:return:

"""

doc = self._testMethodDoc

doc = doc and doc.split()[0].strip() or None

return doc

最后一步,对unittest打一个猴子补丁,将unittest.TestCase替换为自定义的_TestCase

unittest.TestCase = _TestCase

看下运行效果,代码和本文开始的例子一样,只是多了一句utx库的导入。

import unittest

from utx import *

class TestLegion(unittest.TestCase):

def test_create_legion(self):

"""创建军团

:return:

"""

def test_bless(self):

""" 公会祈福

:return:

"""

def test_receive_bless_box(self):

""" 领取祈福宝箱

:return:

"""

def test_quit_legion(self):

"""退出军团

:return:

"""

运行效果:

执行顺序就和我们的预期一致了~

基于这一套,开始加上其他的一些扩展功能,比如

用例自定义标签,可以运行指定标签的测试用例

@unique

class Tag(Enum):

SMOKE = 1 # 冒烟测试标记,可以重命名,但是不要删除

FULL = 1000 # 完整测试标记,可以重命名,但是不要删除

# 以下开始为扩展标签,自行调整

SP = 2

class TestLegion(unittest.TestCase):

@tag(Tag.SMOKE)

def test_create_legion(self):

pass

@tag(Tag.SP, Tag.FULL)

def test_quit_legion(self):

"""退出军团

:return:

"""

print("吧啦啦啦啦啦啦")

assert 1 == 2

from utx import *

if __name__ == '__main__':

setting.run_case = {Tag.SMOKE} # 只运行SMOKE冒烟用例

# setting.run_case = {Tag.FULL} # 运行全部测试用例

# setting.run_case = {Tag.SMOKE, Tag.SP} # 只运行标记为SMOKE和SP的用例

runner = TestRunner()

runner.add_case_dir(r"testcase")

runner.run_test(report_title='接口自动化测试报告')

数据驱动

class TestLegion(unittest.TestCase):

@data(["gold", 100], ["diamond", 500])

def test_bless(self, bless_type, award):

print(bless_type)

print(award)

@data(10001, 10002, 10003)

def test_receive_bless_box(self, box_id):

""" 领取祈福宝箱

:return:

"""

print(box_id)

# 默认会解包测试数据来一一对应函数参数,可以使用unpack=False,不进行解包

class TestBattle(unittest.TestCase):

@data({"gold": 1000, "diamond": 100}, {"gold": 2000, "diamond": 200}, unpack=False)

def test_get_battle_reward(self, reward):

""" 领取战斗奖励

:return:

"""

print(reward)

print("获得的钻石数量是:{}".format(reward['diamond']))

检测测试用例是否编写了说明描述

2017-11-03 12:00:19,334 WARNING legion.test_legion.test_bless没有用例描述

执行测试用例的时候,显示执行进度

2017-11-03 12:00:19,336 INFO 开始进行测试

2017-11-03 12:00:19,436 INFO Start to test legion.test_legion.test_create_legion (1/5)

2017-11-03 12:00:19,536 INFO Start to test legion.test_legion.test_receive_bless_box (2/5)

2017-11-03 12:00:19,637 INFO Start to test legion.test_legion.test_receive_bless_box (3/5)

2017-11-03 12:00:19,737 INFO Start to test legion.test_legion.test_receive_bless_box (4/5)

2017-11-03 12:00:19,837 INFO Start to test legion.test_legion.test_quit_legion (5/5)

setting类提供多个设置选项进行配置

class setting:

# 只运行的用例类型

run_case = {Tag.SMOKE}

# 开启用例排序

sort_case = True

# 每个用例的执行间隔,单位是秒

execute_interval = 0.1

# 开启检测用例描述

check_case_doc = True

# 显示完整用例名字(函数名字+参数信息)

full_case_name = False

# 测试报告显示的用例名字最大程度

max_case_name_len = 80

# 执行用例的时候,显示报错信息

show_error_traceback = True

# 生成ztest风格的报告

create_ztest_style_report = True

# 生成bstest风格的报告

create_bstest_style_report = True

集成 ztest 和 BSTestRunner 生成测试报告,感谢两位作者的测试报告模版

utx库核心源码不到200行,就不做过多讲解了,直接去Github看吧

作者:煎煎煎饼 链接:https://www.jianshu.com/p/d65f97723af7 來源:简书 简书著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。

python按顺序执行函数_Python3的unittest用例按编写顺序执行相关推荐

  1. Python3的unittest用例按编写顺序执行

    unittest是Python标准库自带的单元测试框架,是Python版本的JUnit,关于unittest框架的使用,官方文档非常详细,网上也有不少好的教程,这里就不多说了. 本文主要分享在使用un ...

  2. python获取系统时间函数_python3中datetime库,time库以及pandas中的时间函数区别与详解...

    1介绍datetime库之前 我们先比较下time库和datetime库的区别 先说下time 在 Python 文档里,time是归类在Generic Operating System Servic ...

  3. python读取多行函数_Python3基础 __doc__ 单行与多行函数文档

    ? ????   Python : 3.7.0 ??????   OS : Ubuntu 18.04.1 LTS ??????  IDE : PyCharm 2018.2.4 ????? Conda ...

  4. python写软件测试用例_Python单元测试框架unittest:单个测试用例编写步骤及实例...

    一.Python单元测试框架的编写步骤 导入模块 必须继承unittest.TestCase 主要是配置环境:进行测试前的初始化工作,比如在接口测试前面做一些前置的参数赋值,数据库操作等等 定义测试用 ...

  5. Python自学记录——返回函数、匿名函数、装饰器与偏函数

    国庆节快乐~~虽说今天是假期的最后一天.. 好久没学习Python了..值得高兴的是 <怪物猎人:世界>目前所有的龙我(统枪)都打过一遍了 (/得意). 正题,开始学习.记录: 返回函数 ...

  6. (42)2021-03-01(物体运动、swiper软件、自执行函数)

    物体运动.swiper软件.自执行函数 (一).物体运动 一.物体运动 二.分享1 三.分享2 四.淡入淡出 五.缓冲运动 六.缓冲运动案例 七.多物体运动 八.offset的问题 九.多样式运动 十 ...

  7. JavaScript立即执行函数报错--立即执行函数原理分析

    JavaScript立即执行函数报错 1.'()'在JavaScript环境中有提升运算符优先级和执行函数的作用如果直接在JavaScript环境中执行'()',环境会报出语法错误提示. 2.'()' ...

  8. 立即执行函数和函数劫持

    一.ES6的类的继承 1.继承的好处:在父类中定义的属性和方法,子类继承后就可以直接使用. 2.类继承过程中的向上转型:子类对象的类型一定是父类的类型,父类对象的类型不能是子类的类型 (1)typeo ...

  9. JavaScript中的自执行函数和闭包实现过程

    在JS中,也常常需要复用,将一些常见的操作封装起来,目前看到的很多JS库,比如JQuery就是非常典型的,那么如何在JS中写一个可复用的呢?虽然在JS中有类的这个说法,但是我对这个玩意还是很抵触的,但 ...

  10. JavaScript自执行函数,自执行函数是什么,存在的意义?

    JavaScript自执行函数 1.自执行函数是什么 2.自执行函数存在的意义 2.1封装 3.自执行函数两种常见的的写法 3.1自执行函数的第一种写法 3.2自执行函数的第二种写法 4.自执行函数的 ...

最新文章

  1. Codeforces Round #550 (Div. 3)E. Median String
  2. 对 Excel 工作簿中的数字签名和代码签名的说明
  3. blob转file对象_JavaScript Blob 对象解析
  4. 自编码的matlab代码,深度学习自动编码机MATLAB实现
  5. python 基本操作 多维数组 循环
  6. Android Service介绍
  7. HTML的--iframe
  8. java语言注释符号,高级Java开发必看
  9. mysql left join 几个意思
  10. php设计模式 -- 迭代器模式
  11. [java实战篇]--java的GUI(1)
  12. 奥斯汀页眉怎么设置_wps怎么只删除本页的页眉_Word页眉的设置和删除,这些问题你有遇到过吗?...
  13. 程序员编程规范之注释
  14. 手游沙巴克传奇当前服务器维护,《沙巴克传奇》12月18日安卓、IOS维护公告
  15. 不要迷恋我,虽然我利用Python来耍植物大战僵尸
  16. 美赛论文Latex简易模板 | 快速上手(附注释)
  17. 用Python制作可视化GUI界面,一键实现证件照背景颜色的替换
  18. LS-DYNA (动力分析程序)
  19. JNI的替代者—使用JNA访问Java外部功能接口 DLL
  20. 2019华为新年贺词

热门文章

  1. wps大纲栏显示在右边_隐藏显示word页面标记,就这么几招,你会吗?
  2. canvas图片合成模糊变清晰的方法
  3. 跳转微信公众号首页方式
  4. 大学计算机实践教程咖啡文化,北京大学图书馆学习共享空间.docx
  5. python爬取公众号
  6. tc275怎么移植到ram中去,把函数
  7. python windows自动化 爬虫_python自动化之爬虫原理及简单案例
  8. Unity3d 周分享(8期 2018.12.16)
  9. Pipeline aborted due to error
  10. 计算机网页的设计与应用的前言,网页设计前言.ppt