Python Pytest分布式执行 pytest-xdist
Time will tell.
手工测试用例非常多时,如1000条用例,假设每条用例执行要1分钟,一个测试人员需要1000分钟才能执行完。
项目非常紧急时,我们会用测试人力成本换取时间成本,这时会多找些人分配任务,而缩短时间。如果是10个人执行,1000个用例理论上只需100分钟,时间缩短到了1/10,为项目节省了时间成本。
同样道理,当我们测试用例非常多时,一条一条执行,显然会很慢,那么如何让测试用例并行执行呢?这就是今天要讲的 pytest 分布式执行插件pytest-xdist
。
1、pytest-xdist
cmd 里面使用 pip 安装,pip install pytest-xdist 。
>pip show pytest-xdist
Name: pytest-xdist
Version: 1.23.2
Summary: pytest xdist plugin for distributed testing and loop-on-failing modes
Home-page: https://github.com/pytest-dev/pytest-xdist
Author: holger krekel and contributors
Author-email: pytest-dev@python.org,holger@merlinux.eu
License: MIT
Location: e:\python36\lib\site-packages
Requires: execnet, pytest-forked, six, pytest
pytest-xdist 插件扩展了一些独特的测试执行模式 pytest:
- 测试运行并行化:如果有多个 CPU 或主机,则可以将它们用于组合测试运行。加快运行速度。
- looponfail:在子进程中重复运行测试。每次运行之后, pytest 会等待,直到项目中的文件发生更改,然后重新运行以前失败的测试。重复此过程直到所有测试通过,之后再次执行完整运行。
- 多平台覆盖:您可以指定不同的 Python 解释器或不同的平台,并在所有平台上并行运行测试。
在远程运行测试之前,pytest 有效地将您的程序源代码 “rsyncs” 到远程位置。报告所有测试结果并显示给您的本地终端。您可以指定不同的 Python 版本和解释器。
2、并行测试
多 cpu 并行执行用例,直接加个 -n 参数即可,后面 num 参数就是并行数量,比如 num 设置为3。
pytest -n 3
运行以下代码,项目结构如下:
web_conf_py是项目工程名称
│ conftest.py
│ __init__.py
│
├─baidu
│ │ conftest.py
│ │ test_1_baidu.py
│ │ test_2.py
│ │ __init__.py
│
├─blog
│ │ conftest.py
│ │ test_2_blog.py
│ │ __init__.py
代码参考:
# web_conf_py/conftest.py
import pytest
@pytest.fixture(scope="session")
def start():
print("\n打开首页")
return "yoyo"
# web_conf_py/baidu/conftest.py
import pytest
@pytest.fixture(scope="session")
def open_baidu():
print("打开百度页面_session")
# web_conf_py/baidu/test_1_baidu.py
import pytest
import time
def test_01(start, open_baidu):
print("测试用例test_01")
time.sleep(1)
assert start == "yoyo"
def test_02(start, open_baidu):
print("测试用例test_02")
time.sleep(1)
assert start == "yoyo"
if __name__ == "__main__":
pytest.main(["-s", "test_1_baidu.py"])
# web_conf_py/baidu/test_2.py
import pytest
import time
def test_06(start, open_baidu):
print("测试用例test_01")
time.sleep(1)
assert start == "yoyo"
def test_07(start, open_baidu):
print("测试用例test_02")
time.sleep(1)
assert start == "yoyo"
if __name__ == "__main__":
pytest.main(["-s", "test_2.py"])
# web_conf_py/blog/conftest.py
import pytest
@pytest.fixture(scope="function")
def open_blog():
print("打开blog页面_function")
# web_conf_py/blog/test_2_blog.py
import pytest
import time
def test_03(start, open_blog):
print("测试用例test_03")
time.sleep(1)
assert start == "yoyo"
def test_04(start, open_blog):
print("测试用例test_04")
time.sleep(1)
assert start == "yoyo"
def test_05(start, open_blog):
'''跨模块调用baidu模块下的conftest'''
print("测试用例test_05,跨模块调用baidu")
time.sleep(1)
assert start == "yoyo"
if __name__ == "__main__":
pytest.main(["-s", "test_2_blog.py"])
正常运行消耗时间:7.12 seconds。
E:\PY\web_conf_py>pytest
============================= test session starts =============================
platform win32 -- Python 3.6.0, pytest-3.6.3, py-1.5.4, pluggy-0.6.0
rootdir: E:\PY\web_conf_py, inifile:
plugins: xdist-1.23.2, metadata-1.7.0, html-1.19.0, forked-0.2
collected 7 items
baidu\test_1_baidu.py .. [ 28%]
baidu\test_2.py .. [ 57%]
blog\test_2_blog.py ... [100%]
========================== 7 passed in 7.12 seconds ===========================
设置并行运行数量为3,消耗时间:3.64 seconds,大大缩短了用例时间。
E:\PY\web_conf_py>pytest -n 3
============================= test session starts =============================
platform win32 -- Python 3.6.0, pytest-3.6.3, py-1.5.4, pluggy-0.6.0
rootdir: E:\PY\web_conf_py, inifile:
plugins: xdist-1.23.2, metadata-1.7.0, html-1.19.0, forked-0.2
gw0 [7] / gw1 [7] / gw2 [7]
scheduling tests via LoadScheduling
....... [100%]
========================== 7 passed in 3.64 seconds ===========================
3、测试报告
使用pytest-xdist
插件也能生成 html 报告,完美支持pytest-html
插件。
pytest -n 3 —html=report.html —self-contained-html
好啦,以上内容就分享到这里,如果你对更多内容、Python自动化软件测试、面试题感兴趣的话可以加入我们一起学习175317069。有各项学习资源,更有行业深潜多年的技术人分析讲解。
最后希望看到这里的你终会成为一名极具竞争力的高级测试工程师。
欢迎【点赞】、【评论】、【关注】~
Time will tell.(时间会证明一切)