百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程字典 > 正文

Python之Pytest测试框架(pytest框架原理)

toyiye 2024-08-31 02:52 5 浏览 0 评论

前言:

和大家简单介绍一下如何利用Python的Pytest测试框架来进行UI、接口测试,更多学习内容和资料可以查看:

https://docs.qq.com/doc/DWmxXaVdLUlVyTkZL

1 pytest安装

pip install pytest
复制

  1. pytest是一个非常成熟的单元框架
  2. pytest可以和其他框架整合,例如requests,httprunner
  3. pytest可以实现测试用例跳过、失败重跑
  4. pytest可以和allure生成非常美观的测试报告
  5. pytest可以和jenkins持续集成
  6. 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))
复制

相关推荐

# Python 3 # Python 3字典Dictionary(1)

Python3字典字典是另一种可变容器模型,且可存储任意类型对象。字典的每个键值(key=>value)对用冒号(:)分割,每个对之间用逗号(,)分割,整个字典包括在花括号({})中,格式如...

Python第八课:数据类型中的字典及其函数与方法

Python3字典字典是另一种可变容器模型,且可存储任意类型对象。字典的每个键值...

Python中字典详解(python 中字典)

字典是Python中使用键进行索引的重要数据结构。它们是无序的项序列(键值对),这意味着顺序不被保留。键是不可变的。与列表一样,字典的值可以保存异构数据,即整数、浮点、字符串、NaN、布尔值、列表、数...

Python3.9又更新了:dict内置新功能,正式版十月见面

机器之心报道参与:一鸣、JaminPython3.8的热乎劲还没过去,Python就又双叒叕要更新了。近日,3.9版本的第四个alpha版已经开源。从文档中,我们可以看到官方透露的对dic...

Python3 基本数据类型详解(python三种基本数据类型)

文章来源:加米谷大数据Python中的变量不需要声明。每个变量在使用前都必须赋值,变量赋值以后该变量才会被创建。在Python中,变量就是变量,它没有类型,我们所说的"类型"是变...

一文掌握Python的字典(python字典用法大全)

字典是Python中最强大、最灵活的内置数据结构之一。它们允许存储键值对,从而实现高效的数据检索、操作和组织。本文深入探讨了字典,涵盖了它们的创建、操作和高级用法,以帮助中级Python开发...

超级完整|Python字典详解(python字典的方法或操作)

一、字典概述01字典的格式Python字典是一种可变容器模型,且可存储任意类型对象,如字符串、数字、元组等其他容器模型。字典的每个键值key=>value对用冒号:分割,每个对之间用逗号,...

Python3.9版本新特性:字典合并操作的详细解读

处于测试阶段的Python3.9版本中有一个新特性:我们在使用Python字典时,将能够编写出更可读、更紧凑的代码啦!Python版本你现在使用哪种版本的Python?3.7分?3.5分?还是2.7...

python 自学,字典3(一些例子)(python字典有哪些基本操作)

例子11;如何批量复制字典里的内容2;如何批量修改字典的内容3;如何批量修改字典里某些指定的内容...

Python3.9中的字典合并和更新,几乎影响了所有Python程序员

全文共2837字,预计学习时长9分钟Python3.9正在积极开发,并计划于今年10月发布。2月26日,开发团队发布了alpha4版本。该版本引入了新的合并(|)和更新(|=)运算符,这个新特性几乎...

Python3大字典:《Python3自学速查手册.pdf》限时下载中

最近有人会想了,2022了,想学Python晚不晚,学习python有前途吗?IT行业行业薪资高,发展前景好,是很多求职群里严重的香饽饽,而要进入这个高薪行业,也不是那么轻而易举的,拿信工专业的大学生...

python学习——字典(python字典基本操作)

字典Python的字典数据类型是基于hash散列算法实现的,采用键值对(key:value)的形式,根据key的值计算value的地址,具有非常快的查取和插入速度。但它是无序的,包含的元素个数不限,值...

324页清华教授撰写【Python 3 菜鸟查询手册】火了,小白入门字典

如何入门学习python...

Python3.9中的字典合并和更新,了解一下

全文共2837字,预计学习时长9分钟Python3.9正在积极开发,并计划于今年10月发布。2月26日,开发团队发布了alpha4版本。该版本引入了新的合并(|)和更新(|=)运算符,这个新特性几乎...

python3基础之字典(python中字典的基本操作)

字典和列表一样,也是python内置的一种数据结构。字典的结构如下图:列表用中括号[]把元素包起来,而字典是用大括号{}把元素包起来,只不过字典的每一个元素都包含键和值两部分。键和值是一一对应的...

取消回复欢迎 发表评论:

请填写验证码