背景

公司项目java开发使用swagger工具作为接口文档,每次设计接口测试用例的时候,先是设计好excel表头,然后再一步一步的ctrl+c\ctrl+v很多重复的工作,于是想使用python来解析接口返回的json对象数据,然后清洗重新组成excel新的测试用例,又因为学习了别人的httprunner接口测试框架,十分友善支持json/yaml格式的接口测试用例,感觉很契合,既然有了思路,然而并没有急着用python去解析,而是选择了java,先是模拟了一遍,并没有完整实现,只导出了部分包含接口地址等接口数据

再说一下代码编程的工作者,首先要有编程思维,个人理解就是在做事之前,先要理解需求,需要什么条件,怎么做才能满足,最后才是为什么要这么做,有没有更好的方法?我们都知道代码是可复用的,所以还是百度吧,搜索的结果是出人意料的多:swagger 自动生成接口测试用例\swagger 自动生成接口测试用例\swagger 自动生成接口测试用例,这不是重要的事情要说三遍,而是找到了几篇很雷同又很同步的python脚本,初步断定基本能用。

厚颜说一下<拿来主义>在程序开发中是屡见不鲜的事情,但是有一个点值得注意,并不是所有网上收来的代码都能够正确在本地执行及一步达到你想要的结果,如果不能还是自己乖乖的写,虽然效率方面差了一点,自己一步一步的调试出符合当前环境下的结果,需要不断的优化。前面搜索出来的几个关于swagger的相关脚本,其中有几处不适合我的本地环境:

  1. excel接口测试用例没有请求参数,如params;
  2. 在解析json对象的deprecated描述用法,如果不存在的key直接报错;
  3. 在uri--$ref拆分的时候,如果描述过多(多个//符号),可能取值错误。

本地执行分析完成后,就开始修改他们的代码,尽管也花了一定的时间,但是最后整体的代码是经过优化的,虽然不确定比原来的好,但是一定是适合当前环境下的,或许闲来无事还可以再进一步优化:目前已实现生成json接口测试用例,导出excel表格,增加接口请求参数入列,备份文件、对比接口文件等等功能。

#!/usr/bin/python3
# -*- coding: utf-8 -*-
# @File    : swagger.py
"""
导库顺序:优先基础库\第三方库\自定义封装
格式建议:import一行一个
from导入可以import后面用逗号分隔
"""
import os
import json
import requests
from utils.HandleLogging import log
from utils.HandleJson import write_data
from utils.HandleConfig import HandleConfig
from utils.HandleDirFile import HandleDirFile
from utils.HandleExcel import Write_excel
import config

# 创建可操作配置文件的对象
conf = HandleConfig(config.config_path + "\common.conf")
# 创建可操作目录及文件的对象
handlefile = HandleDirFile()
# 创建可操作xlsx文件的对象
w = Write_excel(config.xlsCase_path)


class AnalysisSwaggerJson(object):
    """
    swagger自动生成接口测试用例的工具类,此类以生成json格式的测试用例
    """
    
    def __init__(self, url):
        '''
           初始化类,指定请求的swagger接口地址
        '''
        self.url = url
        self.interface = {}  # json接口测试用例类型
        self.case_list = []  # 测试用例的名称
        self.tags_list = []  # 测试用例的标签
        # 定义测试用例集格式
        self.http_suite = {"config": {"name": "", "base_url": "", "variables": {}},
                           "testcases": []}
        # 定义测试用例格式
        self.http_testcase = {"name": "", "testcase": "", "variables": {}}
        
        
        # 这些目录的存放,需要统一规划,存在多个不同目录下的文件目录不方便管理,在执行时容易串门;
        # 需要提示:在请求swagger接口文档地址的时候,记得去config配置文件下修改对应的路径
        # 生成api测试用例地址,不存在则创建
        if not os.path.exists(config.case_path):
            os.mkdir(config.case_path)
            
        # 备份文件,如果不存在备份目录则备份,否则的实现方案在其他方法内
        if not os.path.exists(config.back_path):
            handlefile.copy_dir(config.case_path, config.back_path)
        
        
    def analysis_json_data(self, isDuplicated=False):
        """
                       解析json格式数据的主函数
        :return:
        """
        # swagger接口文档地址,其中运营后台的接口地址,请求分模块,全量或者其他服务菜单
        if "9527" in self.url:
            try:
                res = requests.get(self.url + '/v2/api-docs?group=全量接口').json()  # 这才是swagger接口请求的地址
                write_data(res, 'data.json')
            except Exception as e:
                log.error('请求swagger地址错误. 异常如下: {}'.format(e))
                raise e
        else:
            try:
                res = requests.get(self.url + '/v2/api-docs').json()  # 这才是swagger接口请求的地址
                write_data(res, 'data.json')
            except Exception as e:
                log.error('请求swagger地址错误. 异常如下: {}'.format(e))
                raise e
        
        self.data = res['paths']  # 取接口地址返回的path数据,包括了请求的路径
        self.basePath = res['basePath']  # 获取接口的根路径/hcp
        self.url = 'http://' + res['host']  # 第一错,swagger文档是ip地址,使用https协议会错误,注意接口地址的请求协议
        self.title = res['info']['title']  # 获取接口的标题
        self.http_suite['config']['name'] = self.title  # 在初始化用例集字典更新值
        self.http_suite['config']['base_url'] = self.url

        self.definitions = res['definitions']  # body参数
        
        for tag_dict in res['tags']:
            self.tags_list.append(tag_dict['name'])
            
        i = 0
        for tag in self.tags_list:
            self.http_suite['testcases'].append({"name": "", "testcase": "", "variables": {}})
            self.http_suite['testcases'][i]['name'] = tag
            self.http_suite['testcases'][i]['testcase'] = 'testcases/' + tag + '.json'
            i += 1
                
        suite_path = config.testsuites_path
        # 测试用例集目录不存在,则创建
        if not os.path.exists(suite_path):
            os.makedirs(suite_path)
            
        testsuite_json_path = os.path.join(suite_path, '{}_testsuites.json'.format(self.title))
        # 数据写入
        write_data(self.http_suite, testsuite_json_path)
                
        if isinstance(self.data, dict):  # 判断接口返回的paths数据类型是否dict类型
            for tag in self.tags_list:  # 前面已经把接口返回的结果tags分别写入了tags_list空列表,再从json对应的tag往里面插入数据
                self.http_case = {"config": {"name": "", "base_url": "", "variables": {}}, "teststeps": []}
                for key, value in self.data.items():
                    for method in list(value.keys()):
                        params = value[method]
                        if not 'deprecated' in value.keys():  # deprecated字段标识:接口是否被弃用,暂时无法判断,使用consumes偷换
                            if params['tags'][0] == tag:
                                self.http_case['config']['name'] = params['tags'][0]
                                self.http_case['config']['base_url'] = self.url
                                case = self.wash_params(params, key, method, tag)
                                self.http_case['teststeps'].append(case)
                        else:
                            log.info(
                                'interface path: {}, if name: {}, is deprecated.'.format(key, params['operationId']))
                            break
                        
                testcases_path = config.testcases_path
                
                # testcases目录不存在则创建
                if not os.path.exists(testcases_path):
                    os.makedirs(testcases_path)
                    
                testcase_json_path = os.path.join(testcases_path, tag + '.json')
#                 生成testcase文件
                write_data(self.http_case, testcase_json_path.replace("/", "_"))
        
        else:
            log.error('解析接口数据异常!url 返回值 paths 中不是字典.')
            return 'error'
        
        # 生成完整的json测试用例之后,开始备份接口数据 ,以备作为接口变更的依据
        if isDuplicated:
            handlefile.copy_dir(config.case_path, config.back_path)


    def wash_params(self, params, api, method, tag):
        """
        清洗数据json,把每个接口数据都加入到一个字典中
        :param params:
        :param params_key:
        :param method:
        :param key:
        :return:
        replace('false', 'False').replace('true', 'True').replace('null','None')
        """
        # 定义接口数据格式
        http_interface = {"name": "", "variables": {},
                          "request": {"url": "", "method": "", "headers": {}, "json": {}, "params": {}}, "validate": [],
                          "output": []}
        # 测试用例的数据格式:
        http_api_testcase = {"name": "", "api": "", "variables": {}, "validate": [], "extract": [], "output": []}
        
        name = params['summary'].replace('/', '_')  # 这里的问题需要具体来分析,开发有时概要使用其他符号分割
        http_interface['name'] = name
        http_api_testcase['name'] = name
        http_api_testcase['api'] = 'api/{}/{}.json'.format(tag, name)  # 这是写入testcasejson下的名字,不是生成api的目录
        http_interface['request']['method'] = method.upper()
        http_interface['request']['url'] = api.replace('{', '$').replace('}', '')  # 这个是替换uri中的/get请求的拼接方式,有些是?参数=&参数拼接,需要另外解析
        parameters = params.get('parameters')  # 未解析的参数字典
        responses = params.get('responses')
        
        if not parameters:  # 确保参数字典存在
            parameters = {}
        # 给测试用例字典,加入解析出来的参数
        for each in parameters:
            if each.get('in') == 'body':  # body 和 query 不会同时出现
                schema = each.get('schema')
                if schema:
                    ref = schema.get('$ref')
                    if ref:
                        param_key = ref.split('/', 2)[-1]  # 这个uri拆分,根据实际情况来取第几个/反斜杠
                        param = self.definitions[param_key]['properties']
                        for key, value in param.items():
                            if 'example' in value.keys():
                                http_interface['request']['json'].update({key: value['example']})
                            else:
                                http_interface['request']['json'].update({key: ''})
                                
            elif each.get('in') == 'query':
                name = each.get('name')
                for key in each.keys():
                    if not 'example' in key:  # 取反,要把在query的参数写入json测试用例
                        http_interface['request']['params'].update({name: each[key]})
            
        
        for each in parameters:
            if each.get('in') == 'header':
                name = each.get('name')
                for key in each.keys():
                    if 'example' in key:
                        http_interface['request']['headers'].update({name: each[key]})
                    else:
                        if name == 'token':
                            http_interface['request']['headers'].update({name: '$token'})
                        else:
                            http_interface['request']['headers'].update({name: ''})
                            
                            
        for key, value in responses.items():
            schema = value.get('schema')
            if schema:
                ref = schema.get('$ref')
                if ref:
                    param_key = ref.split('/')[-1]
                    res = self.definitions[param_key]['properties']
                    i = 0
                    for k, v in res.items():
                        if 'example' in v.keys():
                            http_interface['validate'].append({"eq": []})
                            http_interface['validate'][i]['eq'].append('content.' + k)
                            http_interface['validate'][i]['eq'].append(v['example'])
                            http_api_testcase['validate'].append({"eq": []})
                            http_api_testcase['validate'][i]['eq'].append('content.' + k)
                            http_api_testcase['validate'][i]['eq'].append(v['example'])
                            i += 1
                else:
                    if  len(http_interface['validate']) != 1:
                        http_interface['validate'].append({"eq": []})
            else:
                if  len(http_interface['validate']) != 1:
                    http_interface['validate'].append({"eq": []})
        
        # 测试用例的请求参数为空字典,则删除这些key
        if http_interface['request']['json'] == {}:
            del http_interface['request']['json']
        
        if http_interface['request']['params'] == {}:
            del http_interface['request']['params']
        
        # 定义接口测试用例
        api_path = config.case_path
        tags_path = os.path.join(api_path, tag).replace("/", "_")
        
        # 创建不存在的文件目录
        if not os.path.exists(api_path):
            os.mkdir(api_path)

        if not os.path.exists(tags_path):
            os.mkdir(tags_path)
        
        json_path = os.path.join(tags_path, http_interface['name'] + '.json')
        
        write_data(http_interface, json_path)  # 写入数据

        return http_api_testcase

    
    def write_excel(self, url, filelist):
        '''
           将生成的json格式的数据,转换成xlsx写入文件
        '''
        li1 = url.split(":")
        host = li1[1].replace("/", "")
        port = li1[2][:4]
        uri = li1[2][4:]
        count = 1
        caseId = 0
        for file in filelist:
            caseId += 1
            count += 1
            inter_name = file.split("\\")[-2]  # 获取接口测试用例的上级目录名称:组成name-tag的用例title
            with open(file, 'r', encoding='utf-8') as rdfile:
                text = json.load(rdfile)
                title = inter_name + '-' + text['name']
                method = text['request']['method'].upper()
                w.write(count, 1, "apiTest_" + str(caseId))
                w.write(count, 2, title)
                w.write(count, 4, method)
                w.write(count, 5, host)
                w.write(count, 6, port)
                if 'json' in text['request'].keys():  # post请求的接口相关数据写入excel
                    url = text['request']['url']
                    params = text['request']['json']
                    w.write(count, 7, uri + url)
                    w.write(count, 8, json.dumps(params))
                elif 'params' in text['request'].keys():  # get请求的接口参数写入
                    url = text['request']['url']
                    jsonp = str(text['request']['params'])
                    join_text = jsonp.replace("{", "").replace("}", "").replace(":", "=").replace("'", "").replace(",", "&").replace(" ", "")
                    w.write(count, 7, uri + url)
                    w.write(count, 8, join_text)
                else:  # 将url中包含$符号的get请求的参数单独提取出来写入params
                    url = text['request']['url'].replace('{', '$').replace('}', '')
                    start_index = url.find("$")
                    url1 = url[:start_index]
                    params = url[start_index:]
                    w.write(count, 7, uri + url1)
                    if "$" in params:
                        w.write(count, 8, params)
            

if __name__ == '__main__':
    url = conf.get_value("swaggerUrl", "dev_trade_url")
    js = AnalysisSwaggerJson(url)
    js.analysis_json_data()
#     for i in url.split(","):
#         AnalysisSwaggerJson(i).AnalysisJsonData()
#     js.analysis_json_data()
    js.write_excel(url, handlefile.get_file_list(config.case_path))
#     handlefile.diff_dir_file(config.case_path, config.back_path)

因为大多数是公司外网地址,隐秘数据基本使用本地读取配置文件获取,再次感谢以上源码的作者,站在先驱的肩膀上前行。

秉承着程序员拿来主义的优良传统,特此提供本人项目的github,敬请各位看官笑纳!