通过Django Admin+HttpRunner1.5.6实现简易接口测试平台

前言

这是一个使用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,内容如下。

串存,读取时转为Python字典或列表。
在apitest目录下新建fields.py,内容如下。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

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,添加:

1

2

3

4

5

6

7

8

9

10

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,添加:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

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的格式如下:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

{'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属性方法,代码如下:

1

2

3

4

5

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文件

1

2

3

- utils

   - config.yaml # 空文件即可

   - debugtalk.py

config的格式可以为:

1

2

3

4

5

config:

   name: ...

   request: ...

   variables: ...

   path: .../config.yaml

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

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

1

2

3

4

5

6

7

8

9

10

11

@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方法。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

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对象保存本次运行结果。

模型完整代码

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

84

85

86

87

88

89

90

91

92

93

94

95

96

97

98

99

100

101

102

103

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,代码如下:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

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: 1b: 2

请求数据:

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

提取请求:

- res_url: content.url

断言:

- eq: [status_code, 200]

点击保存。

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

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

​现在我也找了很多测试的朋友,做了一个分享技术的交流群,共享了很多我们收集的技术文档和视频教程。
如果你不想再体验自学时找不到资源,没人解答问题,坚持几天便放弃的感受
可以加入我们一起交流。而且还有很多在自动化,性能,安全,测试开发等等方面有一定建树的技术大牛
分享他们的经验,还会分享很多直播讲座和技术沙龙
可以免费学习!划重点!开源的!!!
qq群号:485187702【暗号:csdn11】

最后感谢每一个认真阅读我文章的人,看着粉丝一路的上涨和关注,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走! 希望能帮助到你!【100%无套路免费领取】

  • 21
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Django是一款基于Python语言的Web框架,它可以让开发者更加高效地进行Web应用程序的开发。而测试平台是一个用于测试软件产品的工具,具有重要的测试功能。那么,如何将两者结合起来呢? 首先,我们需要基于Django框架来进行测试平台的开发。在开发过程中,可以使用Python内置的unittest库来进行单元测试,并且还可以使用第三方工具进行集成测试,例如Selenium等。在测试平台的代码开发过程中,需要考虑到数据的输入、输出和处理,以及测试用例的管理、执行和报告生成等方面。此外,还需要考虑测试环境的搭建和配置,以确保测试的准确性和有效性。 另外,测试平台的代码需要具备高可维护性和可扩展性。为了实现这些目标,我们可以采用以下方法来进行代码设计: 1. 合理划分模块:根据主要功能进行模块划分,如测试用例管理模块、执行模块、报告生成模块等。这样,可以使代码更易于维护和扩展。 2. 规范命名和注释:为变量、函数、类等元素命名,并添加注释,以便开发人员理解和维护代码。 3. 遵循设计模式和最佳实践:采用设计模式和最佳实践可以使代码更具可读性、可维护性和可扩展性。 4. 编写单元测试:编写单元测试可以有效地检测代码逻辑的正确性,并可以在开发过程中不断地进行测试。 综上所述,django python测试平台代码需要根据测试要求进行开发和测试,并且需要考虑到代码设计、测试环境搭建和代码可维护性等方面。只有这样,才能保证测试工作的质量和有效性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值