使用python完成接口自动化

关键词:接口、python、requests、unittest
引言:一般对于自动化的理解,有两种方式的自动化。
第一,不需要写代码,完全由工具实现,这种方式的工具一般是公司自己研发的,方便黑盒测试人员使用。这种工具的特点是学习成本低,方便使用,但是通用性不强,也就是换了一家公司,就很有可能无法使用之前的工具。
第二,需要自己写代码,在别人的框架下编写代码测试,或者是需要自己搭建自动化测试框架。这种方式对测试人员的代码要求高,学习成本高,但是通用性很强,去任何一家都可以用这套东西。

鉴于以上介绍,本文当然是介绍第二种方式了。
我们的课题是接口自动化,我们的目的是使用python进行接口测试,并完成输出测试报告。我们需要用到的东西有如下:python3,unittest,requests。

一:接口项目
我们使用的项目是发布会签到系统。总共有5个接口,虽然不多,但足够学习使用。
接口文档如下:

添加发布会接口.png

查询发布会接口.png
添加嘉宾接口.png
查询嘉宾接口.png

发布会签到接口.png

二、接口用例
软件测试都需要写测试用例,不管你做的性能,自动化还是其它任何的测试工作。
真实的工作写接口的测试用例,可能考虑很多场景,如接口的功能(正常场景),接口的边界等价,接口的异常场景,接口参数组合,接口的性能等等。本文采用输出法分析,根据出参的不同设计出测试用例。详细用例参考如下:(用例太小看不清楚,可以查看原图,然后放大)

接口测试用例.png

三、代码阶段
3.1 框架的设计
我们使用unittest框架,case目录存放所有的测试用例,lib目录存放自己封装的一些代码,result目录存放测试结果和测试日志,runner.py是主程序。

image.png

3.2 主程序 runner.py
这个主程序跟之前的《selenium unittest实战》文章类似,不再详细介绍,不太一样的地方是使用一个logging模块。不知道大家有没有感受,测试接口的时候,想看完整的请求和响应,以便分析定位问题。

import  unittest
import  time
import os
import logging
from HTMLTestRunner import  HTMLTestRunner

#获取项目的根目录
test_dir = os.path.join(os.getcwd())

# 自动搜索项目根目录下的所有case,构造测试集;返回TestSuite对象
discover = unittest.defaultTestLoader.discover(test_dir, pattern='test_*.py')

# 实例化TextTestRunner类
# runner = unittest.TextTestRunner(verbosity=2)

now = time.strftime('%Y-%m-%d %H_%M_%S')  # 获取当前日期
result = test_dir+ '\\result\\'+now + '_result.html'  # 测试报告的完整路径
log = test_dir+'\\result\\'+now+'_log.txt'  #日志的完整路径

logging.basicConfig(filename=log,level=logging.INFO,format = '%(asctime)s - %(name)s - %(levelname)s - %(message)s') #filename 日志文件路径 level 日志的级别 format 格式

fp = open(result, 'wb')  # wb方式写入
runner = HTMLTestRunner(stream=fp, title='测试报告', description='aguest_master项目用例执行情况',verbosity=2)  #构造runner

# 使用run()方法运行测试套件(即运行测试套件中的所有用例)
runner.run(discover)

3.3 测试用例和lib库
1) 由于总共只有5个接口,所以设计为5个代码文件,分别为:test_add_event,py,test_add_guest.py,test_get_event_list.py,test_get_guest_list.py,test_user_sign.py。

image.png

2)我们使用python的requests测试接口,这个库大名鼎鼎,而且官网还有中文。
官网网址:http://docs.python-requests.org/zh_CN/latest/

3)每个代码文件都是一个接口,接口的url地址是固定的,所以设计成类属性,方便后续测试用例使用。


image.png

4)每个测试用例都写明代码逻辑,方便以后调试。
5)如果遇到经常调用的东西,如获取最新发布会ID,获取添加发布会body数据,都封装成库。
6)最后根据出参的状态码断言是否成功
7)使用logging.info 记录每个测试用例的日志情况


image.png

添加发布会接口代码文件:test_add_event.py

