基于Django Admin+HttpRunner-1.5.6开发简易的接口测试平台

news2024/12/26 10:58:47

前言

这是一个使用HttpRunner开发接口平台的简单Demo。

新建Django项目

安装依赖包

pip install httprunner=1.5.6 -i https://pypi.doubanio.com/simple/

模型规划

  • 项目Project:包含 名称、创建时间、修改时间
  • 测试套件TestSuite:对应HttpRunner的一个yaml文件,包含所属项目、name、base_url、request请求配置、variables用户自定义变量、创建时间、修改时间
  • 测试用例TestCase:对应HttpRunner中的一个test段,包含所属TestSuite、name、skip、request、validate、extract、创建时间、修改时间
  • 测试结果TestResult:测试套件运行的一次结果信息,包含所属TestSuite、HttpRunner运行summary中的时间信息、统计信息、平台信息、详情等

自定义YamlField

由于TestSuite中的request、variables以及用例中的request我们需要使用Python的字典格式,用例中的validate和extract需要使用Python的列表格式。而Django中这些只能按字符串格式TextField存储。
我们编写一个自定义YamlField,存库时按字符串存,读取时转为Python字典或列表。
在apitest目录下新建fields.py,内容如下。

import yaml
from django.db import models

class YamlField(models.TextField):
    def to_python(self, value):  # 将数据库内容转为python对象时调用
        if not value:
            value = {}
        if isinstance(value, (list, dict)):
            return value
        return yaml.safe_load(value)

    def get_prep_value(self, value):  # create时插入数据, 转为字符串存储
        return value if value is None else yaml.dump(value, default_flow_style=False)

    def from_db_value(self, value, expression, connection):  # 从数据库读取字段是调用
        return self.to_python(value)

使用抽象模型

由于好几个项目、测试套件、测试用例都需要名称、创建时间、修改时间三个属性。为了简化代码,这里创建一个抽象模型ModelWithName,抽象模型用来通过继承来复用属性,并不会创建表。
修改apitest/models.py,添加:

from django.db import models
class ModelWithName(models.Model):
    class Meta:
        abstract = True

    name = models.CharField("名称", max_length=200)
    created = models.DateTimeField('创建时间', auto_now_add=True)
    modified = models.DateTimeField('最后修改时间', auto_now=True)
    def __str__(self):
        return self.name

编写模型

修改apitest/models.py,添加:

class Project(ModelWithName):
    class Meta:
        verbose_name_plural = verbose_name = '项目'


class TestSuite(ModelWithName):
    """对应httprunner的一个yaml文件"""
    class Meta:
        verbose_name_plural = verbose_name = '测试套件'
    project = models.ForeignKey(Project, verbose_name='项目', related_name='suites', on_delete=models.CASCADE)
    base_url = models.CharField('域名', max_length=500, blank=True, null=True)  # 对应config/base_url
    request = YamlField('请求默认配置', blank=True)  # 对应config/request
    variables = YamlField('变量', blank=True)

class TestCase(ModelWithName):
    """对应httprunner中的一个test"""
    class Meta:
        verbose_name_plural = verbose_name = '测试用例'

    suite = models.ForeignKey(TestSuite, verbose_name='测试套件', related_name='tests', on_delete=models.CASCADE)
    skip = models.BooleanField('跳过', default=False)
    request = YamlField('请求数据')  # 对应config/request
    extract = YamlField('提取请求', blank=True)
    validate = YamlField('断言', blank=True)

class TestResult(models.Model):
    class Meta:
        verbose_name_plural = verbose_name = '测试结果'

    suite = models.ForeignKey(TestSuite, verbose_name='测试套件', related_name='results', on_delete=models.CASCADE)
    success = models.BooleanField('成功')
    start_at = models.DateTimeField('开始时间')
    duration = models.DurationField('持续时间')
    platform = models.TextField('平台信息')
    test_run = models.SmallIntegerField('运行')
    successes = models.SmallIntegerField('成功')
    skipped = models.SmallIntegerField('跳过')
    failures = models.SmallIntegerField('失败')
    errors = models.SmallIntegerField('出错')
    expected_failures = models.SmallIntegerField('预期失败')
    unexpected_successes = models.SmallIntegerField('非预期成功')
    details = models.TextField('详情')
    created = models.DateTimeField('创建时间', auto_now_add=True)

    def __str__(self):
        return self.suite.name + '-测试结果'

