全网最全pytest大型攻略,单元测试学这就够了

2023-12-20

pytest 是一款以python为开发语言的第三方测试,主要特点如下:

  • 比自带的 unittest 更简洁高效,兼容 unittest框架
  • 支持参数化
  • 可以更精确的控制要测试的测试用例
  • 丰富的插件,已有300多个各种各样的插件,也可自定义扩展,如pytest-selenium、pytest-html、pytest-rerunfailures、pytes-xdish
  • 可很好的和CI工具结合

安装

pip install pytest

测试用例编写规则

  • 测试文件以test_开头 或者 _test结尾
  • 测试类以Test开头,并且不能带有 init 方法
  • 测试文件以 test_开头
  • 断言使用基本的 assert 即可

pytest会递归查找当前目录及子目录下所有 以test_开始 或者 _test结尾的python脚本,执行其中符合规则的函数和方法,不需要显示调用

运行命令:(cmd进入用例所在目录)

pytest folder_name ======》直接运行文件夹内符合规则的所有用例

pytest test_file.py ======》执行某个py文件中的用例

pytest test_file.py::test_func ======》执行模块内的某个函数(节点运行)

pytest
test_file.py::TestClass::test_method ======》执行模块内测试类的某个方法(节点运行)

pytest test_file.py::TestClass ======》执行模块内某个测试类(节点运行)

pytest test_file.py::TestClass
test_file2.py::test_mothod ======》多节点运行,中间用空格隔开

pytest -k pass ======》匹配用例名称的表达式,含有“pass”的被执行,其他的deselected

pytest -k "pass or fail" ======》组合匹配,含有“pass” 和 “fail”的被执行

pytest -k "not pass" ======》排除运行,不含“pass”的被执行

pytest -m finished ======》标记表达式,运行用@pytest.mark.finished 标记的用例

pytest -m "finished and not merged" ======》多个标记逻辑匹配,运行含有finished 不含 merged标记的用例

pytest -v ======》运行时显示详细信息

pytest -s ======》显示打印消息

pytest -x ======》遇到错误就停止运行

pytest -x --maxfail=2 ======》遇到两个错误就停止运行

pytest --setup-show ======》跟踪固件运行

pytest -v --reruns 5 --reruns-delay 1 ======》运行失败的用例间隔1s重新运行5次 pip install pytest-rerunfailures

pytest ======》多条断言,报错后,后面的依然执行, pip install pytest-assume,断言 pytest.assume(2==4)

pytest -n 3 ======》3个cpu并行执行测试用例,需保证测试用例可随机执行, pip install pytest-xdist分布式执行插件,多个cpu或主机执行

pytest -v -n auto ======》自动侦测系统里cpu的数目

pytest --count=2 ======》重复运行测试 pip install pytest-repeat

pytest --html=./report/report.html ======》生成报告,此报告中css是独立的,分享时会丢失样式,pip install pytest-html

pytest --html=report.html --self-containd-html ======》合并css到html报告中,除了passed所有行都被展开

pytest --durations=10 ======》获取最慢的10个用例的执行耗时

用例执行顺序控制

pytest 用例执行顺序默认是按字母顺序去执行,要控制执行顺序,需要安装插件 pytest-ordering:pip install pytest-ordering

在测试方法上加上装饰器:

@pytest.mark.last 最后一个执行

@pytest.mark.run(order=n) n=1则是第一个执行

Mark

标签的使用方法:

注册标签名 / 内置标签—> 在测试用例 / 测试类 / 模块文件 前面加上 @pytest.mark.标签名

注册方法:

1.在conftest.py 文件中添加代码

# 单个标签文件内容

def pytest_configure(config):

config.addinivalue_line("markers", "demo:demo标签名称")

# 多个标签文件内容

def pytest_configure(config):

marker_list = ["p0:p0级别用例", "p1:p1级别用例", "p2:p2级别用例"] # 标签名称

for markers in marker_list:

config.addinivalue_line("markers", markers)

2.项目中添加pytest.ini配置文件

[pytest]

markers =

p0:p0级别用例

p1:p1级别用例

p2:p2级别用例

使用方法:

import pytest

@pytest.mark.p0

def test_mark01():

print("函数级别的mark_p0")

@pytest.mark.p1

def test_mark02():

print("函数级别的mark_p1")

@pytest.mark.P2

class TestDemo:

def test_mark03(self):

print("mark_p2")

def test_mark04(self):

print("mark_p2")

运行方式:

命令行运行

pytest -m "p0 and p1"

文件运行

