Python之Pytest测试框架
off999 2025-05-21 15:45 26 浏览 0 评论
前言:
和大家简单介绍一下如何利用Python的Pytest测试框架来进行UI、接口测试,更多学习内容和资料可以查看:
https://docs.qq.com/doc/DWmxXaVdLUlVyTkZL
1 pytest安装
pip install pytest
复制
- pytest是一个非常成熟的单元框架
- pytest可以和其他框架整合,例如requests,httprunner
- pytest可以实现测试用例跳过、失败重跑
- pytest可以和allure生成非常美观的测试报告
- pytest可以和jenkins持续集成
- pytest有非常强大的插件,并且你这些插件能实现很多的实用的操作
pytest-xdist 测试用例分布式执行,多cpu并发
pytest-ordering 用于改变测试用例的执行顺序
pytest-rerunfailures 用于失败重跑
allure-pytest 用于生成美观的测试报告
复制
2 规范
用Pytest写用例时候,一定要按照下面的规则去写,否则不符合规则的测试用例是不会执行的
1.测试方法必须以test开头
2.测试类必须以Test开头,并且不能有init方法
复制
class TestClass:
def test_one(self):
x = "this"
assert "h" in x
def test_two(self):
x = "hello"
assert "h" in x
def two(self):
x = "hello"
assert "0" in x
复制
3 断言支持
assert xx :判断 xx 为真
assert not xx :判断 xx 不为真
assert a in b :判断 b 包含 a
assert a == b :判断 a 等于 b
assert a != b :判断 a 不等于 b
复制
assert dict["code"]=='0'
assert response.json()['message']=='success'
复制
@pytest.mark.skip 跳过某个单元测试
复制
4 参数化支持(重点)
@pytest.mark.parametrize("username",['admin','张三','admin'])
def test1(username):
print(username)
复制
参数化中文乱码支持 根目录新建conftest.py
def
pytest_collection_modifyitems(items):
"""
该方法解决
"""
for item in items:
item.name = item.name.encode("utf-8").decode("unicode_escape")
item._nodeid = item.nodeid.split('::')[0] +'::'+ item.nodeid.split('::')[1].encode("utf-8").decode("unicode_escape")
复制
pytest会默认读取conftest.py里面的所有fixture
conftest.py 文件名称是固定的,不能改动
不同目录可以有自己的conftest.py,一个项目中可以有多个conftest.py
测试用例文件中不需要手动import conftest.py,pytest会自动查找
复制
#多参数 集合(元组/集合/set)
@pytest.mark.parametrize("username,pwd",[('admin','123456'),('test1','123')])
def test2(username,pwd):
print(username,pwd)
@pytest.mark.parametrize("username,pwd",[['admin', '123456'], ['test1', '123456']])
def test3(username,pwd):
print(username,pwd)
复制
集合(字典)
test_data = [
{
'case': '登入成功',
'usr': 'admin', # 正常登入
'psw': '123456'
},
{
'case': '账号不存在',
'usr': 'admin1', # 账号不存在
'psw': '123456'
},
{
'case': '密码错误',
'usr': 'admin', # 密码错误
'psw': '12345'
},
{
'case': '账号或密码为空',
'usr': '', # 账号或密码为空
'psw': ''
},
]
@pytest.mark.parametrize('param', test_data)
def test4(param):
usr = param.get('usr')
psw = param.get('psw')
print(f'usr: {usr} , psw: {psw}')
复制
ids 优化结果提示
@pytest.mark.parametrize('param', test_data, ids=[data.get('case') for data in test_data])
def test4(param):
usr = param.get('usr')
psw = param.get('psw')
print(f'usr: {usr} , psw: {psw}')
复制
[data.get('case') for data in test_data] 是推导式
复制
推导式是Python的一种独有特性。是Python的可以从一个数据序列构建另一个新的数据序列的结构体
不用推导式
list=[]
for item in test_data:
list.append(item['case'])
复制
多层嵌套结构
# 笛卡尔积,组合数据
data_1 = [1, 2, 3]
data_2 = ['x', 'y']
@pytest.mark.parametrize('a', data_1)
@pytest.mark.parametrize('b', data_2)
def test_parametrize_1(a, b):
print(f'笛卡尔积 测试数据为 : {a},{b}')
复制
5 参数化csv整合
id.csv
id,message
1,success
3,success
4,success
5,success
6,success
7,success
-1,skuId不合法
复制
path =os.path.join(get_project_path(),"files","id.csv")
#list
def get_csv_data1():
list=[]
with open(path, encoding="utf-8") as f:
csv_reader = csv.reader(f)
next(csv_reader)
for line in csv_reader:
list.append(line)
return list
@pytest.mark.parametrize('id,message', get_csv_data1())
def test_Get1(id,message):
url=host+"/pinter/com/getSku?id="+id
@pytest.mark.parametrize('id,message', csv_list,ids=[i[0] for i in csv_list])
#字典类型
def get_csv_data2():
list=[]
with open(path, encoding="utf-8") as f:
csv_reader = csv.DictReader(f)
for line in csv_reader:
list.append(line)
return list
@pytest.mark.parametrize('dic', get_csv_data2())
def test_Get2(dic):
print(dic)
....
assert dict["code"]=='0'
assert response.json()['message']=='success'
复制
6 失败重试
平时在做接口测试的时候,经常会遇到网络抖动或者环境问题导致测试用例运行失败,而这个并不是我们想要的结果,我们想要重新运行失败的测试用例
pip install pytest-rerunfailures
pip show pytest-rerunfailures
复制
@pytest.mark.flaky(reruns=3) #失败最大试3次
@pytest.mark.flaky(reruns=3,reruns_delay=2) #失败最大3次 ,每次延迟2秒执行
复制
7 执行顺序插件
pytest执行顺序一般是按照从上往下的顺序执行的,如果使用过程中想要指定顺序,可以使用order标签
pip install pytest-ordering
pip show pytest-ordering
复制
@pytest.mark.run(order=2)
复制
8 报表插件pytest-allure
pip install allure-pytest
pip show allure-pytest
pytest 脚本名称 --alluredir ./report
解析 report
需要
allure-commandline-2.13.3.zip 工具
mac
https://blog.csdn.net/u014015919/article/details/90292041
allure --version
allure serve report
单独main方法运行 (需要单独建py文件)
pytest.main(['-s','test_allure02.py','--clean-alluredir','--alluredir=allure-results'])
os.system(r"allure generate -c -o allure-report")
复制
-q: 安静模式, 不输出环境信息
-s: 详细打印
-v: 更丰富信息模式, 输出更详细的用例执行信息
8.1 allure扩展注解(了解)
@allure.parent_suite,@allure.suite,@allure.sub_suite对应的是
1,2,3级目录
@allure.parent_suite('我是parent_suite')
@allure.suite('我是suite')
@allure.sub_suite('我是sub_suite')
@allure.feature('测试参数化')
@allure.id('我是id')
@allure.title('我是title')
@allure.link('https://www.baidu.com/', LinkType.LINK, '我是link')
@allure.label('我是label')
@allure.issue('https://www.baidu.com/', '我是issue')
@allure.description('我是description')
@allure.severity('我是severity')
9 并行执行
多进程
pip install pytest-xdis
模拟1000测试用例
data_1 = [x for x in range(10)]
data_2 = [x for x in range(100)]
@pytest.mark.parametrize('a', data_1)
@pytest.mark.parametrize('b', data_2)
def test_parametrize_1(a, b):
#sleep(0.01)
print(f'笛卡尔积 测试数据为 : {a},{b}')
复制
10 fixture特性
@pytest.fixture()
def get_token():
url = host +'/pinter/bank/api/login2'
#表单参数
data = {
'userName':'admin',
'password':'1234'
}
resp = requests.post(url=url,data=data)
resp_json = resp.json()
print(resp_json)
token = jsonpath.jsonpath(resp_json, '$.data')[0] # 使用jsonpath从响应结果中提取
return token
def test_query2(get_token):
url = host + '/pinter/bank/api/query2?userName=admin' # 接口地址
headers = {
'testfan-token':get_token
}
resp = requests.get(url=url,headers=headers)
status_code = resp.status_code #获取响应状态码
print('响应状态码:{}'.format(status_code))
text = resp.text #获取响应内容,结果类型是字符串
print('响应内容:{}'.format(text))
复制
相关推荐
- 全网第一个讲清楚CPK如何计算的Step by stepExcel和Python同时实现
-
在网上搜索CPK的计算方法,几乎全是照搬教材的公式,在实际工作做作用不大,甚至误导人。比如这个又比如这个:CPK=min((X-LSL/3s),(USL-X/3s))还有这个,很规范的公式,也很清晰很...
- [R语言] R语言快速入门教程(r语言基础操作)
-
本文主要是为了从零开始学习和理解R语言,简要介绍了该语言的最重要部分,以快速入门。主要参考文章:R-TutorialR语言程序的编写需要安装R或RStudio,通常是在RStudio中键入代码。但是R...
- Python第123题:计算直角三角形底边斜边【PythonTip题库300题】
-
1、编程试题:编写一个程序,找出已知面积和高的直角三角形的另外两边(底边及斜边)。定义函数find_missing_sides(),有两个参数:area(面积)和height(高)。在函数内,计算另外...
- Tensor:Pytorch神经网络界的Numpy
-
TensorTensor,它可以是0维、一维以及多维的数组,你可以将它看作为神经网络界的Numpy,它与Numpy相似,二者可以共享内存,且之间的转换非常方便。但它们也不相同,最大的区别就是Numpy...
- python多进程编程(python多进程进程池)
-
forkwindows中是没有fork函数的,一开始直接在Windows中测试,直接报错importosimporttimeret=os.fork()ifret==0:...
- 原来Python的协程有2种实现方式(python协程模型)
-
什么是协程在Python中,协程(Coroutine)是一种轻量级的并发编程方式,可以通过协作式多任务来实现高效的并发执行。协程是一种特殊的生成器函数,通过使用yield关键字来挂起函数的执行...
- ob混淆加密解密,新版大众点评加密解密
-
1目标:新版大众点评接口参数_token加密解密数据获取:所有教育培训机构联系方式获取难点:objs混淆2打开大众点评网站,点击教育全部,打开页面,切换到mobile模式,才能找到接口。打开开发者工具...
- python并发编程-同步锁(python并发和并行)
-
需要注意的点:1.线程抢的是GIL锁,GIL锁相当于执行权限,拿到执行权限后才能拿到互斥锁Lock,其他线程也可以抢到GIL,但如果发现Lock仍然没有被释放则阻塞,即便是拿到执行权限GIL也要立刻...
- 10分钟学会Python基础知识(python基础讲解)
-
看完本文大概需要8分钟,看完后,仔细看下代码,认真回一下,函数基本知识就OK了。最好还是把代码敲一下。一、函数基础简单地说,一个函数就是一组Python语句的组合,它们可以在程序中运行一次或多次运行。...
- Python最常见的170道面试题全解析答案(二)
-
60.请写一个Python逻辑,计算一个文件中的大写字母数量答:withopen(‘A.txt’)asfs:count=0foriinfs.read():ifi.isupper...
- Python 如何通过 threading 模块实现多线程。
-
先熟悉下相关概念多线程是并发编程的一种方式,多线程在CPU密集型任务中无法充分利用多核性能,但在I/O操作(如文件读写、网络请求)等待期间,线程会释放GIL,此时其他线程可以运行。GIL是P...
- Python的设计模式单例模式(python 单例)
-
单例模式,简单的说就是确保只有一个实例,我们知道,通常情况下类其实可以有很多实例,我们这么来保证唯一呢,全局访问。如配置管理、数据库连接池、日志处理器等。classSingleton: ...
- 更安全的加密工具:bcrypt(bcrypt加密在线)
-
作为程序员在开发工作中经常会使用加密算法,比如,密码、敏感数据等。初学者经常使用md5等方式对数据进行加密,但是作为严谨开发的程序员,需要掌握一些相对安全的加密方式,今天给大家介绍下我我在工作中使用到...
- 一篇文章搞懂Python协程(python协程用法)
-
前引之前我们学习了线程、进程的概念,了解了在操作系统中进程是资源分配的最小单位,线程是CPU调度的最小单位。按道理来说我们已经算是把cpu的利用率提高很多了。但是我们知道无论是创建多进程还是创建多线...
- Python开发必会的5个线程安全技巧
-
点赞、收藏、加关注,下次找我不迷路一、啥是线程安全?假设你开了一家包子铺,店里有个公共的蒸笼,里面放着刚蒸好的包子。现在有三个顾客同时来拿包子,要是每个人都随便伸手去拿,会不会出现混乱?比如第一个顾...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- python计时 (73)
- python安装路径 (56)
- python类型转换 (93)
- python进度条 (67)
- python吧 (67)
- python字典遍历 (54)
- python的for循环 (65)
- python格式化字符串 (61)
- python静态方法 (57)
- python列表切片 (59)
- python面向对象编程 (60)
- python 代码加密 (65)
- python串口编程 (60)
- python读取文件夹下所有文件 (59)
- java调用python脚本 (56)
- python操作mysql数据库 (66)
- python获取列表的长度 (64)
- python接口 (63)
- python调用函数 (57)
- python多态 (60)
- python匿名函数 (59)
- python打印九九乘法表 (65)
- python赋值 (62)
- python异常 (69)
- python元祖 (57)