日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

通過Django Admin+HttpRunner1.5.6實現簡易接口測試平臺

瀏覽:240日期:2024-09-19 16:10:27

前言

這是一個使用HttpRunner開發接口平臺的簡單Demo。

新建Django項目

通過Django Admin+HttpRunner1.5.6實現簡易接口測試平臺

安裝依賴包

通過Django Admin+HttpRunner1.5.6實現簡易接口測試平臺

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,內容如下。

串存,讀取時轉為Python字典或列表。在apitest目錄下新建fields.py,內容如下。

import yamlfrom django.db import modelsclass 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 modelsclass 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 datetimeimport jsonfrom django.db import modelsfrom httprunner.task import HttpRunnerfrom .fields import YamlFieldclass 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.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_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 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/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 adminfrom 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 makemigrationspython3 manage.py migratepython3 manage.py createsuperuser

運行開發服務器

python3 manage.py runserver

訪問http://127.0.0.1:8000/admin并登錄。

通過Django Admin+HttpRunner1.5.6實現簡易接口測試平臺

創建一個項目,測試項目,然后創建一個TestSuite,如下:

通過Django Admin+HttpRunner1.5.6實現簡易接口測試平臺

請求默認配置:

headers: x-text: abc123

變量:

a: 1b: 2

通過Django Admin+HttpRunner1.5.6實現簡易接口測試平臺

請求數據:

url: /getmethod: GETparams: a: $a b: $b

提取請求:

- res_url: content.url

斷言:

- eq: [status_code, 200]

點擊保存。

回到TestSuite列表,選中測試套件,動作下拉框中選擇“運行”,點擊Go按鈕。

通過Django Admin+HttpRunner1.5.6實現簡易接口測試平臺

返回測試結果列表、查看測試結果。

通過Django Admin+HttpRunner1.5.6實現簡易接口測試平臺

通過Django Admin+HttpRunner1.5.6實現簡易接口測試平臺

程序代碼https://github.com/hanzhichao/apirunner

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持好吧啦網。

