接口测试--测试工具:rap2 接口文档解析
通过百度 OCR 工具识别 rap2 登录中的验证码,从而实现登录~那我们今天来实战解析 rap2 的接口数据,生成我们所需要的接口数据
实践上手
文档分析
1、我们先通过 F12 看看哪个接口是我们需要的?对~就是你了!

2、将接口数据复制到https://www.json.cn/,我们来看看接口数据结构,看看有哪些信息是需要的


3、通过分析 properties 中的数据,发现 pos=1 为请求头,pos=2 为路径参数,pos=3 为 body 参数,name 为字段名,value 为字段值,parentId=1,没有父节点;parentId!=-1,为父节点


代码解析
上一期中,有小伙伴反馈说,可以输入 cookies 不?那这次优化一下吧~
实现 rap2 登录并获取数据
@classmethod
def rap_data(cls, id, cookies = None):
"""
获取接口文档数据
:param id: 接口文档id
:param cookies: 非必填,登录的cookies
:return: 返回json数据
"""
params = {"id": id}
if cookies:
try:
cls.cookies = dict([i.split("=", 1) for i in cookies.split("; ")])
except Exception as e:
raise Exception('cookies格式不对,请重新输入!')
try:
res = requests.get(url=UrlConfig.RAP_PROJRCT_URL, params=params, cookies=cls.cookies).json()
except Exception as e:
raise Exception('cookies过期了,请重新输入!')
return res
else:
login_result = False
#判断是否登录成功
while not login_result:
login_result = Rap2Data.rap_login()
res = requests.get(url=UrlConfig.RAP_PROJRCT_URL, params=params, cookies=cls.cookies).json()
return res
解析关键代码展示:执行主入口
class Rap2Parse(object):
@classmethod
def api_data(cls, id, cookies=None, module=None):
json_obj = Rap2Data.rap_data(id, cookies)
errMsg = json_obj.get("errMsg")
#获取报错信息,将其抛出(权限不足、不存在接口文档,接口都会返回errMsg)
if errMsg:
errMsg = errMsg + f"请将接口文档授权给{Account.RAP['email']}!"
raise Exception(errMsg)
if module is None:
#模块名不传的话,默认获取所有模块数据
json_list = json_obj['data']['modules']
else:
#模块名传入,获取对应的模块数据
json_list = jsonpath(json_obj, f"$.data.modules[?(@.name=='{module}')]")
#校验传入的模块名是否存在
if not json_list:
raise Exception(f"{module}模块名不存在")
# 接口文档标题名,作为后续的文件名
file_name = json_obj.get('data').get('name')
#对标题名特殊处理
special_str = r"[\/\\\:\*\?\"\<\>\|]"
file_name = re.sub(special_str, "-", file_name)
data = []
#遍历模块数据
for i in json_list:
modules_data = {}
#模块名
modules_data['modules'] = i['name']
#模块描述
modules_data['description'] = i['description']
interfaces = []
#遍历interfaces接口数据
for i, dto in enumerate(i['interfaces']):
cases_data = {}
#接口id
cases_data['api_id'] = i + 1
#接口名
cases_data['title'] = dto['name']
#接口描述
cases_data['description'] = dto['description']
#接口路径
cases_data['url'] = dto['url']
#接口请求方法
cases_data['method'] = dto['method']
#获取接口的入参和出参
properties_list = jsonpath(dto['properties'], f"$.[?(@.scope=='request')]")
#接口请求头数据
cases_data['headerData'] = cls.parent_data(properties_list)[2]
#接口query参数
cases_data['queryData'] = cls.parent_data(properties_list)[1]
#接口body参数
cases_data['data'] = cls.parent_data(properties_list)[0]
interfaces.append(cases_data)
modules_data['cases_data'] = interfaces
data.append(modules_data)
return data, file_name
组装参数
@classmethod
def join_data(cls, json_data, properties, name, type, value, parentId, api_id):
"""
组装参数
:param json_data: 字典值(存数据用)
:param properties: 接口的入参和出参数据
:param name: 接口名
:param type: 字段类型
:param value: 字段值
:param parentId: 父id
:param api_id: 所有的参数id
:return: json_data
"""
#特殊字符替换
name = name.replace(".","")
#没有父id,直接塞进json_data
if parentId == -1:
v = cls.change_value(type, value)
json_data[name] = v
#有父id且父id在api_id中
elif parentId != -1 and parentId in api_id:
#根据父id获取父id的数据
parent_dto = jsonpath(properties, f"$.[?(@.id=={parentId})]")
if not parent_dto:
raise Exception("接口文档维护有误")
parent_dto = parent_dto[0]
#如果父的类型为数组
if parent_dto['type'] == 'Array':
#找到父的key
parent_array = cls.search_value(json_data, parent_dto['name'])
v = cls.change_value(type, value)
#将子数据挂载在父下面
parent_array[0][name] = v
# 如果父的类型为对象
elif parent_dto['type'] == 'Object':
# 找到父的key
parent_object = cls.search_value(json_data, parent_dto['name'])
v = cls.change_value(type, value)
# 将子数据挂载在父下面
parent_object[name] = v
else:
raise Exception("父数据类型不支持")
else:
pass
# raise Exception("父数据类型不支持")
return json_data
递归找对应的 key 数据
@classmethod
def search_value(cls, data, key):
"""
递归找到所在key的数据
:param data: 数据
:param key: 需要找的key
:return:
"""
key = key.replace(".", "")
json_data = JsonSearch(object=data)
path = json_data.search_first_path(key)
path_str = cls.path_str(path)
value = jsonpath(data, path_str)[0]
return value
解析后数据展示

