pytest接口自动化测试框架搭建

article/2025/8/29 11:54:18

文章目录

  • 一. 背景
  • 二. 基础环境
  • 三. 项目结构
  • 四、框架解析
    • 4.1 接口数据文件处理
    • 4.2 封装测试工具类
    • 4.3 测试用例代码编写
    • 4.4 测试用例运行生成报告

一. 背景

Pytest目前已经成为Python系自动化测试必学必备的一个框架,网上也有很多的文章讲述相关的知识。最近自己也抽时间梳理了一份pytest接口自动化测试框架,因此准备写文章记录一下,做到尽量简单通俗易懂,当然前提是基本的python基础已经掌握了。如果能够对新学习这个框架的同学起到一些帮助,那就更好了~

二. 基础环境

语言:python 3.8
编译器:pycharm
基础:具备python编程基础
框架:pytest+requests+allure
框架代码下载地址:已上传至网盘,vx搜索关注我的公众号【程序员杨叔】,点击“测试资料”菜单,文章里面有具体下载地址~

三. 项目结构

项目结构图如下:
在这里插入图片描述
每一层具体的含义如下图:
在这里插入图片描述
测试报告如下图:
在这里插入图片描述

四、框架解析

4.1 接口数据文件处理

在这里插入图片描述
框架中使用草料二维码的get和post接口用于demo测试,比如:
get接口:https://cli.im/qrcode/getDefaultComponentMsg
返回值:{“code”:1,“msg”:“”,“data”:{xxxxx}}

数据文件这里选择使用Json格式,文件内容格式如下,test_http_get_data.json:

{"dataItem": [{"id": "testget-1","name": "草料二维码get接口1","headers":{"Accept-Encoding":"gzip, deflate, br"},"url":"/qrcode/getDefaultComponentMsg","method":"get","expectdata": {"code": "1"}},{"id": "testget-2","name": "草料二维码get接口2","headers":{},"url":"/qrcode/getDefaultComponentMsg","method":"get","expectdata": {"code": "1"}}]
}

表示dataitem下有两条case,每条case里面声明了id, name, header, url, method, expectdata。如果是post请求的话,case中会多一个parameters表示入参,如下:

{"id":"testpost-1","name":"草料二维码post接口1","url":"/Apis/QrCode/saveStatic","headers":{"Content-Type":"application/x-www-form-urlencoded","Accept-Encoding":"gzip, deflate, br"},"parameters":{"info":11111,"content":11111,"level":"H","size":500,"fgcolor":"#000000","bggcolor":"#FFFFFF","transparent":"false","type":"text","codeimg":1},"expectdata":{"status":"1","qrtype":"static"}
}

为了方便一套脚本用于不同的环境运行,不用换了环境后挨个儿去改数据文件,比如
测试环境URL为:https://testcli.im/qrcode/getDefaultComponentMsg
生产环境URL为:https://cli.im/qrcode/getDefaultComponentMsg
因此数据文件中url只填写后半段,不填域名。然后config》global_config.py下设置全局变量来定义域名:

# 配置HTTP接口的域名,方便一套脚本用于多套环境运行时,只需要改这里的全局配置就OK
CAOLIAO_HTTP_POST_HOST = "https://cli.im"
CAOLIAO_HTTP_GET_HOST = "https://nc.cli.im"

utils文件夹下,创建工具类文件:read_jsonfile_utils.py, 用于读取json文件内容:

import json
import osclass ReadJsonFileUtils:def __init__(self, file_name):self.file_name = file_nameself.data = self.get_data()def get_data(self):fp = open(self.file_name,encoding='utf-8')data = json.load(fp)fp.close()return datadef get_value(self, id):return self.data[id]@staticmethoddef get_data_path(folder, fileName):BASE_PATH = os.path.dirname(os.path.dirname(os.path.realpath(__file__)))data_file_path = os.path.join(BASE_PATH, folder, fileName)return data_file_pathif __name__ == '__main__':opers = ReadJsonFileUtils("..\\resources\\test_http_post_data.json")#读取文件中的dataItem,是一个list列表,list列表中包含多个字典dataitem=opers.get_value('dataItem')print(dataitem)

运行结果如下:
在这里插入图片描述

4.2 封装测试工具类

utils文件夹下,
除了上面提到的读取Json文件工具类:read_jsonfile_utils.py,
还有封装request 请求的工具类:http_utils.py
从Excel文件中读取数据的工具类:get_excel_data_utils.py(虽然本次框架中暂时未采用存放接口数据到Excel中,但也写了个工具类,需要的时候可以用)
在这里插入图片描述
http_utils.py内容:

