基于Pytest+Requests+Allure实现接口自动化测试

一、整体结构

框架组成:pytest+requests+allure

设计模式:

关键字驱动

项目结构:

工具层:api_keyword/

参数层:params/

用例层:case/

数据驱动:data_driver/

数据层:data/

逻辑层:logic/

二、具体步骤及代码

1、工具层

将get、post等常用行为进行二次封装。

代码(api_key.py)如下:

import allure
import json
import jsonpath
import requests# 定义一个关键字类
class ApiKey:# 将get请求行为进行封装@allure.step("发送get请求")def get(self, url, params=None, **kwargs):return requests.get(url=url, params=params, **kwargs)# 将post请求行为进行封装@allure.step("发送post请求")def post(self, url, data=None, **kwargs):return requests.post(url=url, data=data, **kwargs)# 由于接口之间可能相互关联,因此下一个接口需要上一个接口的某个返回值,此处采用jsonpath对上一个接口返回的值进行定位并取值@allure.step("获取返回结果字典值")def get_text(self, data, key):# json数据转换为字典json_data = json.loads(data)# jsonpath取值value = jsonpath.jsonpath(json_data, '$..{0}'.format(key))return value[0]

其中引用allure.step()装饰器进行步骤详细描述,使测试报告更加详细。

使用jsonpath对接口的返回值进行取值。

2、数据层

数据采用yaml文件。

代码(user.yaml)如下:

-user:username: adminpassword: '123456'msg: successtitle: 输入正确账号、密码,登录成功
-user:username: admin1password: '1234561'msg: 用户名或密码错误title: 输入错误账号1、密码1,登录失败
-user:username: admin2password: '1234562'msg: 用户名或密码错误title: 输入错误账号2、密码2,登录失败

其中title是为了在用例进行时动态获取参数生成标题。

3、数据驱动层

对数据进行读写。

代码(yaml.driver.py)如下:

import yamldef load_yaml(path):file = open(path, 'r', encoding='utf-8')data = yaml.load(file, Loader=yaml.FullLoader)return data

4、参数层

参数层存放公共使用的参数,在使用时对其进行调用。

代码(allParams.py)如下:

'''规则:全局变量使用大写字母表示
'''# 地址
URL = 'http://39.98.138.157:'# 端口
PORT = '5000'

5、逻辑层

用例一:进行登录的接口请求,此处登录请求在yaml文件里设置了三组不同的数据进行请求。

用例二:进行个人查询的接口请求,此处需要用到登录接口返回的token值。

用例三、进行添加商品到购物车的接口请求,此处需要用到登录接口返回的token值以及个人查询接口返回的openid、userid值

用例四、进行下单的接口请求,此处需要用到登录接口返回的token值以及个人查询接口返回的openid、userid、cartid值

注意:由于多数接口需要用到登录接口返回的token值,因此封装一个conftest.py定义项目级前置fixture,在整个项目只执行一次,可以在各个用例中进行调用(其他共用参数也可以采取类似前置定义)。同时由于此处定义的项目级fixture,因此可以将初始化工具类ak = ApiKey()也放入其中。

代码(conftest.py)如下:

from random import randomimport allure
import pytestfrom pytest_demo_2.api_keyword.api_key import ApiKey
from pytest_demo_2.params.allParams import *def pytest_collection_modifyitems(items):"""测试用例收集完成时,将收集到的item的name和nodeid的中文显示在控制台上"""for item in items:item.name = item.name.encode("utf-8").decode("unicode_escape")item._nodeid = item.nodeid.encode("utf-8").decode("unicode_escape")# 项目级fix,整个项目只初始化一次
@pytest.fixture(scope='session')
def token_fix():# 初始化工具类ak = ApiKey()with allure.step("发送登录接口请求,并获取token,整个项目只生成一次"):# 请求接口# url = 'http://39.98.138.157:5000/api/login'url = URL + PORT + '/api/login'# 请求参数userInfo = {'username': 'admin','password': '123456'}# post请求res = ak.post(url=url, json=userInfo)# 获取tokentoken = ak.get_text(res.text, 'token')# 验证代码,验证token只生成一次token_random = random()return ak, token, res, token_random

