Python+Requests实现接口自动化测试(超详细的)

一般对于自动化的理解,有两种方式的自动化。

第一,不需要写代码,完全由工具实现,这种方式的工具一般是公司自己研发的,方便黑盒测试人员使用。这种工具的特点是学习成本低,方便使用,但是通用性不强,也就是换了一家公司,就很有可能无法使用之前的工具。

第二,需要自己写代码,在别人的框架下编写代码测试,或者是需要自己搭建自动化测试框架。这种方式对测试人员的代码要求高,学习成本高,但是通用性很强,去任何一家都可以用这套东西。
鉴于以上介绍,本文当然是介绍第二种方式了。【在文末有视频教程学习,也可以点击文末小卡片免费领取软件测试资料】

接口自动化测试,我们的目的是使用python进行接口测试,并完成输出测试报告。我们需要用到的东西有如下:python3,unittest,requests。

一、接口项目

我们使用的项目是发布会签到系统。总共有5个接口,虽然不多,但足够学习使用。

接口文档如下:

二、接口用例 

软件测试都需要写测试用例,不管你做的性能,自动化还是其它任何的测试工作。

真实的工作写接口的测试用例,可能考虑很多场景,如接口的功能(正常场景),接口的边界等价,接口的异常场景,接口参数组合,接口的性能等等。本文采用输出法分析,根据出参的不同设计出测试用例。详细用例参考如下:(用例太小看不清楚,可以查看原图,然后放大) 

三、代码阶段

3.1 框架的设计

我们使用unittest框架,case目录存放所有的测试用例,lib目录存放自己封装的一些代码,result目录存放测试结果和测试日志,runner.py是主程序。

3.2 主程序 runner.py

这个主程序跟之前的《selenium unittest实战》文章类似,不再详细介绍,不太一样的地方是使用一个logging模块。不知道大家有没有感受,测试接口的时候,想看完整的请求和响应,以便分析定位问题。

import unittest
import time
import os
import logging
from HTMLTestRunner import  HTMLTestRunner#获取项目的根目录
test_dir = os.path.join(os.getcwd())# 自动搜索项目根目录下的所有case,构造测试集;返回TestSuite对象
discover = unittest.defaultTestLoader.discover(test_dir, pattern='test_*.py')# 实例化TextTestRunner类
# runner = unittest.TextTestRunner(verbosity=2)now = time.strftime('%Y-%m-%d %H_%M_%S')  # 获取当前日期
result = test_dir+ '\\result\\'+now + '_result.html'  # 测试报告的完整路径
log = test_dir+'\\result\\'+now+'_log.txt'  #日志的完整路径logging.basicConfig(filename=log,level=logging.INFO,format = '%(asctime)s - %(name)s - %(levelname)s - %(message)s') #filename 日志文件路径 level 日志的级别 format 格式fp = open(result, 'wb')  # wb方式写入
runner = HTMLTestRunner(stream=fp, title='测试报告', description='aguest_master项目用例执行情况',verbosity=2)  #构造runner# 使用run()方法运行测试套件(即运行测试套件中的所有用例)
runner.run(discover)

3.3 测试用例和lib库

1) 由于总共只有5个接口,所以设计为5个代码文件,分别为:test_add_event,py,test_add_guest.py,test_get_event_list.py,test_get_guest_list.py,test_user_sign.py。

2)我们使用python的requests测试接口,这个库大名鼎鼎,而且官网还有中文。

官网网址:http://docs.python-requests.org/zh_CN/latest/

3)每个代码文件都是一个接口,接口的url地址是固定的,所以设计成类属性,方便后续测试用例使用。

4)每个测试用例都写明代码逻辑,方便以后调试。
5)如果遇到经常调用的东西,如获取最新发布会ID,获取添加发布会body数据,都封装成库。
6)最后根据出参的状态码断言是否成功
7)使用http://logging.info 记录每个测试用例的日志情况

添加发布会接口代码文件:test_add_event.py