import requests
import jsonclass HttpUtils:@staticmethoddef http_post(headers, url, parameters):print("接口请求url:" + url)print("接口请求headers:" + json.dumps(headers))print("接口请求parameters:" + json.dumps(parameters))res = requests.post(url, data=parameters, headers=headers)print("接口返回结果:"+res.text)if res.status_code != 200:raise Exception(u"请求异常")result = json.loads(res.text)return result@staticmethoddef http_get(headers, url):req_headers = json.dumps(headers)print("接口请求url:" + url)print("接口请求headers:" + req_headers)res = requests.get(url, headers=headers)print("接口返回结果:" + res.text)if res.status_code != 200:raise Exception(u"请求异常")result = json.loads(res.text)return result

get_excel_data_utils.py内容:

import xlrd
from xlrd import xldate_as_tuple
import datetimeclass ExcelData(object):'''xlrd中单元格的数据类型数字一律按浮点型输出,日期输出成一串小数,布尔型输出0或1,所以我们必须在程序中做判断处理转换成我们想要的数据类型0 empty,1 string, 2 number, 3 date, 4 boolean, 5 error'''def __init__(self, data_path, sheetname="Sheet1"):#定义一个属性接收文件路径self.data_path = data_path# 定义一个属性接收工作表名称self.sheetname = sheetname# 使用xlrd模块打开excel表读取数据self.data = xlrd.open_workbook(self.data_path)# 根据工作表的名称获取工作表中的内容self.table = self.data.sheet_by_name(self.sheetname)# 根据工作表的索引获取工作表的内容# self.table = self.data.sheet_by_name(0)# 获取第一行所有内容,如果括号中1就是第二行,这点跟列表索引类似self.keys = self.table.row_values(0)# 获取工作表的有效行数self.rowNum = self.table.nrows# 获取工作表的有效列数self.colNum = self.table.ncols# 定义一个读取excel表的方法def readExcel(self):# 定义一个空列表datas = []for i in range(1, self.rowNum):# 定义一个空字典sheet_data = {}for j in range(self.colNum):# 获取单元格数据类型c_type = self.table.cell(i,j).ctype# 获取单元格数据c_cell = self.table.cell_value(i, j)if c_type == 2 and c_cell % 1 == 0:  # 如果是整形c_cell = int(c_cell)elif c_type == 3:# 转成datetime对象date = datetime.datetime(*xldate_as_tuple(c_cell, 0))c_cell = date.strftime('%Y/%d/%m %H:%M:%S')elif c_type == 4:c_cell = True if c_cell == 1 else Falsesheet_data[self.keys[j]] = c_cell# 循环每一个有效的单元格,将字段与值对应存储到字典中# 字典的key就是excel表中每列第一行的字段# sheet_data[self.keys[j]] = self.table.row_values(i)[j]# 再将字典追加到列表中datas.append(sheet_data)# 返回从excel中获取到的数据:以列表存字典的形式返回return datasif __name__ == "__main__":data_path = "..\\resources\\test_http_data.xls"sheetname = "Sheet1"get_data = ExcelData(data_path, sheetname)datas = get_data.readExcel()print(datas)

4.3 测试用例代码编写

testcases文件夹下编写测试用例:
在这里插入图片描述
test_caoliao_http_get_interface.py内容:

import logging
import allureimport pytest
from utils.http_utils import HttpUtils
from utils.read_jsonfile_utils import ReadJsonFileUtils
from config.global_config import CAOLIAO_HTTP_GET_HOST@pytest.mark.httptest
@allure.feature("草料二维码get请求测试")
class TestHttpInterface:# 获取文件相对路径data_file_path = ReadJsonFileUtils.get_data_path("resources", "test_http_get_data.json")# 读取测试数据文件param_data = ReadJsonFileUtils(data_file_path)data_item = param_data.get_value('dataItem')  # 是一个list列表,list列表中包含多个字典"""@pytest.mark.parametrize是数据驱动;data_item列表中有几个字典,就运行几次caseids是用于自定义用例的名称"""@pytest.mark.parametrize("args", data_item, ids=['测试草料二维码get接口1', '测试草料二维码get接口2'])def test_caoliao_get_demo(self, args, login_test):# 打印用例ID和名称到报告中显示print("用例ID:{}".format(args['id']))print("用例名称:{}".format(args['name']))print("测试conftest传值:{}".format(login_test))logging.info("测试开始啦~~~~~~~")res = HttpUtils.http_get(args['headers'], CAOLIAO_HTTP_GET_HOST+args['url'])# assert断言,判断接口是否返回期望的结果数据assert str(res.get('code')) == str(args['expectdata']['code']), "接口返回status值不等于预期"