pytest.main(["-m", "P0", "--html=report.html"])

内置标签:

参数化:@pytest.mark.parametrize(argnames, argvalues)

无条件跳过用例:@pytest.mark.skip(reason=“xxx”)

有条件跳过用例:@pytest.mark.skipif(version < 0.3, reason = “not supported until 0.3”)

预测执行失败进行提示标记:@pytest.mark.xfail(version < 0.3, reason = “not supported until 0.3”),运行结果为X(通过xpassed,失败xfailed)

# 参数化

import hashlib

@pytest.mark.parametrize("x", list(range(10)))

def test_somethins(x):

time.sleep(1)

@pytest.mark.parametrize("passwd",["123456", "abcdefgfs", "as52345fasdf4"])

def test_passwd_length(passwd):

assert len(passwd) >= 8

@pytest.mark.parametrize('user, passwd',[('jack', 'abcdefgh'),('tom', 'a123456a')])

def test_passwd_md5(user, passwd):

db = {

'jack': 'e8dc4081b13434b45189a720b77b6818',

'tom': '1702a132e769a623c1adb78353fc9503'

}

assert hashlib.md5(passwd.encode()).hexdigest() == db[user]

# 如果觉得每组测试的默认参数显示不清晰,可以使用 pytest.param 的 id 参数进行自定义

@pytest.mark.parametrize("user, passwd",

[pytest.param("jack", "abcdefgh", id = "User<Jack>"),

pytest.param("tom", "a123456a", id = "User<Tom>")])

def test_passwd_md5_id(user, passwd):

db = {

'jack': 'e8dc4081b13434b45189a720b77b6818',

'tom': '1702a132e769a623c1adb78353fc9503'

}

assert hashlib.md5(passwd.encode()).hexdigest() == db[user]

Fixture

固件:是一些函数,pytest会在执行函数之前或者之后加载运行它们,相当于预处理和后处理。

fixture的目的是提供一个固定基线,在该基线上测试可以可靠地、重复的执行。

名称:默认为定义时的函数名,可以通过 @pytest.fixture(name="demo") 给fixture重命名

定义:在固件函数定义前加上@pytest.fixture();fixture是有返回值的,没return则返回None

使用:作为参数、使用usefixtures、自动执行(定义时指定autouse参数)

def test_demo(fixture_func_name)

@pytest.mark.usefixtures("fixture_func_name1", "fixture_func_name2") 标记函数或者类

预处理和后处理:用yield关键词,yield之前的代码是预处理,之后的是后处理

作用域:通过scope参数控制作用域

function:函数级,每个测试函数都会执行一次(默认)

class:类级别,每个测试类执行一次,所有方法都共享这个fixture

module:模块级别,每个模块.py执行一次,模块中所有测试函数、类方法 或者 其他fixture 都共享这个fixture

session:会话级别,每次会话只执行一次,一次会话中所有的函数、方法都共享这个fixture

集中管理:使用文件conftest.py 集中管理,在不同层级定义,作用于在其所在的目录和子目录,pytest会自动调用

scope、yield、auto的使用

# scope、yield、auto使用

@pytest.fixture(scope = "function", autouse=True)

def function_scope():

pass

@pytest.fixture(scope = "module", autouse=True)

def module_scope():

pass

@pytest.fixture(scope = "session")

def session_scope():

pass

@pytest.fixture(scope = "class", autouse=True)

def class_scope():

pass

import time

DATE_FORMAT = '%Y-%m-%d %H:%M:%S'

@pytest.fixture(scope='session', autouse=True)

def timer_session_scope():

start = time.time()

print('\nsession start: {}'.format(time.strftime(DATE_FORMAT, time.localtime(start))))

yield

finished = time.time()

print('\nsession finished: {}'.format(time.strftime(DATE_FORMAT, time.localtime(finished))))

print('session Total time cost: {:.3f}s'.format(finished - start))

def test_1():

time.sleep(1)

def test_2():

time.sleep(2)

'''

执行命令:pytest --setup-show -s

固件执行结果:

test_pytest_study.py

session start: 2020-04-16 17:29:02

SETUP S timer_session_scope

SETUP M module_scope

SETUP C class_scope

SETUP F function_scope

test_pytest_study.py::test_3 (fixtures used: class_scope, function_scope, module_scope, timer_session_scope).

TEARDOWN F function_scope

TEARDOWN C class_scope

SETUP C class_scope

SETUP F function_scope

test_pytest_study.py::test_4 (fixtures used: class_scope, function_scope, module_scope, timer_session_scope).

TEARDOWN F function_scope

TEARDOWN C class_scope

TEARDOWN M module_scope

session finished: 2020-04-16 17:29:05

session Total time cost: 3.087s

TEARDOWN S timer_session_scope

'''