import requests
import unittest
import logging
import addEventDataTemplate
import getNewID
from urllib import  parse  #使用requests发送post请求,body的汉字会进行url编码,即%xx形式。想看到原始body,需要使用parse.unquote进入url解码
class Test_addEvent(unittest.TestCase):'''添加发布会接口'''@classmethoddef setUpClass(cls):cls.url="http://127.0.0.1:8000/api/add_event/"@classmethoddef tearDownClass(cls):passdef setUp(self):passdef tearDown(self):passdef test_00(self):   #代码逻辑::获取当前最新发布会ID,设置入参,eid置空,发送post请求'''添加发布会-eid为空'''id=getNewID.getNewID()   #获取当前最新发布会IDdata=addEventDataTemplate.getEventData(id)  #获取添加发布会的数据模板data['eid']=''  #eid为空,即参数错误r=requests.post(self.url,data=data)status=r.json()['status']self.assertEqual(10021,status)logging.info(f"case:添加发布会,eid为空\n请求地址:{r.url}\t请求方式:{r.request.method}\n请求头:{r.request.headers}\n请求正文:{parse.unquote(r.request.body)}\n响应头:{r.headers}\n响应正文:{r.text}\n")def test_01(self):    #代码逻辑::获取当前最新发布会ID,设置入参,发送post请求'''添加发布会-成功'''id = getNewID.getNewID()  # 获取当前最新发布会IDdata = addEventDataTemplate.getEventData(id)#获取添加最新发布会的数据模板r=requests.post(self.url,data=data)status=r.json()['status']self.assertEqual(10000,status)logging.info(f"case:添加发布会,成功\n请求地址:{r.url}\t请求方式:{r.request.method}\n请求头:{r.request.headers}\n请求正文:{parse.unquote(r.request.body)}\n响应头:{r.headers}\n响应正文:{r.text}\n")def test_02(self):   #代码逻辑::获取当前最新发布会ID,ID-1即为发布会已经存在的ID(发布会ID是递增加1)'''添加发布会-发布会ID已存在'''id = getNewID.getNewID()  # 获取当前最新发布会IDdata=addEventDataTemplate.getEventData(id)#获取添加最新发布会的数据模板data['eid']=data['eid']-1  #最新模板ID减一即为重复IDr=requests.post(self.url,data=data)status = r.json()['status']self.assertEqual(10022, status)logging.info(f"case:添加发布会,发布会ID已存在\n请求地址:{r.url}\t请求方式:{r.request.method}\n请求头:{r.request.headers}\n请求正文:{parse.unquote(r.request.body)}\n响应头:{r.headers}\n响应正文:{r.text}\n")def test_03(self):   #代码逻辑::先新增发布会,再获取最新发布会ID,设置入参的name为重复。'''添加发布会-发布会标题已存在'''#新增发布会id = getNewID.getNewID()  # 获取当前最新发布会IDr=requests.post(self.url,data=addEventDataTemplate.getEventData(id))  #先新增一个发布会id = getNewID.getNewID()  # 获取当前最新发布会IDdata = addEventDataTemplate.getEventData(id)#获取添加最新发布会的数据模板data['name']=f'发布会测试标题{id}' #最新模板ID减一,标题即为重复r=requests.post(self.url,data=data)status = r.json()['status']self.assertEqual(10023,status)logging.info(f"case:添加发布会,发布会标题已存在\n请求地址:{r.url}\t请求方式:{r.request.method}\n请求头:{r.request.headers}\n请求正文:{parse.unquote(r.request.body)}\n响应头:{r.headers}\n响应正文:{r.text}\n")def test_04(self):   #代码逻辑::获取最新发布会ID,设置入参,开始时间:改为-,再提交请求'''添加发布会-发布会时间错误'''id = getNewID.getNewID()  # 获取当前最新发布会IDdata = addEventDataTemplate.getEventData(id)#获取添加最新发布会的数据模板data['start_time']=data['start_time'].replace(':','-')  #时间 : 改为 - ,即为时间错误r = requests.post(self.url, data=data)status = r.json()['status']self.assertEqual(10024,status)logging.info(f"case:添加发布会,发布会时间错误\n请求地址:{r.url}\t请求方式:{r.request.method}\n请求头:{r.request.headers}\n请求正文:{parse.unquote(r.request.body)}\n响应头:{r.headers}\n响应正文:{r.text}\n")
if __name__ == '__main__':unittest.main(verbosity=2)
lib库 getNewID.py:def getNewID():'''获取最新的(最大的)发布会编号id'''import sqlite3con=sqlite3.connect(r'D:\backup\guest2-master\db.sqlite3')cur=con.cursor()cur.execute("select max(id) from sign_event")new_id=cur.fetchone()new_id=new_id[0]cur.close()con.close()return new_id
lib库 addEventDataTemplate.py:import datetime
def getEventData(id):'''添加发布会 body模板'''startTime=(datetime.datetime.now()+datetime.timedelta(days=30)).strftime("%Y-%m-%d %H:%M:00")  #获得当前时间,并往后30天为发布会时间data={'eid':id+1,'name':f"发布会测试标题{id+1}",  #当前发布会编号加1'limit':100,    #默认值'status':1,    #默认值'address':'新街口金鹰',    #默认值'start_time':startTime  #%格式   Y-%m-%d %H:%M:00}return data