HttpRunner运行结果的summary的格式如下:

 {'platform': {'httprunner_version': '1.5.6', 'platform': 'Darwin-19.2.0-x86_64-i386-64bit', 'python_version': 'CPython 3.6.5'},
 'stat': {'errors': 0, 'expectedFailures': 0,'failures': 0,'skipped': 0,'successes': 1,'testsRun': 1,'unexpectedSuccesses': 0},
 'success': True,
 'time': {'duration': 2.2655465602874756, 'start_at': 1587895780.3771362}}
 'details': [   # 每个对应一个测试套件
    {'name': '套件名称',
     'base_url': 'https://httpbin.org',
     'stat':  {'errors': 0, 'expectedFailures': 0,'failures': 0,'skipped': 0,'successes': 1,'testsRun': 1,'unexpectedSuccesses': 0},
     'success': True,
     'time': {'duration': 2.2655465602874756, 'start_at': 1587895780.3771362}},
     'output': [],
     'records': [   # 对应每一条用例
         {
           'name': '用例名',
           'status': 'success',
           'meta_data': {'request': {'url': ..., 'method': ..., 'start_timestamp': ...}, 
                                  'response': {'content': ..., 'text': ..., 'json': ..., 'headers': ..., 'status_code': ..., 'elapsed_ms': ...}}
           'attachment': ['出错信息']
         }
     ]
 }

这里TestResult模型,对summary结果的信息做了简单的拆解。

组装用例数据

对于用例TestCase,我们需要将其name、skip、request、validate、extract组装成HttpRunner的字典格式。
在apitest/models.py的TestCase类中添加data属性方法,代码如下:

class TestCase(ModelWithName):
    ....
    @property
    def data(self):
        return dict(name=self.name,skip=self.skip,request=self.request,extract=self.extract,validate=self.validate)

组装测试套件数据

一个套件最后解析后应该是包含name、config、apis、testcases的一个字典,我们需要将TestSuite对象及包含的所有TestCase对象组装成如下格式。

{"name": "套件名称", "config" : {...}, "apis": {}, "testcases": []}

补充:加载debugtalk.py的方法
config中可以指定一个yaml的path路径,会自动加载该路径下的debugtalk.py文件

- utils
      - config.yaml  # 空文件即可
      - debugtalk.py

config的格式可以为:

config: 
      name: ...
      request: ...
      variables: ...
      path: .../config.yaml

这样可以自动加载debugtalk.py中的函数以供使用。

在apitest/models.py的TestSuite类中添加data属性方法,代码如下:

@property
    def data(self):
        request = self.request
        request['base_url'] = self.base_url
        data = dict(
            name=self.name,
            config=dict(request=self.request, variables=self.variables),
            api={},
            testcases=[test.data for test in self.tests.all()]
        )
        return data

由于TestCase在外联TestSuite时设置了关联名称tests,因此TestSuite对象可以通过self.tests.all()查询出所有关联它的用例。

注:HttpRunner-1.5.6版本的base_url是放在config/request中的,这里做了分离,要重新放入config/request中。

编写套件运行方法

从 httprunner.task模块中导入HttpRunner类,使用TestSuite数据,运行即可。由于运行时是安多个TestSuite模式运行的,因此TestSuite的数据要放到一个列表中。
在apitest/models.py的TestSuite类添加run方法。

from httprunner.task import HttpRunner
...

class TestSuite(ModelWithName):
    ...
    def run(self):
        runner = HttpRunner().run([self.data])
        summary = runner.summary
        if summary:
            # 保存结果到TestResult
            _time = summary['time']
            _stat = summary['stat']
            TestResult.objects.create(
                suite=self, success=summary['success'],
                start_at=datetime.datetime.fromtimestamp(_time['start_at']),
                duration=datetime.timedelta(seconds=_time['duration']),
                test_run=_stat['testsRun'], successes=_stat['successes'], skipped=_stat['skipped'], errors=_stat['errors'],
                failures=_stat['failures'], expected_failures=_stat['expectedFailures'],
                unexpected_successes=_stat['unexpectedSuccesses'],
                platform=json.dumps(summary['platform'], indent=2, ensure_ascii=False),
                details=summary['details']
            )
        return summary

运行后,解析summary并创建TestResult对象保存本次运行结果。

模型完整代码

import datetime
import json

from django.db import models
from httprunner.task import HttpRunner

from .fields import YamlField


class ModelWithName(models.Model):
    class Meta:
        abstract = True

    name = models.CharField("名称", max_length=200)
    created = models.DateTimeField('创建时间', auto_now_add=True)
    modified = models.DateTimeField('最后修改时间', auto_now=True)
    
    def __str__(self):
        return self.name

class Project(ModelWithName):
    class Meta:
        verbose_name_plural = verbose_name = '项目'