import requests
import unittest
import logging
import addEventDataTemplate
import getNewID
from urllib import  parse  #使用requests发送post请求,body的汉字会进行url编码,即%xx形式。想看到原始body,需要使用parse.unquote进入url解码
class Test_addEvent(unittest.TestCase):
    '''添加发布会接口'''
    @classmethod
    def setUpClass(cls):
        cls.url="http://127.0.0.1:8000/api/add_event/"
    @classmethod
    def tearDownClass(cls):
        pass
    def setUp(self):
        pass
    def tearDown(self):
        pass
    def test_00(self):   #代码逻辑::获取当前最新发布会ID,设置入参,eid置空,发送post请求
        '''添加发布会-eid为空'''
        id=getNewID.getNewID()   #获取当前最新发布会ID
        data=addEventDataTemplate.getEventData(id)  #获取添加发布会的数据模板
        data['eid']=''  #eid为空,即参数错误
        r=requests.post(self.url,data=data)
        status=r.json()['status']
        self.assertEqual(10021,status)
        logging.info(f"case:添加发布会,eid为空\n请求地址:{r.url}\t请求方式:{r.request.method}\n请求头:{r.request.headers}\n请求正文:{parse.unquote(r.request.body)}\n响应头:{r.headers}\n响应正文:{r.text}\n")

    def test_01(self):    #代码逻辑::获取当前最新发布会ID,设置入参,发送post请求
        '''添加发布会-成功'''
        id = getNewID.getNewID()  # 获取当前最新发布会ID
        data = addEventDataTemplate.getEventData(id)#获取添加最新发布会的数据模板
        r=requests.post(self.url,data=data)
        status=r.json()['status']
        self.assertEqual(10000,status)
        logging.info(f"case:添加发布会,成功\n请求地址:{r.url}\t请求方式:{r.request.method}\n请求头:{r.request.headers}\n请求正文:{parse.unquote(r.request.body)}\n响应头:{r.headers}\n响应正文:{r.text}\n")

    def test_02(self):   #代码逻辑::获取当前最新发布会ID,ID-1即为发布会已经存在的ID(发布会ID是递增加1)
        '''添加发布会-发布会ID已存在'''
        id = getNewID.getNewID()  # 获取当前最新发布会ID
        data=addEventDataTemplate.getEventData(id)#获取添加最新发布会的数据模板
        data['eid']=data['eid']-1  #最新模板ID减一即为重复ID
        r=requests.post(self.url,data=data)
        status = r.json()['status']
        self.assertEqual(10022, status)
        logging.info(f"case:添加发布会,发布会ID已存在\n请求地址:{r.url}\t请求方式:{r.request.method}\n请求头:{r.request.headers}\n请求正文:{parse.unquote(r.request.body)}\n响应头:{r.headers}\n响应正文:{r.text}\n")
    def test_03(self):   #代码逻辑::先新增发布会,再获取最新发布会ID,设置入参的name为重复。
        '''添加发布会-发布会标题已存在'''
        #新增发布会
        id = getNewID.getNewID()  # 获取当前最新发布会ID
        r=requests.post(self.url,data=addEventDataTemplate.getEventData(id))  #先新增一个发布会

        id = getNewID.getNewID()  # 获取当前最新发布会ID
        data = addEventDataTemplate.getEventData(id)#获取添加最新发布会的数据模板
        data['name']=f'发布会测试标题{id}' #最新模板ID减一,标题即为重复
        r=requests.post(self.url,data=data)
        status = r.json()['status']
        self.assertEqual(10023,status)
        logging.info(f"case:添加发布会,发布会标题已存在\n请求地址:{r.url}\t请求方式:{r.request.method}\n请求头:{r.request.headers}\n请求正文:{parse.unquote(r.request.body)}\n响应头:{r.headers}\n响应正文:{r.text}\n")
    def test_04(self):   #代码逻辑::获取最新发布会ID,设置入参,开始时间:改为-,再提交请求
        '''添加发布会-发布会时间错误'''
        id = getNewID.getNewID()  # 获取当前最新发布会ID
        data = addEventDataTemplate.getEventData(id)#获取添加最新发布会的数据模板
        data['start_time']=data['start_time'].replace(':','-')  #时间 : 改为 - ,即为时间错误
        r = requests.post(self.url, data=data)
        status = r.json()['status']
        self.assertEqual(10024,status)
        logging.info(f"case:添加发布会,发布会时间错误\n请求地址:{r.url}\t请求方式:{r.request.method}\n请求头:{r.request.headers}\n请求正文:{parse.unquote(r.request.body)}\n响应头:{r.headers}\n响应正文:{r.text}\n")
if __name__ == '__main__':
    unittest.main(verbosity=2)


lib库 getNewID.py:

def getNewID():
    '''获取最新的(最大的)发布会编号id'''
    import sqlite3
    con=sqlite3.connect(r'D:\backup\guest2-master\db.sqlite3')
    cur=con.cursor()
    cur.execute("select max(id) from sign_event")
    new_id=cur.fetchone()
    new_id=new_id[0]
    cur.close()
    con.close()
    return new_id

lib库 addEventDataTemplate.py:

import datetime
def getEventData(id):
    '''添加发布会 body模板'''
    startTime=(datetime.datetime.now()+datetime.timedelta(days=30)).strftime("%Y-%m-%d %H:%M:00")  #获得当前时间,并往后30天为发布会时间

    data={
          'eid':id+1,
          'name':f"发布会测试标题{id+1}",  #当前发布会编号加1
          'limit':100,    #默认值
          'status':1,    #默认值
          'address':'新街口金鹰',    #默认值
          'start_time':startTime  #%格式   Y-%m-%d %H:%M:00
        }
    return data

由于篇幅的原因,其它的代码省略。

最后的测试结果:


测试报告.png

日志结果如下:


日志.png

总结:在写代码的过程中,每个测试用例的代码逻辑非常重要,不管是什么逻辑,得保证每个测试用例代码都可以独立运行,不会产生耦合。还有在测试接口的时候,经常与数据库打交道,比如获取数据,判断测试结果等。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,294评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,493评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,790评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,595评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,718评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,906评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,053评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,797评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,250评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,570评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,711评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,388评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,018评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,796评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,023评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,461评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,595评论 2 350

推荐阅读更多精彩内容