其中也包含了防止中文乱码,加入了pytest_collection_modifyitems(函数)。

设置好conftest后,就可以应用在逻辑层里面了。

代码(shopingApi.py)如下:

import pytest
import allure
from pytest_demo_2.api_keyword.api_key import ApiKey
from pytest_demo_2.params.allParams import *class ApiCase():# 登录逻辑def params_login(self, userdata):# 动态获取参数生成标题allure.dynamic.title(userdata['title'])# 初始化工具类ak = ApiKey()# 请求接口url = URL + PORT + '/api/login'# 请求参数userInfo = {'username': userdata['user']['username'],'password': userdata['user']['password']}res = ak.post(url=url, json=userInfo)with allure.step("接口返回信息校验及打印"):print("/api/login登录接口请求响应信息")print(res.text)# 获取响应结果msg = ak.get_text(res.text, 'msg')print(msg)# 断言assert msg == userdata['msg']def params_getuserinfo(self, token_fix):# 从fix中获取预置的工具类和token,所有返回值都需要接收ak, token, res, token_random01 = token_fixwith allure.step("发送个人查询接口请求"):url = URL + PORT + '/api/getuserinfo'headers = {'token': token}res1 = ak.get(url=url, headers=headers)with allure.step("接口返回信息校验及打印"):print("/api/getuserinfo个人用户查询接口请求响应信息")print(res1.text)# print("验证的random值,测试用")# print(token_random01)name = ak.get_text(res1.text, 'nikename')# 断言assert "风清扬" == namereturn res1def params_addcart(self, token_fix):# 从fix中获取预置的工具类和token# 所有返回都要获取,不然会报错ak, token, res, token_random01 = token_fixwith allure.step("调用getuserinfo接口获取返回信息"):res1 = self.params_getuserinfo(token_fix)with allure.step("发送添加商品到购物车请求"):# 添加商品到购物车,基于token、userid、openid、productidurl = URL + PORT + '/api/addcart'hd = {"token": token}data = {"userid": ak.get_text(res1.text, 'userid'),"openid": ak.get_text(res1.text, 'openid'),"productid": 8888}# 发送请求res2 = ak.post(url=url, headers=hd, json=data)with allure.step("接口返回信息校验及打印"):print("/api/addcart添加商品到购物车请求响应信息")print(res2.text)# print("验证的random值,测试用")# print(token_random01)result = ak.get_text(res2.text, 'result')assert 'success' == resultreturn res2def params_createorder(self, token_fix):ak, token, res, token_random01 = token_fixwith allure.step("调用addcart接口获取返回信息"):res1 = self.params_addcart(token_fix)with allure.step("发送下单请求"):url = URL + PORT + '/api/createorder'# 从项目级fix中获取tokenhd = {"token": token}# 从添加商品到购物车接口中获取userid,openid,cartiddata = {"userid": ak.get_text(res1.text, 'userid'),"openid": ak.get_text(res1.text, 'openid'),"productid": 8888,"cartid": ak.get_text(res1.text, 'cartid')}res2 = ak.post(url=url, headers=hd, json=data)with allure.step("接口返回信息校验及打印"):print("/api/createorder下单请求响应信息")print(res2.text)# print("验证的random值,测试用")# print(token_random01)result = ak.get_text(res1.text, 'result')assert 'success' == result

6、用例层

调用逻辑层进行用例管理和数据传输。

代码(test_Tree.py)如下:

import allure
import pytest
from pytest_demo_2.data_driver import yaml_driver
from pytest_demo_2.logic.shopingApi import ApiCase@allure.epic("shopXo电商平台接口-接口测试")
class TestTree():# 初始化用例库actions1 = ApiCase()@allure.feature("01.登陆")@allure.story("02.一般场景")@pytest.mark.parametrize('userdata', yaml_driver.load_yaml('./data/user.yaml'))def test_case01(self, userdata):self.actions1.params_login(userdata)@allure.feature("02.个人查询")@allure.story("01.典型场景")@allure.title("个人查询")def test_case02(self, token_fix):self.actions1.params_getuserinfo(token_fix)@allure.feature("03.添加商品到购物车")@allure.story("01.典型场景")@allure.title("添加商品到购物车")def test_case03(self, token_fix):self.actions1.params_addcart(token_fix)@allure.feature("04.下单")@allure.story("01.典型场景")@allure.title("下单")def test_case04(self, token_fix):self.actions1.params_createorder(token_fix)