由于篇幅的原因,其它的代码省略。

最后的测试结果:

日志结果如下:

四、总结

在写代码的过程中,每个测试用例的代码逻辑非常重要,不管是什么逻辑,得保证每个测试用例代码都可以独立运行,不会产生耦合。还有在测试接口的时候,经常与数据库打交道,比如获取数据,判断测试结果等。

在这我为大家准备了一份软件测试视频教程(含面试、接口、自动化、性能测试等),就在下方,需要的可以直接去观看,也可以直接【点击文末小卡片免费领取资料文档】

软件测试视频教程观看处:

【2024最新版】Python自动化测试15天从入门到精通,10个项目实战,允许白嫖。。。

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

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

相关文章

高压放大器输出接法及其注意事项

高压放大器应用场景非常广泛,非常适用于半导体高压驱动、TFT产业高压驱动、各种高压工程等应用;也很适用当作音频信号产生器或函数波形产生器的波形放大使用。使用场景广泛,放大器的输出接法也多种,对于不同的放大器也有对应的输出…

C++补充内容--EasyX-UI界面

esay x 其他 地图打印(利用二维数组) 双缓冲 当我们绘制一张图 然后另一张图盖住前一张图的某个部分的时候 由于while的存在 会导致 两张图不停的闪烁 所以加入双缓冲可以解决这个问题 开启双缓冲 之后等待Flush或者End 才会进行图片的绘制 不然不会进行图片的绘制,这样就可…

东信免驱系列身份证阅读器串口通讯协议解析示例,适用于单片机、ARM等系统开发集成使用

完整的一次读卡流程包括: 身份证寻卡 > 身份证选卡 > 身份证读卡,三个步骤 缺一不可(见通讯协议)。 寻卡:EA EB EC ED 04 00 B0 B4 BB 返回:EA EB EC ED 05 00 00 B0 B5 BB 选卡:EA …

1.4 Unity协程

