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

news/2024/11/29 13:27:49/

前言

这是一个使用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 modelsclass YamlField(models.TextField):def to_python(self, value):  # 将数据库内容转为python对象时调用if not value:value = {}if isinstance(value, (list, dict)):return valuereturn 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 = Truename = 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_urlrequest = YamlField('请求默认配置', blank=True)  # 对应config/requestvariables = 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/requestextract = 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):....@propertydef 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属性方法,代码如下:

@propertydef data(self):request = self.requestrequest['base_url'] = self.base_urldata = 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.summaryif 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 jsonfrom django.db import models
from httprunner.task import HttpRunnerfrom .fields import YamlFieldclass ModelWithName(models.Model):class Meta:abstract = Truename = models.CharField("名称", max_length=200)created = models.DateTimeField('创建时间', auto_now_add=True)modified = models.DateTimeField('最后修改时间', auto_now=True)def __str__(self):return self.nameclass 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_urlrequest = YamlField('请求默认配置', blank=True)  # 对应config/requestvariables = YamlField('变量', blank=True)@propertydef data(self):request = self.requestrequest['base_url'] = self.base_urldata = dict(name=self.name,config=dict(request=self.request, variables=self.variables),api={},testcases=[test.data for test in self.tests.all()])return datadef run(self):runner = HttpRunner().run([self.data])summary = runner.summaryif 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 summaryclass 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/requestextract = YamlField('提取请求', blank=True)validate = YamlField('断言', blank=True)@propertydef 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 adminfrom apitest import models@admin.register(models.Project)
class ProjectAdmin(admin.ModelAdmin):list_display = ('name', 'created', 'modified')class TestCaseInline(admin.StackedInline):model = models.TestCaseextra = 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: $ab: $b

提取请求:

- res_url: content.url

断言:

- eq: [status_code, 200]

点击保存。

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

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

实战案例

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

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

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

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

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

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


http://www.ppmy.cn/news/224321.html

相关文章

ITU-T G.781标准解读(二)

同步信号的常见种类 STM-N Synchronous Transport Module level N,同步传输模块n级 STM-N的帧结构设计使得它可以在低速信号支路上均匀、有规律的分布,这就使得它可以很好的满足同步复用与交叉连接的需求,这样的设计也使得可以它可以满足从…

sim868模块

最近使用了SIM868模块做了项目,单片机使用的是STM32F103C8T6,使用串口2与SIM模块通信,实现了打电话、发短信、GPS定位、GPRS发送数据的功能,能够实时定位,1s发送一次定位数据到服务器。 SIM模块代码如下: …

ISO7816 智能卡 接口

原文 1 了解ISO7816接口 一、主要从两个方面来了解 ISO7816触点的电气特性 ISO7816的通讯协议流程 二、电气特性 有三种类型的7816接口 A类7816:工作电压4.5~5.5V B类7816:工作电压2.7~3.3V C类7816:工作电压1.8V (供电电压…

写好Java代码的30条经验总结

1、 类名首字母应该大写。字段、方法以及对象(句柄)的首字母应小写。对于所有标识符,其中包含的所有单词都应紧靠在一起,而且大写中间单词的首字母。例如: ThisIsAClassNamethisIsMethodOrFieldName 若在定义中出现了…

浅入浅出 iptables 网络隔离原理

01 iptables简介 iptables ipfirewall(内核1.x时代) ipchains(内核2.x时代) iptables 网络协议栈 Link Layer 数据链路层的数据流向,根据mac寻址找到对应的网卡后向上进入网络层 Network Layer 网络层的数据流向&am…

【Linux】LNMP框架的架构与环境配置

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 LNMP框架的架构与环境配置 一、安装 Nginx 服务1.关闭防火墙及安装依赖包2、创建运行用户3、编译安装4、优化路径5、添加 Nginx 系统服务 二、安装 MySQL 服务1、安装Mysql环…

【使用VS2010配合多平台工具集开发ARX14~2012CAD程序的注意事项】

在这里插入代码片使用VS2010配合多平台工具集开发ARX14~2012CAD程序的注意事项 对于计算机上已安装了VS6.0、 VS7.0、VS8.0、VS9.0的,只要在网站http://daffodil.codeplex.com/ 下载Daffodil for Visual Studio - Native Multitargeting Platform Toolsets.(多平台工具集)安…

安卓系统浏览器开发

预置某个浏览器为系统默认的浏览器 描述: 当系统存在多个浏览器时,如何预置某个浏览器为系统默认的浏览器? 方法: 1.在PackageManagerService.java中的构造函数结尾添加:setDefaultBrowser(); 2.setDefaultBrowser()的具体实现&#xff…