7、运行

代码(main_run.py)如下:

import os
import pytestdef run():pytest.main(['-v', './case/test_Tree.py','--alluredir', './result', '--clean-alluredir'])os.system('allure serve result')# os.system('allure generate ./result/ -o ./report_allure/ --clean')if __name__ == '__main__':run()

8、结果

图片

最后感谢每一个认真阅读我文章的人,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:

这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你! 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/170067.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

AtCoder Beginner Contest 330 题解

目录 A - Counting PassesB - Minimize Abs 1C - Minimize Abs 2D - Counting LsE - Mex and Update A - Counting Passes 原题链接 题目描述 给定N个数和一个整数L,输出大于等于L的数的个数。 public static void solve() throws IOException{int n readInt(), m…

Blender快捷键总结

以下是一些Blender中常用的快捷键: 移动、旋转和缩放: G:移动选中的对象、顶点、边或面 R:旋转选中的对象、顶点、边或面 S:缩放选中的对象、顶点、边或面 Shift D:复制选中的对象 选择和编辑&#xff…

2311skia,01渲染架构

一,渲染层级 从渲染流程上分,Skia可分为如下三个层级: 1,指令层:SkPicture,SkDeferredCanvas->SkCanvas 这一层决定要绘图的操作,绘图操作的预变换矩阵,当前裁剪区域,在哪些层上绘图,层的生成与合并. 2,解析层:SkBitmapDevice->SkDraw->SkScan,SkDraw1Glyph::Proc 这…

基于51单片机的人体追踪可控的电风扇系统

**单片机设计介绍, 基于51单片机超声波测距汽车避障系统 文章目录 一 概要概述硬件组成工作原理优势应用场景总结 二、功能设计设计思路 三、 软件设计原理图 五、 程序六、 文章目录 一 概要 # 基于51单片机的人体追踪可控的电风扇系统介绍 概述 该系统是基于51…

AI 视频 | Stable Video Diffusion 来了!(附体验地址)

1. 介绍 11 月 21 日,Stability AI 推出了 Stable Video Diffusion,这是 Stability AI 的第一个基于图像模型 Stable Diffusion 的生成式视频基础模型。 目前 Stability AI 已经在 GitHub 上开源了 Stable Video Diffusion 的代码,在 Huggin…

c语言刷题12周(1~5)

输入年月日,显示这一天是这一年的第几天,保证输入日期合法。 题干输入年月日,显示这一天是这一年的第几天,保证输入日期合法。输入样例2022 1 1 2022 12 31 2024 12 31 2022 4 5输出样例2022-1 2022-365 2024-366 2022-9…

【数据结构实验】图(二)将邻接矩阵存储转换为邻接表存储

文章目录 1. 引言2. 邻接表表示图的原理2.1 有向权图2.2 无向权图2.3 无向非权图2.1 有向非权图 3. 实验内容3.1 实验题目(一)数据结构要求(二)输入要求(三)输出要求 3.2 算法实现 4. 实验结果 1. 引言 图是…

node.js解决输出中文乱码问题

个人简介 👨🏻‍💻个人主页:九黎aj 🏃🏻‍♂️幸福源自奋斗,平凡造就不凡 🌟如果文章对你有用,麻烦关注点赞收藏走一波,感谢支持! 🌱欢迎订阅我的…

shell脚本循环语句

目录 一. 循环语句 1. 循环条件 2. 循环次数 3. 循环命令区别 4. for 循环 ①. 第一种语法 ②. 第二种语法 5. while 循环 6. until 循环 二. 跳出循环 1. break 结束循环 2. continue 结束循环 3. exit 结束循环 三. 补充 1. 偶数的表示 2. 奇数的表示 一. 循环…

TOGAF —体系结构治理

本章提供了架构治理的框架和指南。 3.1 引言 本部分介绍治理的性质和治理级别。 3.1.1 企业内部的治理级别 架构治理是管理企业架构和其他架构的实践和方向 并在企业范围内进行控制。 架构治理通常不是孤立地运作,而是在治理结构的层次结构中运作,其中, 特别是在大型企…

【测试开发工程师】TestNG测试框架零基础入门(上)

哈喽大家好,我是小浪。那么今天是一期基于JavaTestNG测试框架的入门教学的博客,从只会手工测试提升到自动化测试,这将对你的测试技术提升是非常大的,有助于我们以后在找工作、面试的时候具备更大的竞争力~ 文章目录 一、什么是T…

香港证监会再点名,三家“可疑虚拟资产平台交易平台”!

JPEX事件之后,香港证监会通过对可疑平台进行曝光,以提升虚拟资产交易平台牌照申请的透明性。最近,香港证监会于官网再点名三家“可疑虚拟资产平台交易平台”。截至目前,香港证监会在官网共点名9家可疑虚拟资产交易平台。 据了解&a…

【数据结构实验】图(一)Warshall算法(求解有向图的可达矩阵)

文章目录 1. 引言2. Warshall算法原理2.1 初始化可及矩阵2.2 迭代更新可及矩阵 3. 实验内容3.1 实验题目(一)输入要求(二)输出要求 3.2 算法实现 4. 实验结果 1. 引言 Warshall算法是一种用于求解有向图的可达矩阵的经典算法。该算…

一文带你深入理解MySQL的三大连接方式:左连接、右连接和内连接!

个人网站 本文首发于公众号小肖学数据分析 在MySQL中,常见的三种关联方式是左连接(left join)、右连接(right join)和内连接(inner join)。 本文我将详细介绍这三种关联方式,并提…

用Python进行数据分析:探索性数据分析的实践与技巧(文末送书)

🤵‍♂️ 个人主页:艾派森的个人主页 ✍🏻作者简介:Python学习者 🐋 希望大家多多支持,我们一起进步!😄 如果文章对你有帮助的话, 欢迎评论 💬点赞&#x1f4…

死锁是什么?死锁是如何产生的?如何破除死锁?

1. 死锁是什么 多个线程同时被阻塞,它们中的一个或者全部都在等待某个资源被释放。由于线程被无限期地阻塞,因此程序不可能正常终止。 2. 死锁的三种典型情况 一个线程, 一把锁, 是不可重入锁, 该线程针对这个锁连续加锁两次, 就会出现死锁. 两个线程…

通过JMeter压测结果来分析Eureka多种服务下线机制后的服务感知情况

文章目录 前言1. Eureka-Server的设计2. EurekaRibbon感知下线服务机制3.服务调用接口压测模型4.Eureka几种服务下线的方式4.1强制下线压测 4.2 发送delete()请求压测 4.3 调用DiscoveryManager压测 4. 三方工具Actuator 总结 前言 上文末尾讲到了Eurek…

Java 8 lambda的一个编译bug

最近利用github action向Maven中央仓库发布企业微信SDK时会失败,从日志中发现是系统资源耗尽了,日志如下: [INFO] Changes detected - recompiling the module! :dependency [INFO] Compiling 35 source files with javac [debug target 8] …

MySQL索引 Error1071

MySQL创建索引错误 Error 1071:Specified key was too long 目录 1. 起因2. 使用环境3. 查找问题的原因4. 解决方案5. 另外的解决方案6. 前缀索引 起因 事情的起因在一次生产环境部署启动服务,发现启动服务时间特别长,超过了5分钟的时间。于是&#…

isis基础大全学习案例

R1配置: isis 1 is-level level-2 //本区域只启用level-2级别 cost-style wide //默认为narrow窄度量,开销只能最大63,并且不能打tag,wide宽度量的tlv和narrow不匹配,不能相互计算路由,两边都要改。 netwo…