標簽: Django
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
亚洲精选91| 久久国产日韩欧美精品| 久久午夜影院| 精品国产91| 国产精品久久777777毛茸茸| 日韩一区二区三区高清在线观看| 亚洲电影在线一区二区三区| 久久久久蜜桃| 自拍日韩欧美| 日韩高清电影免费| 国产精品高清一区二区| 精品视频高潮| 成人日韩在线观看| 亚洲视频播放| 视频一区二区三区入口| 日本麻豆一区二区三区视频| 麻豆成人av在线| 精品中文字幕一区二区三区av| 男女性色大片免费观看一区二区| 日本午夜精品久久久久| 国产粉嫩在线观看| 免费日韩av| 精品一区视频| 一区二区91| 国产一区二区三区四区| 国产精品三上| 91欧美在线| 日本亚洲最大的色成网站www| 91综合视频| 亚洲乱亚洲高清| 免费在线小视频| 国产亚洲人成a在线v网站 | 国产欧洲在线| 日韩 欧美一区二区三区| 99热精品在线观看| 日韩黄色免费网站| 午夜日韩福利| 久久av一区二区三区| 亚洲精品九九| 精品国产免费人成网站| 国产乱码精品一区二区三区四区 | 国产aⅴ精品一区二区三区久久| 美女毛片一区二区三区四区| 日本免费一区二区视频| 亚洲欧美不卡| 国产精品网在线观看| 久久精选视频| 国产一区二区三区久久| 97精品久久| 亚洲日本在线观看视频| 亚洲国内精品| 98精品视频| 日本强好片久久久久久aaa| 爽好多水快深点欧美视频| 韩国久久久久久| 国产精品1区| 日本麻豆一区二区三区视频| 久久亚洲一区| 香蕉久久国产| 久热精品在线| 美女久久一区| 视频一区中文字幕国产| 国产精品试看| 亚洲一级在线| 免费观看在线综合| 一本一本久久| 亚洲免费观看| 免费国产自久久久久三四区久久| 亚洲不卡系列| 激情视频一区二区三区| 国产主播一区| 亚洲涩涩av| 国产视频一区二| 久久久久九九精品影院| 欧美国产美女| 99亚洲精品| 日本午夜免费一区二区 | 国产欧美日韩免费观看| 国产欧美日韩一区二区三区在线| 日韩精品福利一区二区三区| 国产精品色在线网站| 欧美日韩精品免费观看视完整| 黄色av日韩| 国产精品videossex| 亚洲va中文在线播放免费| 日韩视频一区二区三区在线播放免费观看| 亚洲自拍另类| 精品国产一区二区三区av片| 亚洲精品国产偷自在线观看| 精品国产中文字幕第一页| 91精品国产自产在线丝袜啪| 日韩精品专区| 日韩一二三区在线观看| 国产精品va| 国产一在线精品一区在线观看| 亚洲免费观看高清完整版在线观| 国产亚洲久久| 精品国产欧美| 日本一区二区免费高清| 日韩在线视频一区二区三区| 日韩精品国产精品| 欧美日韩亚洲一区二区三区在线 | 国产精品试看| 青青久久av| 日韩视频不卡| 亚洲精品在线国产| 奇米色欧美一区二区三区| 成人综合一区| 福利视频一区| 久久三级毛片| 欧美aa在线观看| 涩涩涩久久久成人精品| 日韩电影免费在线观看| 老牛影视一区二区三区| 久久久精品国产**网站| 日本一区二区三区中文字幕| 国产精品一级| 亚洲天堂久久| 日韩高清在线一区| 99热国内精品| 欧美一级久久| 老司机久久99久久精品播放免费| 久久久一本精品| 亚洲精品激情| 久久精品一区| 欧美日韩国产高清| 国产精品久久久亚洲一区| 久久中文字幕二区| 色偷偷偷在线视频播放| 你懂的亚洲视频| 日韩成人精品一区二区三区| 国产精品国产一区| 日韩精品一级二级 | 日韩精品成人在线观看| 欧美日本久久| 欧美精品激情| 久久只有精品| 亚洲精一区二区三区| 日韩精品欧美激情一区二区| 国产精品午夜av| 蜜桃视频一区二区三区在线观看| 国产精品三上| 麻豆精品新av中文字幕| 美国三级日本三级久久99 | 精品国产精品国产偷麻豆| 老牛国产精品一区的观看方式| 亚洲成人三区| 国产精品久久久久久久免费观看| 免费国产自线拍一欧美视频| 亚洲日本网址| 久久精品一区二区不卡| 国产成人精品亚洲日本在线观看| 成人日韩在线观看| 日韩欧美激情| 久久av免费| 欧美影院精品| 日韩一区精品视频| 在线视频亚洲| 亚洲一区二区三区免费在线观看| 五月天综合网站| 日韩中文欧美| 丝袜亚洲另类欧美| 亚洲字幕久久| 日韩中文字幕无砖| 日韩中文字幕一区二区三区| 蜜臀av一区二区三区| 日韩精品视频网站| 伊伊综合在线| 石原莉奈在线亚洲三区| 黄色亚洲免费| 久久国产亚洲| 三级在线观看一区二区| 亚洲理论在线| 亚州av一区| 国产一区福利| 精品捆绑调教一区二区三区| 亚洲综合电影一区二区三区| 亚洲激情二区| 香蕉久久久久久| 精品国产亚洲一区二区在线观看| 亚洲国产综合在线看不卡| 日韩动漫一区| 国产精品亚洲片在线播放| 国产精品调教| 岛国av在线网站| 99热精品在线观看| 日本不良网站在线观看| 99在线|亚洲一区二区| 99久久久国产精品美女| 欧美在线综合| 亚洲精品护士| 成人一区而且| 国产美女亚洲精品7777| 成人羞羞在线观看网站| 日韩成人a**站| 国产精品婷婷| 国产欧美亚洲一区| 欧美在线资源| 日本久久黄色| 国产高清精品二区|