赞
踩
单进程串行
执行的话会非常耗费时间。假设每条用例耗时2s,1000条就需要2000s $\approx$ 33min;还要加上用例加载、测试前/后置套件等耗时;导致测试执行效率会相对低。并行执行
;这就是一种分布式场景
来缩短测试用例的执行时间,提高效率。 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 |
|
正常执行:需要8.10s
多cpu并行执行用例,直接加个-n参数即可,后面num参数就是并行数量,比如num设置为3
1 |
|
参数:
多进程并行执行:耗时2.66s
大大的缩短了测试用例的执行时间。
xdist的分布式类似于一主多从的结构,master负责下发命令,控制slave;slave根据master的命令执行特定测试任务。
在xdist中,主是master,从是workers;xdist会产生一个或多个workers,workers都通过master来控制,每个worker相当于一个mini版pytest执行器
。
master不执行测试任务,只对worker收集到的所有用例进行分发;每个worker负责执行测试用例,然后将执行结果反馈给master;由master统计最终测试结果。
master在测试会话(test session)
开始前产生一个或多个worker。
实际编译执行测试代码的worker可能是本地机器也可能是远程机器。
每个worker类似一个迷你型的pytest执行器
。
worker会执行一个完整的test collection
过程。【收集所有测试用例的过程】
然后把测试用例的ids
返回给master。【ids表示收集到的测试用例路径】
master不执行任何测试用例。
注意:分布式测试(pytest-xdist)方式执行测试时不会输出测试用例中的print内容,因为master并不执行测试用例。
master接收到所有worker收集的测试用例集之后,master会进行一些完整性检查,以确保所有worker都收集到一样的测试用例集(包括顺序)。
如果检查通过,会将测试用例的ids列表转换成简单的索引列表,每个索引对应一个测试用例的在原来测试集中的位置。
这个方案可行的原因是:所有的节点都保存着相同的测试用例集。
并且使用这种方式可以节省带宽,因为master只需要告知workers需要执行的测试用例对应的索引,而不用告知完整的测试用例信息。
有以下四种分发策略:命令行参数 --dist=mode选项
(默认load
)
each:master将完整的测试索引列表分发到每个worker,即每个worker都会执行一遍所有的用例。
load:master将大约$\frac{1}{n}$的测试用例以轮询的方式分发到各个worker,剩余的测试用例则会等待worker执行完测试用例以后再分发;每个用例只会被其中一个worker执行一次。
loadfile:master分发用例的策略为按ids
中的文件名(test_xx.py/xx_test.py)进行分发,即同一个测试文件中的测试用例只会分发给其中一个worker;具有一定的隔离性。
loadscope:master分发用例对策略为按作用域进行分发,同一个模块下的测试函数或某个测试类中的测试函数会分发给同一个worker来执行;即py文件中无测试类的话(只有测试function)将该模块分发给同一个worker执行,如果有测试类则会将该文件中的测试类只会分发给同一个worker执行,多个类可能分发给多个worker;目前无法自定义分组,按类 class 分组优先于按模块 module 分组。
注意:可以使用pytest_xdist_make_scheduler
这个hook来实现自定义测试分发逻辑。
如:想按目录级别来分发测试用例:
1 2 3 4 5 6 7 8 9 10 |
|
xdist.scheduler.LoadScopeScheduling
并重写_split_scope
方法pytest_xdist_make_scheduler
1 |
|
pytest_runtestloop
:pytest的默认实现是循环执行所有在test_session
这个对象里面收集到的测试用例。pytest_runtest_protocol
。pytest_runtest_protocol(item, nextitem)
hook的参数要求,为了将nextitem
传给hook。pytest_runtest_protocol
,因为这时nextitem
参数已经可以确定。shutdown
信号, 那么就将nextitem
参数设为None
, 然后执行 pytest_runtest_protocol
shutdown
信号给所有worker。pytest hooks
比如:pytest_runtest_logstart
、pytest_runtest_logreport
确保整个测试活动进行正常运作。注意:pytest-xdist 是让每个 worker 进程执行属于自己的测试用例集下的所有测试用例。这意味着在不同进程中,不同的测试用例可能会调用同一个 scope 范围级别较高(例如session)的 fixture,该 fixture 则会被执行多次,这不符合 scope=session 的预期。
pytest-xdist 没有内置的支持来确保会话范围的 fixture 仅执行一次,但是可以通过使用锁定文件进行进程间通信来实现;让scope=session 的 fixture 在 test session 中仅执行一次。
示例:需要安装 filelock 包,安装命令pip install filelock
FileLock
仅产生一次fixture数据。 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 |
|
用于并行
和并发
测试的 pytest 插件
1 |
|
--workers=n
:多进程运行需要加此参数, n是进程数。默认为1
--tests-per-worker=n
:多线程需要添加此参数,n是线程数
如果两个参数都配置了,就是进程并行;每个进程最多n个线程,总线程数:进程数*线程数
【注意】
在windows上进程数永远为1。
需要使用 if name == “main” :
在命令行窗口运行测试用例会报错
示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 |
|
简而言之,pytest-xdist
并行性pytest-parallel
是并行性和并发性。
- 现在我也找了很多测试的朋友,做了一个分享技术的交流群,共享了很多我们收集的技术文档和视频教程。
- 如果你不想再体验自学时找不到资源,没人解答问题,坚持几天便放弃的感受
- 可以加入我们一起交流。而且还有很多在自动化,性能,安全,测试开发等等方面有一定建树的技术大牛
- 分享他们的经验,还会分享很多直播讲座和技术沙龙
- 可以免费学习!划重点!开源的!!!
- qq群号:485187702【暗号:csdn11】
最后感谢每一个认真阅读我文章的人,看着粉丝一路的上涨和关注,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走! 希望能帮助到你!【100%无套路免费领取】
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。