一、pytest结合数据驱动-yaml
1、什么是数据驱动?
数据驱动就是数据的改变从而驱动自动化测试的执行,最终引起测试结果的改变。简单来说,就是参数化的应用。数据量小的测试用例可以使用代码的参数化来实现数据驱动,数据量大的情况下建议大家使用一种结构化的文件(例如 yaml,json 等)来对数据进行存储,然后在测试用例中读取这些数据
2、yaml 文件介绍
1)对象:键值对的集合,用冒号 “:” 表示,在python中为字典
2)数组:一组按次序排列的值,前加 “-”,在python中为列表
3)纯量:单个的、不可再分的值,包括字符串、布尔值、整数、浮点数、Null、时间、日期
# {"languages":["PHP","Java","Python"], "book":{"Python入门":{"price": 55.5, "author": "Lily", "available": True, "repertory": 20, "date": "2018-02-17"}, "Java入门": {"price": 60, "author": "Lily", "available": "False", "repertory": "Null", "date": "2018-05-11"}}}
languages:
- PHP
- Java
- Python
book:
Python入门: # 书籍名称
price: 55.5
author: Lily
available: True
repertory: 20
date: 2018-02-17
Java入门:
price: 60
author: Lily
available: False
repertory: Null
date: 2018-05-11
3、查看 yaml 文件
1)pycharm
2)txt 记事本
4、读取 yaml 文件
1)工程目录结构
①data 目录:存放 yaml 测试数据文件data.yaml
②func 目录:存放被测函数文件operation.py
③testcase 目录:存放测试用例文件test_add.py
2)安装:pip install pyyaml
3)方法:yaml.safe_load(f)
# yaml测试数据文件内容
# [[1,1,2],[3,6,9],[100,200,300]]
-
- 1
- 1
- 2
-
- 3
- 6
- 9
-
- 100
- 200
- 300
# operation.py 文件内容
# 被测方法,相加功能
def my_add(x, y):
result = x + y
return result
# test_add.py 文件内容
import pytest
import yaml
from testing.func.operation import my_add
# 读取yaml文件
def get_data():
"""
获取json数据
:return: 返回数据的结构:[[1, 1, 2], [3, 6, 9], [100, 200, 300]]
"""
# 如果yaml文件中有中文,必须要加上encoding="utf-8"
with open('../data/data.yaml', 'r', encoding="utf-8") as f:
data = yaml.safe_load(f)
return data
class TestWithYAML:
@pytest.mark.parametrize('x,y,expected', get_data()) # 测试数据调用get_data()方法读取出来的数据,需要添加数据时只需在yaml文件即可
def test_add(self, x, y, expected):
assert my_add(int(x), int(y)) == int(expected)
二、pytest结合数据驱动-excel
1、安装:pip install openpyxl
2、工程目录结构
①data 目录:存放 excel 测试数据文件params.xlsx
②func 目录:存放被测函数文件operation.py
③testcase 目录:存放测试用例文件test_excel.py
# operation.py 文件内容
# 被测方法,相加功能
def my_add(x, y):
result = x + y
return result
# test_excel.py 测试用例文件内容
import openpyxl
import pytest
from testing.func.operation import my_add
def get_excel():
# 获取工作簿
book = openpyxl.load_workbook('../data/params.xlsx')
# 获取活动行(非空白的)
sheet = book.active
# 提取数据,格式:[[1, 2, 3], [3, 6, 9], [100, 200, 300]]
values = []
for row in sheet:
line = []
for cell in row:
line.append(cell.value)
values.append(line)
return values
class TestWithEXCEL:
@pytest.mark.parametrize('x,y,expected', get_excel())
def test_add(self, x, y, expected):
assert my_add(int(x), int(y)) == int(expected)
三、pytest结合数据驱动-csv
1、csv文件介绍
1)csv:逗号分隔值,是 Comma-Separated Values 的缩写
2)以纯文本形式存储数字和文本
3)文件由任意数目的记录组成
4)每行记录由多个字段组成
2、csv文件的使用
1)读取数据
①内置函数:open()
②内置模块:csv
③方法:csv.reader(iterable)
参数:iterable ,文件或列表对象
返回:迭代器,每次迭代会返回一行数据。
2)工程目录结构
①data 目录:存放csv 测试数据文件demo.csv
②func 目录:存放被测函数文件operation.py
③testcase 目录:存放测试用例文件test_csv.py
# csv文件内容
1,1,2
3,6,9
100,200,300
# operation.py 文件内容
# 被测方法,相加功能
def my_add(x, y):
result = x + y
return result
# 测试用例文件内容
import pytest
import csv
from testing.func.operation import my_add
# 读取csv文件内容
def get_csv():
"""
获取csv数据
:return: 返回数据的结构:[[1, 1, 2], [3, 6, 9], [100, 200, 300]]
"""
with open('../data/demo.csv', 'r') as file:
raw = csv.reader(file)
data = []
for line in raw:
data.append(line)
return data
class TestWithCSV:
@pytest.mark.parametrize('x,y,expected', get_csv())
def test_add(self, x, y, expected):
assert my_add(int(x), int(y)) == int(expected)
四、Pytest 结合数据驱动 json
1、json文件介绍
①json是JS对象,全是JavaScript Object Notation
②是一种轻量级的数据交换格式;json 结构
对象 {"key": value}
数组 [value1, value2 ...]
2、查看 json 文件
①pycharm
②txt 记事本
3、工程目录结构
①data 目录:存放json数据文件params.xlsx
②func 目录:存放被测函数文件operation.py
③testcase 目录:存放测试用例文件test_json.py
4、读取 json 文件
①内置函数 open()
②内置库 json
③方法:json.loads()
# json数据文件
{
"case1": [1, 1, 2],
"case2": [3, 6, 9],
"case3": [100, 200, 300]
}
# operation.py 文件内容
# 被测方法,相加功能
def my_add(x, y):
result = x + y
return result
# 测试用例文件内容
import pytest
import json
from testing.func.operation import my_add
# 读取json文件
def get_json():
"""
获取json数据
:return: 返回数据的结构:[[1, 1, 2], [3, 6, 9], [100, 200, 300]]
"""
with open('../data/param.json', 'r') as f:
data = json.loads(f.read())
return list(data.values())
class TestWithJSON:
@pytest.mark.parametrize('x,y,expected', [[1, 1, 2]])
def test_add(self, x, y, expected):
assert my_add(int(x), int(y)) == int(expected)
五、pytest测试用例生命周期管理
1、Fixture 用法
1)Fixture 特点及优势
①命令灵活:对于 setup,teardown,可以不起这两个名字
②数据共享:在 conftest.py 配置⾥写⽅法可以实现数据共享,不需要 import 导⼊。可以跨⽂件共享
③scope 的层次及神奇的 yield 组合相当于各种 setup 和 teardown
④实现参数化
2)Fixture 在自动化中的应用- 基本用法
①使用场景:测试⽤例执⾏时,有的⽤例需要登录才能执⾏,有些⽤例不需要登录。
②setup 和 teardown ⽆法满⾜。fixture 可以。默认 scope(范围)function
③步骤:
a.导⼊ pytest
b.在登录的函数上⾯加@pytest.fixture()
c.在要使⽤的测试⽅法中传⼊(登录函数名称),就先登录
d.不传⼊的就不登录直接执⾏测试⽅法。
import pytest
# 定义了登录的fixture,尽量避免以test开头
@pytest.fixture()
def login():
print("完成登录操作")
def test_search():
print("搜索")
def test_cart(login): # 传入定义了fixture的方法,不需要加括号,执行用例时就会优先执行登录操作
print("购物车")
def test_order(login): # 传入定义了fixture的方法,不需要加括号,执行用例时就会优先执行登录操作
print("下单功能")
2、Fixture 在自动化中的应用 - 作用域
【取值:范围:说明】
①function:函数级:每一个函数或方法都会调用
②class:类级别:每个测试类只运行一次
③module:模块级:每一个.py 文件调用一次
④package:包级:每一个 python 包只调用一次(暂不支持)
⑤session:会话级:每次会话只需要运行一次,会话内所有方法及类,模块都共享这个方法
# fixture的作用域
import pytest
@pytest.fixture(scope="class")
def login1():
print("完成登录操作")
def test_search1():
print("搜索")
def test_cart1(login1): # 传入定义了fixture的方法,不需要加括号,执行用例时就会优先执行登录操作
print("购物车")
def test_order1(login1): # 传入定义了fixture的方法,不需要加括号,执行用例时就会优先执行登录操作
print("下单功能")
class TestDemo:
def test_case1(self, login1):
print("case1")
def test_case2(self, login1):
print("case2")
3、Fixture 在自动化中的应用 - yield 关键字
1)场景:测试⽅法后销毁清除数据
2)解决:通过在 fixture 函数中加⼊ yield 关键字,yield 是调⽤第⼀次返回结果,第⼆次执⾏它下⾯的语句返回。
3)步骤:
①在@pytest.fixture(scope=module)。
②在登陆的⽅法中加 yield,之后加销毁清除的步骤
import pytest
@pytest.fixture
def login():
# setup操作
print("完成登录操作")
# yield相当于return,不添加返回默认为None,可以添加返回值,在yield后面添加,如返回token
token = "abc"
username = "hello"
yield token # 前面为setup操作,后面为teardown操作
# teardown操作
print("完成登出操作")
def test_search():
print("搜索")
def test_cart(login): # 传入定义了fixture的方法,不需要加括号,执行用例时就会优先执行登录操作
print(f"token:{login}") # 添加返回值时,token为login,拿到token值
print("购物车")
4、Fixture 在自动化中的应用 - 数据共享
1)场景:与其他测试⼯程师合作⼀起开发时,公共的模块要在不同⽂件中,要在⼤家都访问到的地⽅。
2)解决:使⽤ conftest.py 这个⽂件进⾏数据共享,并且他可以放在不同位置起着不同的范围共享作⽤。
3)前提:conftest ⽂件名是不能换的,放在项⽬下是全局的数据共享的地⽅
4)执⾏:系统执⾏到参数 login 时先从本模块中查找是否有这个名字的变量什么的,之后在 conftest.py 中找是否有。
5)步骤:将登陆模块带@pytest.fixture 写在 conftest.py文件
# conftest.py文件
import pytest
@pytest.fixture(scope="session")
def login():
# setup操作
print("完成登录操作")
# yield相当于return,不添加返回默认为None,可以添加返回值,在yield后面添加,如返回token
token = "abc"
username = "hello"
yield token # 前面为setup操作,后面为teardown操作
# teardown操作
print("完成登出操作")
@pytest.fixture()
def loginDB():
print("连接数据库")
yield
print("断开数据库连接")
# 新建测试用例文件
# 将登录操作放在conftest文件中,且方法需要定义为fixture,实现共享,可以直接调用
def test_cart(login):
print("购物车")
def test_order(login):
print("下单功能")
class TestDemo:
def test_case1(self, login, loginDB): # 可以添加多个fixture
print("case1")
def test_case2(self, login, loginDB):
print("case2")
5、Fixture在自动化中的应用-自动应用
1)场景:
①不想原测试⽅法有任何改动,或全部都⾃动实现⾃动应⽤,
②没特例,也都不需要返回值时可以选择⾃动应⽤
2)解决:使⽤ fixture 中参数 autouse=True 实现
3)步骤:在⽅法上⾯加 @pytest.fixture(autouse=True)
# conftest.py文件
@pytest.fixture(scope="session", autouse=True)
def login():
# setup操作
print("完成登录操作")
# yield相当于return,不添加返回默认为None,可以添加返回值,在yield后面添加,如返回token
token = "abc"
username = "hello"
yield token # 前面为setup操作,后面为teardown操作
# teardown操作
print("完成登出操作")
# 测试用例文件
import pytest
# conftest文件中登录操作,fixture方法添加参数autouse=True,其他文件调用不需添加login方法名称,可自动生效
def test_cart():
print("购物车")
def test_order():
print("下单功能")
6、Fixture在自动化中的应用-参数化
1)场景:测试离不开数据,为了数据灵活,⼀般数据都是通过参数传的
2)解决:fixture 通过固定参数 request 传递
3)步骤:
①在 fixture 中增加@pytest.fixture(params=[1, 2, 3, ‘linda’])
②在⽅法参数写 request,方法体里面使用 request.param 接收参数
import pytest
@pytest.fixture(params=[["hali", 123], ["bob", 123456]])
def login(request):
print(f"用户名和密码是:{request.param}")
return request.param
def test_case(login):
print(f"用户名和密码数据为:{login}"
六、pytest配置文件
1、pytest.ini是什么
①pytest.ini 是 pytest 的配置文件
②可以修改 pytest 的默认行为
③不能使用任何中文符号,包括汉字、空格、引号、冒号等等
2、pytest.ini作用
①修改用例的命名规则
②配置日志格式,比代码配置更方便
③添加标签,防止运行过程报警告错误
④指定执行目录
⑤排除搜索目录
新建一个pytest.ini文件,添加以下目录
[pytest]
;执行check_开头和 test_开头的所有的文件,后面一定要加*
python_files = check_* test_*
;执行所有的以Test和Check开头的类
python_classes = Test* Check*
;执行所有以test_和check_开头的方法
python_functions= test_* check_*
;-vs打印日志文件,--alluredir ./results 生成报告并存放在results中,执行用例时就不用再该命令了
addopts = -vs --alluredir ./results
;设置执行的路径
testpaths = testing testcase
;忽略某些文件夹/目录
norecursedirs = result logs datas test_demo*
3、日志配置:
1)注意:windows系统 需要把中文 注释去掉。
[pytest]
;日志开关 true false
log_cli = true
;日志级别
log_cli_level = info
;打印详细日志,相当于命令行加 -vs
addopts = --capture=no
;日志格式
log_cli_format = %(asctime)s [%(levelname)s] %(message)s (%(filename)s:%(lineno)s)
;日志时间格式
log_cli_date_format = %Y-%m-%d %H:%M:%S
;日志文件位置
log_file = ./log/test.log
;日志文件等级
log_file_level = info
;日志文件格式
log_file_format = %(asctime)s [%(levelname)s] %(message)s (%(filename)s:%(lineno)s)
;日志文件日期格式
log_file_date_format = %Y-%m-%d %H:%M:%S
七、pytest插件
1、pytest 插件分类
①外部插件:pip install 安装的插件
②本地插件:pytest自动模块发现机制,自己编写的插件(conftest.py 存放的)
③内置插件:代码内部的_pytest 目录加载
2、pytest 常用的插件
(1)pytest-ordering:控制用例的执行顺序
1)场景:对于集成测试,经常会有上下文依赖关系的测试用例。比如 10 个步骤,拆成 10 条 case,这时候能知道到底执行到哪步报错。用例默认执行顺序:自上而下执行
2)解决:可以通过 setup,teardown 和 fixture 来解决。也可以使用对应的插件。
3)安装:pip install pytest-ordering
4)用法:@pytest.mark.run(order=2)
5)注意:多个插件装饰器(>2)的时候,有可能会发生冲突
import pytest
@pytest.mark.run(order=2)
def test_case01():
print("case01")
@pytest.mark.run(order=1)
def test_case02():
print("case02")
(2)pytest-xdist:分布式并发执行测试用例
1)场景:假设有个报名系统,对报名总数统计,数据同时进行修改操作的时候有可能出现问题,需要模拟这个场景,需要多用户并发请求数据。
2)使用分布式并发执行测试用例。分布式插件:pytest-xdist
3)安装及运行: pip install pytest-xdist
4)注意: 用例多的时候效果明显,多进程并发执行,同时支持 allure
5)分布式执行测试用例原则
①用例之间是独立的,不要有依赖关系
②用例执行没有顺序,随机顺序都能正常执行
③每个用例都能重复运行,运行结果不会影响其他用例
6)运行命令
①pytest -n numcpus :numcpus 是指几核
②pytest -n auto :不知道是几核时,auto会自动去找空闲的cpu
(3)pytest-dependency:控制用例的依赖关系
(4)pytest-rerunfailures:失败重跑
(5)pytest-assume:多重较验
(6)pytest-random-order:用例随机执行
(7)pytest-html:测试报告
3、pytest 内置插件 hook 体系
1)pytest hook介绍
①是个函数,在系统消息触发时被系统调用
②自动触发机制
③Hook 函数的名称是确定的
④pytest 有非常多的勾子函数
⑤使用时直接编写函数体