class TestSuite(ModelWithName):
    """对应httprunner的一个yaml文件"""
    class Meta:
        verbose_name_plural = verbose_name = '测试套件'
    project = models.ForeignKey(Project, verbose_name='项目', related_name='suites', on_delete=models.CASCADE)
    base_url = models.CharField('域名', max_length=500, blank=True, null=True)  # 对应config/base_url
    request = YamlField('请求默认配置', blank=True)  # 对应config/request
    variables = YamlField('变量', blank=True)

    @property
    def data(self):
        request = self.request
        request['base_url'] = self.base_url
        data = dict(
            name=self.name,
            config=dict(request=self.request, variables=self.variables),
            api={},
            testcases=[test.data for test in self.tests.all()]
        )
        return data

    def run(self):
        runner = HttpRunner().run([self.data])
        summary = runner.summary
        if summary:
            # 保存结果到TestResult
            _time = summary['time']
            _stat = summary['stat']
            TestResult.objects.create(
                suite=self, success=summary['success'],
                start_at=datetime.datetime.fromtimestamp(_time['start_at']),
                duration=datetime.timedelta(seconds=_time['duration']),
                test_run=_stat['testsRun'], successes=_stat['successes'], skipped=_stat['skipped'], errors=_stat['errors'],
                failures=_stat['failures'], expected_failures=_stat['expectedFailures'],
                unexpected_successes=_stat['unexpectedSuccesses'],
                platform=json.dumps(summary['platform'], indent=2, ensure_ascii=False),
                details=summary['details']
            )
        return summary


class TestCase(ModelWithName):
    """对应httprunner中的一个test"""
    class Meta:
        verbose_name_plural = verbose_name = '测试用例'

    suite = models.ForeignKey(TestSuite, verbose_name='测试套件', related_name='tests', on_delete=models.CASCADE)
    skip = models.BooleanField('跳过', default=False)
    request = YamlField('请求数据')  # 对应config/request
    extract = YamlField('提取请求', blank=True)
    validate = YamlField('断言', blank=True)

    @property
    def data(self):
        return dict(name=self.name,skip=self.skip,request=self.request,extract=self.extract,validate=self.validate)


class TestResult(models.Model):
    class Meta:
        verbose_name_plural = verbose_name = '测试结果'

    suite = models.ForeignKey(TestSuite, verbose_name='测试套件', related_name='results', on_delete=models.CASCADE)
    success = models.BooleanField('成功')
    start_at = models.DateTimeField('开始时间')
    duration = models.DurationField('持续时间')
    platform = models.TextField('平台信息')
    test_run = models.SmallIntegerField('运行')
    successes = models.SmallIntegerField('成功')
    skipped = models.SmallIntegerField('跳过')
    failures = models.SmallIntegerField('失败')
    errors = models.SmallIntegerField('出错')
    expected_failures = models.SmallIntegerField('预期失败')
    unexpected_successes = models.SmallIntegerField('非预期成功')
    details = models.TextField('详情')
    created = models.DateTimeField('创建时间', auto_now_add=True)

    def __str__(self):
        return self.suite.name + '-测试结果'

使用Django Admin

修改apitest/admin.py,代码如下:

from django.contrib import admin

from apitest import models


@admin.register(models.Project)
class ProjectAdmin(admin.ModelAdmin):
    list_display = ('name', 'created', 'modified')


class TestCaseInline(admin.StackedInline):
    model = models.TestCase
    extra = 1


@admin.register(models.TestSuite)
class TestSuiteAdmin(admin.ModelAdmin):
    inlines = [TestCaseInline]
    list_display = ('name', 'project', 'base_url', 'created', 'modified')
    list_filter = ('project', )

    actions = ("run", )

    def run(self, request, queryset):
        for suite in queryset:
            suite.run()
    run.short_description = "运行"


@admin.register(models.TestResult)
class TestResultAdmin(admin.ModelAdmin):
    readonly_fields = ('suite', 'success', 'start_at', 'duration', 'platform',
                       'test_run', 'successes', 'skipped', 'failures', 'errors',
                       'expected_failures', 'unexpected_successes', 'details', 'created')
    fields = (('suite', 'success'),
              ('start_at', 'duration'),
              ('platform',),
              ('test_run', 'successes', 'skipped', 'failures', 'errors', 'expected_failures', 'unexpected_successes'),
              ('details',)
              )
    list_display = ('suite', 'success', 'test_run', 'successes', 'errors', 'failures', 'start_at', 'duration')
    list_filter = ('suite', )

这里将项目、测试套件、测试结果三个模型注册到Admin后台,测试用例则作为内联模型放到测试套件中进行编辑。
在测试套件模型中,自定义了一个“运行”,操作,支持运行选中的用例。