一、先说接口 接口是不能实例化的,想实例化接口,只能实例化继承了接口的类。 using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks;namespace InterfaceTest {interface IMyInterfa…

基于Springboot的旅游管理系统(有报告)。Javaee项目,springboot项目。

演示视频: 基于Springboot的旅游管理系统(有报告)。Javaee项目,springboot项目。 项目介绍: 采用M(model)V(view)C(controller)三层体系结构&…

使用STM32微控制器驱动LCD1602显示器

驱动LCD1602显示器是嵌入式系统常见的任务之一,而STM32微控制器因其灵活性和丰富的外设而成为了广泛采用的解决方案。在这篇文章中,我们将探讨如何使用STM32微控制器来驱动LCD1602显示器。我们将从STM32的GPIO配置、延时函数以及LCD1602的初始化和写入数…

MIT_线性代数笔记:第 24 讲 马尔可夫矩阵;傅里叶级数

目录 马尔可夫矩阵 Markov matrices傅里叶级数 Fourier series 本讲学习马尔可夫矩阵和傅里叶级数,两者是关于特征值和投影矩阵的应用。 马尔可夫矩阵 Markov matrices A [ 0.1 0.01 0.3 0.2 0.99 0.3 0.7 0 0.4 ] A \begin{bmatrix} 0.1 & 0.01 & 0.3 \\…

所有单片机使用的汇编语言是统一的吗?

所有单片机使用的汇编语言是统一的吗? 在开始前我有一些资料,是我根据网友给的问题精心整理了一份「单片机的资料从专业入门到高级教程」, 点个关注在评论区回复“888”之后私信回复“888”,全部无偿共享给大家!&…

十大性能测试工具

这篇关于“性能测试工具”的文章将按以下顺序让您了解不同的软件测试工具: 什么是性能测试? 为什么我们需要性能测试? 性能测试的优势 性能测试的类型 十大性能测试工具 什么是性能测试? 性能测试是一种软件测试,可确…

Origin无法使用主题管理器相关功能或报错:Err, Save Theme dialog error!

问题描述 在使用origin绘图时,往往需要进行大批量绘制同样类型的图。如果每个图都不断地去修改相关设置,无疑是浪费了许多宝贵的时间。为了提高绘图效率,了解到了主题管理器,可在“工具–主题管理器”找到。 然而,当我…

自然语言处理24-T5模型的介绍与训练过程,利用简单构造数据训练微调该模型,体验整个过程

大家好,我是微学AI,今天给大家介绍一下自然语言处理24-T5模型的介绍与训练过程,利用简单构造数据训练微调该模型,体验整个过程。在大模型ChatGPT发布之前,NLP领域是BERT,T5模型为主导,T5(Text-to-Text Transfer Transformer)是一种由Google Brain团队在2019年提出的自然…

ARM工控机Node-red使用教程

嵌入式ARM工控机Node-red安装教程 从前车马很慢书信很远,而现在人们不停探索“科技改变生活”。 智能终端的出现改变了我们的生活方式,钡铼技术嵌入式工控机协助您灵活布建能源管理、大楼自动化、工业自动化、电动车充电站等各种多元性IoT应用&#xff…

Linux 编译安装 Nginx

目录 一、前言二、四种安装方式介绍三、本文安装方式:源码安装3.1、安装依赖库3.2、开始安装 Nginx3.3、Nginx 相关操作3.4、把 Nginx 注册成系统服务 四、结尾 一、前言 Nginx 是一款轻量级的 Web 服务器、[反向代理]服务器,由于它的内存占用少&#xf…

【Pytorch】学习记录分享10——TextCNN用于文本分类处理

【Pytorch】学习记录分享10——PyTorchTextCNN用于文本分类处理 1. TextCNN用于文本分类2. 代码实现 1. TextCNN用于文本分类 具体流程: 2. 代码实现 # coding: UTF-8 import torch import torch.nn as nn import torch.nn.functional as F import numpy as np…

【机器学习:欧氏距离 】机器学习中欧氏距离的理解和应用

【机器学习:欧氏距离 】机器学习中欧氏距离的理解和应用 距离公式二维更高的维度点以外的物体属性欧几里得距离的平方概括历史 在数学中,欧氏距离’是指欧氏空间中任意两点之间的直线距离。这种距离可以通过应用勾股定理来计算,利用两点的笛卡…

再检查下这些测试思维面试题你都会了么?

创建坐席组的功能模块,如何进行测试用例设计? 解答: 功能测试,使用等价类划分法去分析创建坐席的每个输入项的有效及无效类,同步考虑边界值去设计对应的测试用例: 先进行冒烟测试,正常创建坐席…

操作系统期末复习知识点

目录 一.概论 1.操作系统的介绍 2.特性 3.主要功能 4.作用 二.进程的描述与控制 1.进程的定义 2.特性 3.进程的创建步骤 4.基本状态转化 5.PCB的作用 6.进程与线程的比较 三.进程同步 1.同步的概念(挺重要的) 2.临界区 3.管程和进程的区…

62.网游逆向分析与插件开发-游戏增加自动化助手接口-游戏公告类的C++还原

内容来源于:易道云信息技术研究院VIP课 上一个内容:游戏红字公告功能的逆向分析-CSDN博客 码云地址(master分支):https://gitee.com/dye_your_fingers/sro_-ex.git 码云版本号:0888e34878d9e7dd0acd08ef…

Redis第四讲——Redis的数据库结构、删除策略及淘汰策略

一、redis中的数据库 redis服务器将所有数据库都保存在服务器状态redis.h/redisServer结构的db数组中。db数组的每项都是一个redis.h/redisDb结构,而每个redisDb结构就代表一个数据库。在初始化服务器时,程序会根据服务器状态的dbnum属性来决定应该创建多…

outlook邮件群发单显技巧?群发怎么单显?

outlook邮件群发单显如何设置?QQ邮箱怎么群发单显? 在群发邮件时,如何让每个收件人只看到自己的名字,而不是其他人的名字,这就涉及到所谓的“单显”技巧。下面蜂邮EDM就为大家揭秘Outlook邮件群发单显的奥秘。 outlo…