亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

通過(guò)Django Admin+HttpRunner1.5.6實(shí)現(xiàn)簡(jiǎn)易接口測(cè)試平臺(tái)

 更新時(shí)間:2020年11月11日 10:20:54   作者:臨淵  
這篇文章主要介紹了通過(guò)Django Admin+HttpRunner1.5.6實(shí)現(xiàn)簡(jiǎn)易接口測(cè)試平臺(tái),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下

前言

這是一個(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:包含 名稱(chēng)、創(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 yaml
from django.db import models

class 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è)試用例都需要名稱(chēng)、創(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 models
class ModelWithName(models.Model):
  class Meta:
    abstract = True

  name = models.CharField("名稱(chēng)", 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': '套件名稱(chēng)',
   '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": "套件名稱(chēng)", "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)名稱(chēng)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 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("名稱(chēng)", 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

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)

  @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 summary


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)

  @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 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 = "運(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 makemigrations
python3 manage.py migrate
python3 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í)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • PyTorch基礎(chǔ)之torch.nn.Conv2d中自定義權(quán)重問(wèn)題

    PyTorch基礎(chǔ)之torch.nn.Conv2d中自定義權(quán)重問(wèn)題

    這篇文章主要介紹了PyTorch基礎(chǔ)之torch.nn.Conv2d中自定義權(quán)重問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-02-02
  • Python騷操作之動(dòng)態(tài)定義函數(shù)

    Python騷操作之動(dòng)態(tài)定義函數(shù)

    這篇文章主要介紹了Python騷操作之動(dòng)態(tài)定義函數(shù),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2019-03-03
  • python3.x提取中文的正則表達(dá)式示例代碼

    python3.x提取中文的正則表達(dá)式示例代碼

    這篇文章主要介紹了python3.x中提取中文的正則表達(dá)式的書(shū)寫(xiě),需要的朋友可以參考下
    2019-07-07
  • Python生成器以及應(yīng)用實(shí)例解析

    Python生成器以及應(yīng)用實(shí)例解析

    這篇文章主要介紹了Python生成器以及應(yīng)用實(shí)例解析,分享了相關(guān)代碼示例,小編覺(jué)得還是挺不錯(cuò)的,具有一定借鑒價(jià)值,需要的朋友可以參考下
    2018-02-02
  • 關(guān)于Numpy中的行向量和列向量詳解

    關(guān)于Numpy中的行向量和列向量詳解

    今天小編就為大家分享一篇關(guān)于Numpy中的行向量和列向量詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2019-11-11
  • Python中列表list常用方法總結(jié)

    Python中列表list常用方法總結(jié)

    在?Python?中,列表(List)是一種有序的數(shù)據(jù)集合,可以存儲(chǔ)任意類(lèi)型的數(shù)據(jù)。本文主要幫大家總結(jié)了一下列表list的常用方法,需要的可以參考一下
    2023-05-05
  • python復(fù)合條件下的字典排序

    python復(fù)合條件下的字典排序

    這篇文章主要介紹了python復(fù)合條件下的字典排序,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-12-12
  • 使用C語(yǔ)言擴(kuò)展Python程序的簡(jiǎn)單入門(mén)指引

    使用C語(yǔ)言擴(kuò)展Python程序的簡(jiǎn)單入門(mén)指引

    這篇文章主要介紹了使用C語(yǔ)言擴(kuò)展Python程序的簡(jiǎn)單入門(mén)指引,來(lái)自于IBM官網(wǎng)網(wǎng)站技術(shù)文檔,需要的朋友可以參考下
    2015-04-04
  • python切片及sys.argv[]用法詳解

    python切片及sys.argv[]用法詳解

    Sys.argv[]是用來(lái)獲取命令行參數(shù)的,sys.argv[0]表示代碼本身文件路徑,所以參數(shù)從1開(kāi)始。下面通過(guò)實(shí)例代碼給大家介紹python切片及sys.argv[]用法,需要的朋友參考下吧
    2018-05-05
  • Python OpenCV利用筆記本攝像頭實(shí)現(xiàn)人臉檢測(cè)

    Python OpenCV利用筆記本攝像頭實(shí)現(xiàn)人臉檢測(cè)

    這篇文章主要為大家詳細(xì)介紹了Python OpenCV利用筆記本攝像頭實(shí)現(xiàn)人臉檢測(cè),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-04-04

最新評(píng)論