运行并测试项目

打开terminal终端,执行数据库变更并创建超级管理员。

python3 manage.py makemigrations
python3 manage.py migrate
python3 manage.py createsuperuser

运行开发服务器

python3 manage.py runserver

访问http://127.0.0.1:8000/admin并登录。

创建一个项目,测试项目,然后创建一个TestSuite,如下:

请求默认配置:

headers:
  x-text: abc123

变量:

a: 1
b: 2

请求数据:

url: /get
method: GET
params: 
  a: $a
  b: $b

提取请求:

- res_url: content.url

断言:

- eq: [status_code, 200]

点击保存。

回到TestSuite列表,选中测试套件,动作下拉框中选择“运行”,点击Go按钮。

返回测试结果列表、查看测试结果。

实战案例

光学理论是没用的,要学会跟着一起敲,要动手实操,才能将自己的所学运用到实际当中去,这时候可以搞点实战案例来学习。

如果对你有帮助的话,点个赞收个藏,给作者一个鼓励。也方便你下次能够快速查找。

如有不懂还要咨询下方小卡片,博主也希望和志同道合的测试人员一起学习进步

在适当的年龄,选择适当的岗位,尽量去发挥好自己的优势。

我的自动化测试开发之路,一路走来都离不每个阶段的计划,因为自己喜欢规划和总结,

测试开发视频教程、学习笔记领取传送门!!!

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

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

相关文章

自学测试半年,终于收到了字节的offer,那一刻我哭出了声...

我是一名毕业于普通一本的化学专业学生,毕业的两年时间里,我一直奔波在化工厂里。每天工作三班倒,下了班就是一包烟一瓶酒,生活过得非常堕落。 原本想着虽然每天很累,但是至少稳定。然而没有想到的是,化工…

【复杂网络建模】——通过平均度和随机概率构建ER网络(Python)

🤵‍♂️ 个人主页:Lingxw_w的个人主页 ✍🏻作者简介:计算机科学与技术研究生在读 🐋 希望大家多多支持,我们一起进步!😄 如果文章对你有帮助的话, 欢迎评论 &#x1f4a…

频谱仪设置积分功率

按键作用说明: 1,freq:设置频谱仪显示要采集的中心频率范围,先观察明白要测的是哪一段。 按下freq后,手动输入数字大小,然后从竖着的一列选择单位。 2,SPAN:以扫描频率为中心,信号…

大型企业数智化关键举措太难懂?这本数智平台白皮书带你秒理解

“IDC观点:未来企业都会成为数字原生企业,数智化业务将成为主流,因而企业需要积极探索适合自己的数智化转型方法,统筹结合外部业务商业创新、内部管理变革以及产业互联进程,并密切关注数智化升级和转型过程的安全性和可…

Python的缩进规则

目录 缩进规则 缩进异常 IDLE 开发环境对缩进量的设置 缩进规则 和其它程序设计语言(如 Java、C 语言)采用大括号“{}”分隔代码块不同,Python 采用代码缩进和冒号( : )来区分代码块之间的层次。 在 Python 中&…

Android 反编译工具 jadx-gui

jadx-gui 是一种基于 jadx 项目的图形界面工具,用于反编译 Android 应用程序的工具。通过使用jadx-gui,开发人员可以打开 APK(Android应用程序包)文件,并查看其反编译的源代码。这对于分析、理解和调试 Android 应用程…

组合预测模型 | ARIMA-LSTM时间序列预测(Python)