项目地址:https://github.com/JokerChat/ApiDocParse
后续扩展
- 将数据生成 Excel 格式的接口自动化用例
- 将数据生成 yaml 格式接口自动化用例
- 将数据生成生成 jmeter 脚本 (jmx)
- 基于上述功能,做成前端页面,供其他人使用
接口测试--测试工具:rap2 接口文档解析的更多相关文章
- api接口测试工具和接口文档管理工具
api接口测试工具和接口文档管理工具 1.postman(https://www.getpostman.com) Postman 是一个很强大的 API调试.Http请求的工具.她可是允许用户发送任何 ...
- PCB WebAPI 接口测试工具与接口文档生成
我们自己写WebAPI或调用对方系统提供的WebAPI时,测试WebAPI接口工具用哪些工具呢. 这里将3种WebAPI常用到的工具使用说明.主要是讲对第3种WebApiTestClientWebAp ...
- [工具]Swagger-api接口文档描述
摘要 工作中经常的用到webapi,之前都是提供的使用postman模拟请求的截图,非常的不方便,如果能在项目中集成一个在线查看接口说明的地方,肯定更方便更直观.在网上看到swagger这个组件,界面 ...
- Postman----基础使用篇(没有接口文档的情况下如何着手做接口测试)
[备注说明]内文中的图片由于页面的限制,图片显示不清晰,为了能更加的看清图片,请点击"图片",点击"右键"选择"在新标签页中打开图片",可查 ...
- 接口文档管理工具-Postman、Swagger、RAP(转载)
接口文档管理工具-Postman.Swagger.RAP 转自:http://www.51testing.com/html/10/n-3715910.html 在项目开发测试中,接口文档是贯穿始终的. ...
- .net core的Swagger接口文档使用教程(一):Swashbuckle
现在的开发大部分都是前后端分离的模式了,后端提供接口,前端调用接口.后端提供了接口,需要对接口进行测试,之前都是使用浏览器开发者工具,或者写单元测试,再或者直接使用Postman,但是现在这些都已经o ...
- SpringBoot集成Swagger(Swagger的使用),生成接口文档,方便前后端分离开发
首先上一张成果图. 1.Maven依赖 <dependency> <groupId>io.springfox</groupId> <artifactId&g ...
- Go语言使用swagger生成接口文档
swagger介绍 Swagger本质上是一种用于描述使用JSON表示的RESTful API的接口描述语言.Swagger与一组开源软件工具一起使用,以设计.构建.记录和使用RESTful Web服 ...
- 前后端分离之【接口文档管理及数据模拟工具docdoc与dochelper】
前后端分离的常见开发方式是: 后端:接收http请求->根据请求url及params处理对应业务逻辑->将处理结果序列化为json返回 前端:发起http请求并传递相关参数->获取返 ...
随机推荐
- C++ 11 多线程初探-std::memory_order
std::memory_order(可译为内存序,访存顺序) 动态内存模型可理解为存储一致性模型,主要是从行为(behavioral)方面来看多个线程对同一个对象同时(读写)操作时(concurren ...
- 前端-Vue基础3(父子组件交互)
1.子组件往父组件传值 点击子组件的值,子组件自增,父组件的值也跟着自增 通过:this.$emit('change')方法向父组件暴露事件,在子组件中引用,可以调用父组件的方法 点击子组件触发cli ...
- MySQL8.0.26安装与卸载
一.安装 1.官网下载 百度进入官网,学习用社区版够了,我下的是压缩版点这直达下载页 据说8.X版本性能优化,比5.7版本快2倍! 接着,不登录直接下载 2.创建配置 下载完后,建议解压到一个没有中文 ...
- PAT乙级:1076 Wifi密码 (15分)
PAT乙级:1076 Wifi密码 (15分) 题干 下面是微博上流传的一张照片:"各位亲爱的同学们,鉴于大家有时需要使用 wifi,又怕耽误亲们的学习,现将 wifi 密码设置为下列数学题 ...
- vulnhub-DC:1靶机渗透记录
准备工作 在vulnhub官网下载DC:1靶机https://www.vulnhub.com/entry/dc-1,292/ 导入到vmware 打开kali准备进行渗透(ip:192.168.200 ...
- JS对DOM的操作优化法则
html页面显示过程 解析HTML,并生成一棵DOM tree 解析各种样式并结合DOM tree生成一棵Render tree 对Render tree的各个节点计算布局信息,比如box的位置与尺寸 ...
- QML用同一模版多开主窗口
如何动态地创建多个长的一样的主窗口哪(数据当然不一样), 用QML也是可以实现的. 简单的地说, 就是调用多次load即可. QCoreApplication::setAttribute(Qt::AA ...
- Mybatis学习笔记-ResultMap结果集映射
解决属性名与字段名不一致的问题 新建项目 --> 测试实体类字段不一致的情况 数据库字段:id,name,pwd 实体类属性:id,name,password 输出结果 User{id=1, n ...
- vue日记②之兼容各种情况的可跳转链接
兼容各种情况的可跳转链接 需求 因为聊天气泡颜色原因,发送出去的链接通常模糊不清,而且不能直接跳转,所以我打算已a链接的显示直接抓取所有的网页链接,同时还要兼容富文本框的直接输入图片 这是运行效果 实 ...
- ArrayList 源码底层实现解析 基于1.8
ArrayList 介绍 ArrayList是一种线性数据结构,它的底层是用数组实现的,是动态数组.与Java中的数组相比,它的容量能动态增长.源代码里有解释.当创建一个数组的时候,就必须确定它的大小 ...