通過(guò)Django Admin+HttpRunner1.5.6實(shí)現(xiàn)簡(jiǎn)易接口測(cè)試平臺(tái)
前言
這是一個(gè)使用HttpRunner開(kāi)發(fā)接口平臺(tái)的簡(jiǎn)單Demo。
新建Django項(xiàng)目
安裝依賴包
pip install httprunner=1.5.6 -i https://pypi.doubanio.com/simple/
模型規(guī)劃
項(xiàng)目Project:包含 名稱、創(chuàng)建時(shí)間、修改時(shí)間 測(cè)試套件TestSuite:對(duì)應(yīng)HttpRunner的一個(gè)yaml文件,包含所屬項(xiàng)目、name、base_url、request請(qǐng)求配置、variables用戶自定義變量、創(chuàng)建時(shí)間、修改時(shí)間 測(cè)試用例TestCase:對(duì)應(yīng)HttpRunner中的一個(gè)test段,包含所屬TestSuite、name、skip、request、validate、extract、創(chuàng)建時(shí)間、修改時(shí)間 測(cè)試結(jié)果TestResult:測(cè)試套件運(yùn)行的一次結(jié)果信息,包含所屬TestSuite、HttpRunner運(yùn)行summary中的時(shí)間信息、統(tǒng)計(jì)信息、平臺(tái)信息、詳情等自定義YamlField
由于TestSuite中的request、variables以及用例中的request我們需要使用Python的字典格式,用例中的validate和extract需要使用Python的列表格式。而Django中這些只能按字符串格式TextField存儲(chǔ)。
我們編寫(xiě)一個(gè)自定義YamlField,存庫(kù)時(shí)按字符串存,讀取時(shí)轉(zhuǎn)為Python字典或列表。
在apitest目錄下新建fields.py,內(nèi)容如下。
串存,讀取時(shí)轉(zhuǎn)為Python字典或列表。在apitest目錄下新建fields.py,內(nèi)容如下。
import yamlfrom django.db import modelsclass YamlField(models.TextField): def to_python(self, value): # 將數(shù)據(jù)庫(kù)內(nèi)容轉(zhuǎn)為python對(duì)象時(shí)調(diào)用 if not value: value = {} if isinstance(value, (list, dict)): return value return yaml.safe_load(value) def get_prep_value(self, value): # create時(shí)插入數(shù)據(jù), 轉(zhuǎn)為字符串存儲(chǔ) return value if value is None else yaml.dump(value, default_flow_style=False) def from_db_value(self, value, expression, connection): # 從數(shù)據(jù)庫(kù)讀取字段是調(diào)用 return self.to_python(value)
使用抽象模型
由于好幾個(gè)項(xiàng)目、測(cè)試套件、測(cè)試用例都需要名稱、創(chuàng)建時(shí)間、修改時(shí)間三個(gè)屬性。為了簡(jiǎn)化代碼,這里創(chuàng)建一個(gè)抽象模型ModelWithName,抽象模型用來(lái)通過(guò)繼承來(lái)復(fù)用屬性,并不會(huì)創(chuàng)建表。修改apitest/models.py,添加:
from django.db import modelsclass ModelWithName(models.Model): class Meta: abstract = True name = models.CharField('名稱', max_length=200) created = models.DateTimeField(’創(chuàng)建時(shí)間’, auto_now_add=True) modified = models.DateTimeField(’最后修改時(shí)間’, auto_now=True) def __str__(self): return self.name
編寫(xiě)模型
修改apitest/models.py,添加:
class Project(ModelWithName): class Meta: verbose_name_plural = verbose_name = ’項(xiàng)目’class TestSuite(ModelWithName): '''對(duì)應(yīng)httprunner的一個(gè)yaml文件''' class Meta: verbose_name_plural = verbose_name = ’測(cè)試套件’ project = models.ForeignKey(Project, verbose_name=’項(xiàng)目’, related_name=’suites’, on_delete=models.CASCADE) base_url = models.CharField(’域名’, max_length=500, blank=True, null=True) # 對(duì)應(yīng)config/base_url request = YamlField(’請(qǐng)求默認(rèn)配置’, blank=True) # 對(duì)應(yīng)config/request variables = YamlField(’變量’, blank=True)class TestCase(ModelWithName): '''對(duì)應(yīng)httprunner中的一個(gè)test''' class Meta: verbose_name_plural = verbose_name = ’測(cè)試用例’ suite = models.ForeignKey(TestSuite, verbose_name=’測(cè)試套件’, related_name=’tests’, on_delete=models.CASCADE) skip = models.BooleanField(’跳過(guò)’, default=False) request = YamlField(’請(qǐng)求數(shù)據(jù)’) # 對(duì)應(yīng)config/request extract = YamlField(’提取請(qǐng)求’, blank=True) validate = YamlField(’斷言’, blank=True)class TestResult(models.Model): class Meta: verbose_name_plural = verbose_name = ’測(cè)試結(jié)果’ suite = models.ForeignKey(TestSuite, verbose_name=’測(cè)試套件’, related_name=’results’, on_delete=models.CASCADE) success = models.BooleanField(’成功’) start_at = models.DateTimeField(’開(kāi)始時(shí)間’) duration = models.DurationField(’持續(xù)時(shí)間’) platform = models.TextField(’平臺(tái)信息’) test_run = models.SmallIntegerField(’運(yùn)行’) successes = models.SmallIntegerField(’成功’) skipped = models.SmallIntegerField(’跳過(guò)’) failures = models.SmallIntegerField(’失敗’) errors = models.SmallIntegerField(’出錯(cuò)’) expected_failures = models.SmallIntegerField(’預(yù)期失敗’) unexpected_successes = models.SmallIntegerField(’非預(yù)期成功’) details = models.TextField(’詳情’) created = models.DateTimeField(’創(chuàng)建時(shí)間’, auto_now_add=True) def __str__(self): return self.suite.name + ’-測(cè)試結(jié)果’
HttpRunner運(yùn)行結(jié)果的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’: [ # 每個(gè)對(duì)應(yīng)一個(gè)測(cè)試套件 {’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’: [ # 對(duì)應(yīng)每一條用例 { ’name’: ’用例名’, ’status’: ’success’, ’meta_data’: {’request’: {’url’: ..., ’method’: ..., ’start_timestamp’: ...}, ’response’: {’content’: ..., ’text’: ..., ’json’: ..., ’headers’: ..., ’status_code’: ..., ’elapsed_ms’: ...}} ’attachment’: [’出錯(cuò)信息’] } ] }
這里TestResult模型,對(duì)summary結(jié)果的信息做了簡(jiǎn)單的拆解。
組裝用例數(shù)據(jù)
對(duì)于用例TestCase,我們需要將其name、skip、request、validate、extract組裝成HttpRunner的字典格式。在apitest/models.py的TestCase類(lèi)中添加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)
一個(gè)套件最后解析后應(yīng)該是包含name、config、apis、testcases的一個(gè)字典,我們需要將TestSuite對(duì)象及包含的所有TestCase對(duì)象組裝成如下格式。
{'name': '套件名稱', 'config' : {...}, 'apis': {}, 'testcases': []}
補(bǔ)充:加載debugtalk.py的方法config中可以指定一個(gè)yaml的path路徑,會(huì)自動(dòng)加載該路徑下的debugtalk.py文件如
- utils - config.yaml # 空文件即可 - debugtalk.py
config的格式可以為:
config: name: ... request: ... variables: ... path: .../config.yaml
這樣可以自動(dòng)加載debugtalk.py中的函數(shù)以供使用。
在apitest/models.py的TestSuite類(lèi)中添加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在外聯(lián)TestSuite時(shí)設(shè)置了關(guān)聯(lián)名稱tests,因此TestSuite對(duì)象可以通過(guò)self.tests.all()查詢出所有關(guān)聯(lián)它的用例。
注:HttpRunner-1.5.6版本的base_url是放在config/request中的,這里做了分離,要重新放入config/request中。
編寫(xiě)套件運(yùn)行方法
從 httprunner.task模塊中導(dǎo)入HttpRunner類(lèi),使用TestSuite數(shù)據(jù),運(yùn)行即可。由于運(yùn)行時(shí)是安多個(gè)TestSuite模式運(yùn)行的,因此TestSuite的數(shù)據(jù)要放到一個(gè)列表中。
在apitest/models.py的TestSuite類(lèi)添加run方法。
from httprunner.task import HttpRunner...class TestSuite(ModelWithName): ... def run(self): runner = HttpRunner().run([self.data]) summary = runner.summary if summary: # 保存結(jié)果到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
運(yùn)行后,解析summary并創(chuàng)建TestResult對(duì)象保存本次運(yùn)行結(jié)果。
模型完整代碼
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(’創(chuàng)建時(shí)間’, auto_now_add=True) modified = models.DateTimeField(’最后修改時(shí)間’, auto_now=True) def __str__(self): return self.nameclass Project(ModelWithName): class Meta: verbose_name_plural = verbose_name = ’項(xiàng)目’class TestSuite(ModelWithName): '''對(duì)應(yīng)httprunner的一個(gè)yaml文件''' class Meta: verbose_name_plural = verbose_name = ’測(cè)試套件’ project = models.ForeignKey(Project, verbose_name=’項(xiàng)目’, related_name=’suites’, on_delete=models.CASCADE) base_url = models.CharField(’域名’, max_length=500, blank=True, null=True) # 對(duì)應(yīng)config/base_url request = YamlField(’請(qǐng)求默認(rèn)配置’, blank=True) # 對(duì)應(yīng)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: # 保存結(jié)果到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): '''對(duì)應(yīng)httprunner中的一個(gè)test''' class Meta: verbose_name_plural = verbose_name = ’測(cè)試用例’ suite = models.ForeignKey(TestSuite, verbose_name=’測(cè)試套件’, related_name=’tests’, on_delete=models.CASCADE) skip = models.BooleanField(’跳過(guò)’, default=False) request = YamlField(’請(qǐng)求數(shù)據(jù)’) # 對(duì)應(yīng)config/request extract = YamlField(’提取請(qǐng)求’, 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 = ’測(cè)試結(jié)果’ suite = models.ForeignKey(TestSuite, verbose_name=’測(cè)試套件’, related_name=’results’, on_delete=models.CASCADE) success = models.BooleanField(’成功’) start_at = models.DateTimeField(’開(kāi)始時(shí)間’) duration = models.DurationField(’持續(xù)時(shí)間’) platform = models.TextField(’平臺(tái)信息’) test_run = models.SmallIntegerField(’運(yùn)行’) successes = models.SmallIntegerField(’成功’) skipped = models.SmallIntegerField(’跳過(guò)’) failures = models.SmallIntegerField(’失敗’) errors = models.SmallIntegerField(’出錯(cuò)’) expected_failures = models.SmallIntegerField(’預(yù)期失敗’) unexpected_successes = models.SmallIntegerField(’非預(yù)期成功’) details = models.TextField(’詳情’) created = models.DateTimeField(’創(chuàng)建時(shí)間’, auto_now_add=True) def __str__(self): return self.suite.name + ’-測(cè)試結(jié)果’
使用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 = '運(yùn)行'@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’, )
這里將項(xiàng)目、測(cè)試套件、測(cè)試結(jié)果三個(gè)模型注冊(cè)到Admin后臺(tái),測(cè)試用例則作為內(nèi)聯(lián)模型放到測(cè)試套件中進(jìn)行編輯。在測(cè)試套件模型中,自定義了一個(gè)“運(yùn)行”,操作,支持運(yùn)行選中的用例。
運(yùn)行并測(cè)試項(xiàng)目
打開(kāi)terminal終端,執(zhí)行數(shù)據(jù)庫(kù)變更并創(chuàng)建超級(jí)管理員。
python3 manage.py makemigrationspython3 manage.py migratepython3 manage.py createsuperuser
運(yùn)行開(kāi)發(fā)服務(wù)器
python3 manage.py runserver
訪問(wèn)http://127.0.0.1:8000/admin并登錄。
創(chuàng)建一個(gè)項(xiàng)目,測(cè)試項(xiàng)目,然后創(chuàng)建一個(gè)TestSuite,如下:
請(qǐng)求默認(rèn)配置:
headers: x-text: abc123
變量:
a: 1b: 2
請(qǐng)求數(shù)據(jù):
url: /getmethod: GETparams: a: $a b: $b
提取請(qǐng)求:
- res_url: content.url
斷言:
- eq: [status_code, 200]
點(diǎn)擊保存。
回到TestSuite列表,選中測(cè)試套件,動(dòng)作下拉框中選擇“運(yùn)行”,點(diǎn)擊Go按鈕。
返回測(cè)試結(jié)果列表、查看測(cè)試結(jié)果。
程序代碼https://github.com/hanzhichao/apirunner
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持好吧啦網(wǎng)。
相關(guān)文章:
