当前位置: 首页> 技术文章> Python接口框架封装

Python接口框架封装

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

一:接口项目

二、接口用例

三、代码阶段

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

upfile1601027355216.png

测试用例和lib库

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

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

import requestsimport unittestimport loggingimport addEventDataTemplateimport getNewIDfrom 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 datetimedef 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

最后的测试结果:


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

上一篇: Pytest 单元测试框架之断言

下一篇: 软件测试之手工测试人员如何转测试开发?

QQ技术交流群

多测师官方学习交流
556733550

加入群聊