test_caoliao_http_post_interface.py内容:

import pytest
import logging
import allure
from utils.http_utils import HttpUtils
from utils.read_jsonfile_utils import ReadJsonFileUtils
from config.global_config import CAOLIAO_HTTP_POST_HOST# pytest.ini文件中要添加markers = httptest,不然会有warning,说这个Mark有问题
@pytest.mark.httptest
@allure.feature("草料二维码post请求测试")
class TestHttpInterface:# 获取文件相对路径data_file_path = ReadJsonFileUtils.get_data_path("resources", "test_http_post_data.json")# 读取测试数据文件param_data = ReadJsonFileUtils(data_file_path)data_item = param_data.get_value('dataItem') #是一个list列表,list列表中包含多个字典"""@pytest.mark.parametrize是数据驱动;data_item列表中有几个字典,就运行几次caseids是用于自定义用例的名称"""@pytest.mark.parametrize("args", data_item, ids=['测试草料二维码post接口1','测试草料二维码post接口2'])def test_caoliao_post_demo(self, args):# 打印用例ID和名称到报告中显示print("用例ID:{}".format(args['id']))print("用例名称:{}".format(args['name']))logging.info("测试开始啦~~~~~~~")res = HttpUtils.http_post(args['headers'], CAOLIAO_HTTP_POST_HOST+args['url'], args['parameters'])# assert断言,判断接口是否返回期望的结果数据assert str(res.get('status')) == str(args['expectdata']['status']), "接口返回status值不等于预期"assert str(res.get('data').get('qrtype')) == str(args['expectdata']['qrtype']), "接口返回qrtype值不等于预期"

企业中的系统接口,通常都有认证,需要先登录获取token,后续接口调用时都需要认证token。因此框架需要能处理在运行用例前置和后置做一些动作,所以这里用到了conftest.py文件,内容如下:

import logging
import tracebackimport pytest
import requests"""
如果用例执行前需要先登录获取token值,就要用到conftest.py文件了
作用:conftest.py 配置里可以实现数据共享,不需要import导入 conftest.py,pytest用例会自动查找
scope参数为session,那么所有的测试文件执行前执行一次
scope参数为module,那么每一个测试文件执行前都会执行一次conftest文件中的fixture
scope参数为class,那么每一个测试文件中的测试类执行前都会执行一次conftest文件中的fixture
scope参数为function,那么所有文件的测试用例执行前都会执行一次conftest文件中的fixture"""# 获取到登录请求返回的ticket值,@pytest.fixture装饰后,testcase文件中直接使用函数名"login_ticket"即可得到ticket值
@pytest.fixture(scope="session")
def login_ticket():header = {'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'}params = {"loginId": "username","pwd": "xxxxxx",}url = 'http://testxxxxx.xx.com/doLogin'logging.info('开始调用登录接口:{}'.format(url))res = requests.post(url, data=params, headers=header, verify=False)  # verify:忽略https的认证try:ticket = res.headers['Set-Cookie']except Exception as ex:logging.error('登录失败!接口返回:{}'.format(res.text))traceback.print_tb(ex)logging.info('登录成功,ticket值为:{}'.format(ticket))return ticket#测试一下conftest.py文件和fixture的作用
@pytest.fixture(scope="session")
def login_test():print("运行用例前先登录!")# 使用yield关键字实现后置操作,如果上面的前置操作要返回值,在yield后面加上要返回的值# 也就是yield既可以实现后置,又可以起到return返回值的作用yield "runBeforeTestCase"print("运行用例后退出登录!")

由于用例中用到了@pytest.mark.httptest给用例打标,因此需要创建pytest.ini文件,并在里面添加markers = httptest,不然会有warning,说这个Mark有问题。并且用例中用到的日志打印logging模板也需要在pytest.ini文件中增加日志配置。pytest.ini文件内容如下:
在这里插入图片描述

[pytest]
markers =httptest: run http interface testdubbotest: run dubbo interface testlog_cli = true
log_cli_level = INFO
log_cli_format = %(asctime)s - %(pathname)s[line:%(lineno)d] - %(levelname)s: %(message)s
log_cli_date_format=%Y-%m-%d %H:%M:%S
log_format = %(asctime)s - %(pathname)s[line:%(lineno)d] - %(levelname)4s: %(message)s
log_date_format=%Y-%m-%d %H:%M:%S