组合预测模型 | ARIMA-LSTM时间序列预测(Python) 目录 组合预测模型 | ARIMA-LSTM时间序列预测(Python)预测结果基本介绍程序设计参考资料 预测结果 基本介绍 ARIMA-LSTM时间序列预测(Python完整源码和数据&#xff09…

设计模式——中介者

1.定义 用一个中介对象封装一系列对象交互,中介者使各对象不需要显示的相互引用,从而使其耦合松散,而且可以独立的改变它们之间的交互。 2.使用场景 1、系统中对象之间存在比较复杂的引用关系,导致它们之间的依赖关系结构混乱而…

关于Nginx的那些事

关于Nginx的那些事 一、Nginx的基础nginx VS apache 二、 编译安装Nginx服务1.关闭防火墙,将安装nginx所需软件包传到/opt目录下2.安装依赖包3.创建运行用户、组4.编译安装Nginx5.检查、启动、重启、停止 nginx服务新版本升级: 6.添加 Nginx 系统服务方法…

【机器学习】正则化详解和过拟合的解决

https://blog.csdn.net/weixin_45434953/article/details/130970273 上一篇文章的例子中,如果使用一个四次多项式去拟合房价函数,会导致过拟合问题 而正则化是解决过拟合的一个方法。右图过拟合是因为其三次方项和四次方项的影响,我们再回顾…

ChatGPT们对今后社会生活的影响

探索ChatGPT,协助工作学习创作。加入「阿杰与AI」公众号,一同探讨,一同成长,比他人更进一步。 1.AI、OpenAI、MidJourney发展史2.ChatGPT们对今后社会生活的影响3.目前市面比较好的AI产品介绍4.注册方式汇总5.针对初学者的 ChatG…

C#,码海拾贝(28)——求解“对称正定方程组”的“平方根法”之C#源代码

using System; namespace Zhou.CSharp.Algorithm { /// <summary> /// 求解线性方程组的类 LEquations /// 原作 周长发 /// 改编 深度混淆 /// </summary> public static partial class LEquations { /// <summary> /…

淘宝618每日一猜答案(6月5日) 淘宝大赢家今日答案

淘宝6月5日每日一猜答案是什么&#xff1f;&#xff0c;接下来也会给大家来介绍一下6月5日淘宝大赢家每日一猜的答案。 淘宝每日一猜6月5日答案分享 活动问题&#xff1a;亚特兰蒂斯体型最大的员工 活动答案&#xff1a;【白鲸】 注意&#xff1a;打开手机淘宝&#xff0c;搜…

记录 vue3 webpack 使用 iframe 遇到的坑

需求 我尝试用Vue3写一个自己的主页&#xff0c;把常用的功能集中到主页中&#xff0c;如下图 后发现一个好玩的东西&#xff0c;js实现的在网页底部出现鱼和波浪&#xff0c;如下图&#xff0c;就像想也放到自己的主页中&#xff0c;搜索后发现可以在Vue中用iframe标签直接引…

Nginx踩坑记录 配置文件中的include不能嵌套 (include的文件里不能再include其他文件了)

上一篇文章介绍了配置文件有多个位置&#xff0c;并且可以采用include实现主配置文件分文件的个性化配置模式&#xff1a;Nginx配置文件 所在路径 到底在哪&#xff1f;_身价五毛的博客-CSDN博客 然后就又踩坑了&#xff0c;include不能嵌套调用&#xff0c;即&#xff1a; …

战斗机上的无线电设备都有哪些?

无线电在我们的生活中无处不在&#xff0c;而无线电设备也有着广泛的应用。那么有没有老铁知道&#xff0c;战斗机上都有哪些无线电设备呢&#xff1f;下面河南宝蓝小编带大家一起来了解下&#xff1a; 战斗机上的无线电设备都有哪些 1、通信系统 通信系统是战斗机比较基本的…

IP-GUARD如何将触发策略的报警信息按照一定周期发送到指定邮箱?

如何将触发策略的报警信息按照一定周期发送到指定邮箱&#xff1f; 1、设置邮件服务器 控制台菜单【工具】-【选项】-【邮件报告服务器设置】中&#xff0c;设置好邮件服务器&#xff1b; 2、邮件报告设置 控制台菜单【工具】-【邮件报告设置】中&#xff0c;新建邮件配置&am…

技术的交流 思想的碰撞|2023 开放原子全球开源峰会 TOC 面对面分论坛即将启幕

由开放原子开源基金会主办的 2023 开放原子全球开源峰会 TOC 面对面分论坛即将于 6 月 13 日在北京经开区北人亦创国际会展中心隆重举办。 作为开放原子开源基金会最高技术决策机构&#xff0c;TOC&#xff08;Technical Oversight Committee&#xff09;汇聚了基金会单位的开…

大语言模型的创意能力到底几何?探索从GPT-2到GPT-4的演进

编者按&#xff1a;大语言模型可以提供许多创意性内容&#xff0c;如写诗、写小说等。那么到底应该如何评估大语言模型生成的创意性内容的水平呢&#xff1f; 本文探讨了GPT-2到GPT-4的创造性&#xff0c;并分析了这些模型在不同创造性测试中的表现。作者使用了三种测试来衡量模…

分布式事务·入门与解决·贰·Saga

文章目录 1 saga分布式事务框架1.1 分布式事务相关理论1.1.1 分布式事务的出现1.1.2 CAP定理1.1.3 BASE定理——CAP的解决思想1.1.4 分布式事务四种模式1.1.5 XA、AT与TCC模式1.1.6 Saga模式 1.2 分布式事务框架ElegentACTX对Saga模式的解决1.2.1 ElegentACTX介绍1.2.2 Elegent…