使用文件conftest.py 集中管理

# conftest.py

# conding=utf-8

import pytest

@pytest.fixture()

def postcode():

print("执行postcode fixture")

return "010"

# test_demo.py

# coding=utf-8

import pytest

class TestDemo():

def test_postcode(self, postcode):

assert postcode == "010"

if __name__=="__main__":

pytest.main(["--setup-show", "-s", "test_demo.py"])

python test_demo.py

执行过程:

test_demo.py 执行postcode fixture

SETUP F postcode

test_demo.py::TestDemo::test_postcode (fixtures used: postcode).

TEARDOWN F postcode

# 如果整个文件都用一个fixture,可以用pytestmark标记

pytestmark = pytest.mark.usefixtures("login")

fixture参数化

固件参数化需要使用pytest内置的固件request,并通过 request.param 获取参数。

# test_demo.py

@pytest.fixture(params=[

("user1", "passwd1"),

("user2", "passwd2")

])

def param(request):

return request.param

@pytest.fixture(autouse=True)

def login(param):

print("\n登录成功 %s %s" %param)

yield

print("\n退出成功 %s %s" %param)

def test_api():

assert 1 == 1

'''

pytest -s -v test_demo.py

运行结果:

test_demo.py::test_api[param0]

登录成功 user1 passwd1

PASSED

退出成功 user1 passwd1

test_demo.py::test_api[param1]

登录成功 user2 passwd2

PASSED

退出成功 user2 passwd2

'''

assert

assert "h" in "hello"

assert 3==4

assert 3!=4

assert f()==4

assert 5>6

assert not xx

assert {"0", "1", "2"} == {"0", "1", "2"}

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

全网最全pytest大型攻略,单元测试学这就够了 的相关文章