4.4 测试用例运行生成报告

运行方式:
Terminal窗口,进入到testcases目录下,执行命令:运行某一条case:pytest test_caoliao_http_post_interface.py
运行所有case: pytest
运行指定标签的case:pytest -m httptest运行并打印过程中的详细信息:pytest -s test_caoliao_http_post_interface.py
运行并生成pytest-html报告:pytest test_caoliao_http_post_interface.py --html=../testoutput/report.html
运行并生成allure测试报告:
1. 先清除掉testoutput/result文件夹下的所有文件
2. 运行case,生成allure文件:pytest --alluredir ../testoutput/result
3. 根据文件生成allure报告:allure generate ../testoutput/result/ -o ../testoutput/report/ --clean
4. 如果不是在pycharm中打开,而是直接到report目录下打开index.html文件打开的报告无法展示数据,需要双击generateAllureReport.bat生成allure报告;

pytest-html报告:
在这里插入图片描述
generateAllureReport.bat文件位置:
在这里插入图片描述
文件内容:

allure open report/

Allure报告:
在这里插入图片描述
框架中用到的一些细节知识点和问题,如:

conftest.py和@pytest.fixture()结合使用
pytest中使用logging打印日志
python中获取文件相对路径的方式
python虚拟环境
pytest框架下Allure的使用

我会在后续写文章再介绍。另外框架同样适用于dubbo接口的自动化测试,只需要添加python调用dubbo的工具类即可,后续也会写文章专门介绍。

==============================================================================
以上就是本次的全部内容,都看到这里了,如果对你有帮助,麻烦点个赞+收藏+关注,一键三连啦~

欢迎关注我的vx公众号:程序员杨叔,各类文章都会第一时间在上面发布,持续分享全栈测试知识干货,你的支持就是作者更新最大的动力~


http://chatgpt.dhexx.cn/article/w8qaYnP4.shtml

相关文章

接口自动化测试项目实战

目录 1. 什么是接口测试 2. 基本流程 2.1 示例接口 3. 需求分析 4. 用例设计 5. 脚本开发 5.1 相关lib安装 5.2 接口调用 5.3 结果校验 5.4 执行测试 5.5 发送邮件报告 6. 结果分析 7. 完整脚本 8、总结 1. 什么是接口测试 顾名思义,接口测试是对系统或…

接口自动化测试用例详解

phpunit 接口自动化测试系列 Post接口自动化测试用例 Post方式的接口是上传接口,需要对接口头部进行封装,所以没有办法在浏览器下直接调用,但是可以用Curl命令的-d参数传递接口需要的参数。当然我们还以众筹网的登录接口为例,讲…

接口自动化测试流程

文章目录 接口自动化测试的基本流程一、需求分析二、自动化接口挑选三、设计自动化测试用例四、搭建自动化测试环境五、设计自动化执行框架六、编写代码七、执行用例八、测试报告 接口自动化测试的基本流程 1、需求分析 2、挑选需要做自动化测试的功能接口 3、设计测试用例 4、…

接口自动化测试框架

本文介绍一个接口自动化测试框架。 Pythonunittestrequests 实现结果:读取Excel接口测试用例并执行,输出测试报告。 框架脑图 如图,各个模块及作用如上。 处理数据库 db_funcs用来处理数据库,实现数据库数据的读取操作。&…

基于 python 的接口自动化测试,让你轻松掌握接口自动化

目录 目录 一、简介 ​编辑二、引言 三、环境准备 四、测试接口准备 接口信息 五、编写接口测试 六、优化 封装接口调用 一、简介 本文从一个简单的登录接口测试入手,一步步调整优化接口调用姿势; 然后简单讨论了一下接口测试框架的要点&#xff…

接口自动化测试工程化——了解接口测试

什么是接口测试 接口测试也是一种功能测试 我理解的接口测试,其实也是一种功能测试,只是平时大家说的功能测试更多代指 UI 层面的功能测试,而接口测试更偏向于服务端层面的功能测试。 接口测试的目的 测试左移,尽早介入测试&#…

接口自动化测试【完整版】

1. 什么是接口测试 顾名思义,接口测试是对系统或组件之间的接口进行测试,主要是校验数据的交换,传递和控制管理过程,以及相互逻辑依赖关系。其中接口协议分为HTTP,WebService,Dubbo,Thrift,Socket等类型,测试类型又主…

接口自动化测试

