【python+requests】接口自动化测试

这两天一直在找直接用python做接口自动化的方法,在网上也搜了一些博客参考,今天自己动手试了一下。

一、整体结构

上图是项目的目录结构,下面主要介绍下每个目录的作用。

Common:公共方法:主要放置公共的操作的类,比如数据库sqlhelper、文件操作类等

Config:公共变量:主要放置公共变量,比如ST、UAT、生产环境的url地址、用户名密码、数据库连接

Data:数据层,有点类似三层架构中的DAL,它是数据的来源,根据数据存放的格式再细分json、xml、表单和数据库

Log:日志层:存放日志,便于跟踪调试

Page:页面层:先把整个系统划分若干子系统,每个子系统包含若干页面。这个把用户操作的页面抽象成了page对象,页面的操作抽象成方法,这样测试人员可以传递不同的测试案例进行测试,如果是面向服务的纯接口性质的,没有页面那就没必要再这样划分,这样就把接口测试转换成了python的单元测试。

Result:存放单元测试的执行结果,也可以把每次执行的结果存到数据库打点,然后做测试结果趋势分析,如果后续把项目集成到Jenkins中的话,相当于Jenkins集成python单元测试,这样的话这层也可以不需要。

Case:测试案例层,针对上面Page对应的单个方法利用测试数据和期望数据进行assert判断,这里用到的测试数据和期望数据后续可以放在Excel中,测试人员只需填充测试数据。

Run:这里用来组装成suite然后进行运行案例。

二、测试
1.安装HTMLTestRunner

把它下载下来放到python安装目录的lib目录下

2.业务逻辑层

这里模拟一些业务处理,这里做接口自动化时会使用requests库进行请求。
 

  1. # -*- coding: utf-8 -*-

  2. import sys

  3. reload(sys)

  4. sys.setdefaultencoding('utf-8')

  5. import requests

  6. def Add(name,pwd):

  7. session=requests.session()

  8. response=session.get('http://www.baidu.com')

  9. print(response.status_code)

  10. return response.status_code==200

  11. def Edit(name,pwd):

  12. return {'name':name,'pwd':pwd}

  13. def Delete(name,pwd):

  14. return {'name':name,'pwd':pwd}

  15. def Search(name,pwd):

  16. return {'name':name,'pwd':pwd}

3.案例层

原本计划增加一个套件suite层,如果是单个接口的不加也可以,如果是多个接口进行流程测试,使用suite时案例的顺序就不会改变。如果是流程的,也可以写成case,只是里面需要多次调用业务逻辑层。

  1. # -*- coding: utf-8 -*-

  2. import sys

  3. reload(sys)

  4. sys.setdefaultencoding('utf-8')

  5. import unittest

  6. from Root.Page import Login

  7. from Root.Page.UserManager import Index

  8. import HTMLTestRunner

  9. import time

  10. class index(unittest.TestCase):

  11. def setUp(self):

  12. print('setUp')

  13. def tearDown(self):

  14. print('tearDown')

  15. def test_add(self):

  16. arr= Login.Login('admin', '123456')

  17. flag= Index.Add(arr[0], arr[1])

  18. self.assertTrue(flag)

  19. flag= Index.Add(arr[0], arr[1])

  20. self.assertTrue(flag==False)

  21. def test_edit(self):

  22. response= Login.Login('admin', '123456')

  23. dic= Index.Edit(response[0], response[1])

  24. self.assertNotEqual(dic,{'name':'123'})

  25. def test_delete(self):

  26. response= Login.Login('admin', '123456')

  27. dic= Index.Delete(response[0], response[1])

  28. self.assertNotEqual(dic,{'name':'123'})

4.运行

这里主要考虑可能整个系统会分成不同的模块进行运行,这样也能维护上也必将方便,可以多执行机执行。这里使用的HTMLTestRunner来生成报告.

  1. # -*- coding: utf-8 -*-

  2. import sys

  3. reload(sys)

  4. sys.setdefaultencoding('utf-8')

  5. import os

  6. import unittest

  7. from HTMLTestRunner import HTMLTestRunner

  8. from Root.Test.Case.UserManager import Index

  9. import HTMLTestRunner

  10. import time

  11. if __name__ == '__main__':

  12. # 1、构造用例集

  13. suite = unittest.TestSuite()

  14. # 2、执行顺序是安加载顺序:先执行test_sub,再执行test_add

  15. suite.addTest(Index.index("test_add"))

  16. suite.addTest(Index.index("test_edit"))

  17. suite.addTest(Index.index("test_delete"))

  18. suite.addTest(Index.index("test_edit"))

  19. suite.addTest(Index.index("test_edit"))

  20. filename = "../../../Result/{0}Report.html".format(time.strftime("%Y%m%d%H%M%S", time.localtime()) ) # 定义个报告存放路径,支持相对路径

  21. f = file(filename, 'wb') # 结果写入HTML 文件

  22. runner = HTMLTestRunner.HTMLTestRunner(stream=f, title='测试报告', description='XXX系统接口自动化测试测试报告',verbosity=2) # 使用HTMLTestRunner配置参数,输出报告路径、报告标题、描述

  23. runner.run(suite)

三、测试案例参数化

上面的每个单元测试只能运行一个测试案例的数据,就是如何实现参数化,这样配置一下案例数据就能运行多次单元测试,这样就会方便很多。找了下python自带的单元测试框架不支持,这里使用了nose和parameterized 。

  1. # -*- coding: utf-8 -*-

  2. import sys

  3. reload(sys)

  4. sys.setdefaultencoding('utf-8')

  5. from nose.tools import assert_equal

  6. from parameterized import parameterized

  7. import HTMLTestRunner

  8. import time

  9. import unittest

  10. import math

  11. @parameterized([

  12. (2, 2, 4),

  13. (2, 3, 8),

  14. (1, 9, 1),

  15. (0, 9, 0),

  16. ])

  17. def test_pow(base, exponent, expected):

  18. assert_equal(math.pow(base, exponent), expected)

  19. class TestMathUnitTest(unittest.TestCase):

  20. @parameterized.expand([

  21. ("negative", -1.5, -2.0),

  22. ("integer", 1, 1.0),

  23. ("large fraction", 1.6, 1),

  24. ])

  25. def test_floor(self, name, input, expected):

  26. assert_equal(math.floor(input), expected)

然后cmd跳转到该python文件的目录下,输入命令,它会把该文件中test开头的案例都跑了,然后就可以看到有一个案例运行输出结果的html文件.

nosetests testRuncase.py --with-html --html-report=nose_report2_test.html

 

总结:

感谢每一个认真阅读我文章的人!!!

作为一位过来人也是希望大家少走一些弯路,如果你不想再体验一次学习时找不到资料,没人解答问题,坚持几天便放弃的感受的话,在这里我给大家分享一些自动化测试的学习资源,希望能给你前进的路上带来帮助。

软件测试面试文档

我们学习必然是为了找到高薪的工作,下面这些面试题是来自阿里、腾讯、字节等一线互联网大厂最新的面试资料,并且有字节大佬给出了权威的解答,刷完这一套面试资料相信大家都能找到满意的工作。

 

          视频文档获取方式:
这份文档和视频资料,对于想从事【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴我走过了最艰难的路程,希望也能帮助到你!以上均可以分享,点下方小卡片即可自行领取。

  • 16
    点赞
  • 27
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值