随机推荐

  • python之pyQt5实例:鼠标创建点

    实现一个基于PyQt5的画布 可以通过鼠标左键点击的方式创建点 并在画布中显示出来 from PyQt5 QtWidgets import QApplication QWidget QMainWindow QInputDialog from
  • yolov5障碍物识别-雪糕筒识别(代码+教程)

    简介 这是一个检测交通锥并识别颜色的项目 我使用 yolov5 来训练和检测视锥细胞 此外 我使用 k 均值来确定主色 以对锥体颜色进行分类 目前 支持的颜色为红色 黄色 绿色和蓝色 其他颜色被归类为未知 数据集和注释 我使用了一个自收集的
  • 软件测试/测试开发|如何在Ubuntu系统中安装docker

    Docker是一种流行的容器化平台 它能够简化应用程序的部署和管理 本文将介绍在Ubuntu操作系统上安装Docker的步骤 以便我们可以开始使用Docker来构建和运行容器化应用程序 系统版本 本文以Ubuntu20 05系统为例安装do
  • C语言—每日选择题—Day55

    第一题 1 若有如下定义 则 p1 m p2 p1 是正确赋值语句 说法是否正确 int p1 int p2 int m 5 int n A 正确 B 错误 答案及解析 A 本题考查的是对类型的认识 p1是指针类型 赋值的右操作数就必须是地
  • 自然语言处理初学者指南(附1000页的PPT讲解)

    自然语言处理是计算机科学领域和人工智能领域的重要研究方向之一 旨在探索实现人与计算机之间用自然语言进行有效交流的理论与方法 它融合了语言学 计算机科学 机器学习 数学 认知心理学等多学科内容 涉及从字 词 短语到句子 段落 篇章的多种语言单
  • nodejs微信小程序+python+PHP的大学生社会实践申报系统-计算机毕业设计推荐

    系统主要包括首页 个人中心 项目分类管理 学生管理 教师管理 学生活动申报管理 社会实践活动管理 活动报名申请管理 系统管理等功能模块 根据需求分析制定模块并设计数据库结构 大学生社会实践申报系统的需求和管理上的不断提升 大学生社会实践申报
  • nodejs+vue+微信小程序+python+PHP协同过滤算法的电影推荐系统-计算机毕业设计推荐python

    信息数据的处理完全依赖人工进行操作 会耗费大量的人工成本 特别是面对大量的数据信息时 传统人工操作不仅不能对数据的出错率进行保证 所以电子化信息管理的出现就能缓解以及改变传统人工方式面临的处境 一方面可以确保信息数据在短时间被高效处理 还能
  • 学术编辑青睐《乡村振兴战略下传统村落文化旅游设计》期货刊印辉少许

    学术编辑青睐 乡村振兴战略下传统村落文化旅游设计 期货刊印辉少许 学术编辑青睐 乡村振兴战略下传统村落文化旅游设计 期货刊印辉少许
  • 力扣面试题 16.19. 水域大小(java DFS解法)

    Problem 面试题 16 19 水域大小 文章目录 题目描述 思路 解题方法 复杂度 Code 题目描述 思路 该问题可以归纳为一类 遍历二维矩阵 的题目 此类中的一部分题目可以利用 DFS 来解决 具体到本题目 该题目可以的写法大体不
  • Java 基础学习(十四)Map集合与Set集合

    1 Map集合 1 1 Map接口 1 1 1 Map接口概述 Map接口是一种双列集合 Map的每个元素都包含一个键对象Key和一个值对象Value 键对象和值对象之间存在对应关系 这种关系称为映射 Mapping Map接口中的元素 可
  • Android 12.0 Launcher3定制化之动态日历图标功能实现

    1 概述 在12 0的系统产品rom开发中 在Launcher3中的相关定制化功能中 对于一些产品要求需要动态日历图标功能 在日期改变的时候 日历图标也需要跟着改变 所以需要自定义日历图标 监听日历改变的广播 收到日期改变的广播后 刷新日历
  • 计算机网络 网络层下 | IPv6 路由选择协议,P多播,虚拟专用网络VPN,MPLS多协议标签

    文章目录 5 IPv6 5 1 组成 5 2 IPv6地址 5 3 从IPv4向IPv6过渡 5 3 1 双协议栈 5 3 2 隧道技术 6 因特网的
  • 每天都很煎熬,领导派的活太难,真的想跑路了

    每天都很煎熬 领导派的活太难 真的想跑路了 人在江湖身不由己 无论是领导的亲信还是团队的边缘 都可能遇到这种情况 不得不干一件特别难以推进的事情 茫然无措 不知如何推进 每天陷入焦虑和自我怀疑中 这种事情一般有一些共同特点 结果和目标极其模
  • 创建个人网站(二)前端主页设计和编写一(太阳移动)

    前言 以下内容纯纯当乐子来看就行 知识分享一下这样设计的原因 想看正文直接见下一节 为什么创建个人网站一之后几天没有动静了呢 一个是家里有事实在比较忙 第二个原因是没想到主页要设计成什么样 知道前两天问我姐什么样的主页比较炫酷 我们得出的结
  • 网工内推 | 美团、中通快递,网络运维,最高30K*15薪

    01 美团 招聘岗位 网络运维开发工程师 职责描述 1 负责新零售业务门店 仓库网络的日常运维 故障处理 应急响应 保障网络及相关业务的稳定运行 处理突发事件 对疑难问题进行跟踪并最终解决 2 负责新零售业务门店 仓库网络的建设和运维 对涉
  • 软件测试/测试开发|Ubuntu系统常用文件管理命令详解

    前言 Ubuntu是一种广泛使用的Linux操作系统 提供了丰富而强大的文件管理命令 使用户能够通过命令行轻松管理文件和目录 本文将介绍一些常用的Ubuntu文件管理命令 帮助用户更好地理解和利用系统资源 ls 列出目录内容 ls 选项 目
  • vue-springboot美食菜谱分享平台in9c2

    1 以人为本原则 这个厨房达人美食分享平台 它的本质是为人们服务 是希望其可以实现并满足广大使用者对科技时代下的厨房达人美食分享平台的憧憬与向往 如何解放需要人工才能完成的部分和提升使用者的使用观感是最为基础的也是最重要的 它的设计宗旨就是
  • 基于FPGA的简易BPSK和QPSK

    1 框图 2 顶层 3 m generator M序列的生成 输出速率为500Kbps 4 S2P是串并转换模块 将1bit的m序列转换到50M时钟下的2bit M序列数据 就有4个象限 5 my pll是生成256M的时钟作为载波 因为s
  • 线上环境如何正确配置 Django 的 DEBUG?

    Author rab Django Version 3 2 Python Version 3 9 目录 前言 一 DEBUG True 二 DEBUG False 三 页面异常解决 总结
  • 全网最全pytest大型攻略,单元测试学这就够了

    pytest 是一款以python为开发语言的第三方测试 主要特点如下 比自带的 unittest 更简洁高效 兼容 unittest框架 支持参数化 可以更精确的控制要测试的测试用例 丰富的插件 已有300多个各种各样的插件 也可自定义扩