接口自动化测试 1.基础知识1.接口测试原理2.接口测试点及用例设计方法3.接口测试要点4.常见HTTP状态码5.HTTP基础知识6.接口自动化测试工具7.token 2.抓包工具1.chrom抓包2.Fiddler抓包(PC端,手机端)1.原理2.下载安装3.抓取数据信息说明&…

接口自动化测试(Python+Requests+Unittest)

(1)接口自动化测试的意义、前后端分离思想 接口自动化测试的优缺点: 优点: 测试复用性。 维护成本相对UI自动化低一些。 为什么UI自动化维护成本更高? 因为前端页面变化太快,而且UI自动化比较耗时(比如等待页面元素的…

接口自动化测试,一看就会

引言 与UI相比,接口一旦研发完成,通常变更或重构的频率和幅度相对较小。因此做接口自动化的性价比更高,通常运用于迭代版本上线前的回归测试中。 手工做接口测试,测试数据和参数都可以由测试人员手动填写和更新。 因此我们在考…

Pytorch损失函数(nn.L1Loss、nn.SmoothL1Loss、nn.MSELoss 、nn.CrossEntropyLoss、nn.NLLLoss)

文章目录 nn.L1Lossnn.SmoothL1Lossnn.MSELossnn.CrossEntropyLossnn.NLLLoss 损失函数,是编译一个神经网络模型必须的两个参数之一,另一个是优化器。损失函数是指用于计算标签值和预测值之间差异的函数,常见的有多种损失函数可供选择&#x…

链表L->next

2020/10/26更新 #include <iostream> using namespace std;struct MyStruct {int data 0;MyStruct* next; }; int main() {MyStruct* L;MyStruct head ,firstNode;head.data 1;firstNode.data 2;head.next &firstNode;L &head;cout << L->next-&g…

L多样化

为了解决同质性攻击和背景知识攻击所带来的隐私泄露&#xff0c;Machanavajjhala等人提出了L-多样性(l-diversity)模型。简单来说&#xff0c;就是在公开的数据中&#xff0c;每一个等价类里的敏感属性必须具有多样性&#xff0c;即L-多样性保证每一个等价类里&#xff0c;敏感…

L'Hospital法则

1.LHospital法则利用柯西中值定理证明 2.该法则适用f与g同时为无穷小量或g为无穷大量的情况 3.注意该法则只给出了充分不必要条件 今天我们介绍一种函数求极限的方法——LHospital法则。它主要用在求解"待定型"极限 1 首先阐述定理: 我们证明这个定理: 微分中值定理…

李沐老师 d2l库画图在 pycharm 动态显示问题(已解决)

修改 d2l.Animator 类的部分代码 1. 在 Animator 类中的 add() 方法中添加两行代码 plt.draw() plt.pause(0.001) 2. 在 Animator 添加 show() 方法 def show(self):display.display(self.fig) 3. 在需要显示的时候调用 animator.show() 结果

L298N——真正的玄学驱动板

总结 L298N 电机就是不转问题&#xff01;&#xff01;&#xff01; 目录 前言 1、共地问题 PWM 不能直接控制电机转动&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;别把PWM直接接电机&#xff01;&…

QMC5883L说明文档

个人博客&#xff1a;http://brainware360.cn/ QMC5883L源于Honeywell的HMC5883L&#xff0c;是一款表面贴装的集成了信号处理电路的三轴磁性传感器&#xff0c;应用场景主要包括罗盘、导航、无人机、机器人和手持设备等一些高精度的场合。 外形如下图所示&#xff1a; Figur…

安装 d2l

1.下载whl &#xff1a;https://www.cnpython.com/pypi/d2l/download 下载后放在哪个目录下都行 2.切换到刚刚下载的这个文件的所在目录&#xff0c;打开cmd 3.如上图标示&#xff0c;打开cmd&#xff0c;在cmd窗口中输入安装命令&#xff1a;pip install whl文件的名字 4.…

L298n

一时兴起&#xff0c;把之前的东西整一下。L298N资料&#xff08;非原创&#xff0c;贴出来便于日后查阅&#xff0c;或许能给给初学者一点帮助&#xff09;

L9110电机驱动电路

我需要设计一个驱动电路去驱动这样一个电机。电机的两个端子的输入波形如图&#xff1a; 驱动的波形为50%占空比的方波&#xff0c;频率为5Hz&#xff0c;额定电压12V。 单片机IO口可以输出3.3V/5V的PWM波&#xff0c;不过电压不够&#xff0c;驱动能力也不够&#xff0c;电机…