草庐IT

Python+Requests实现接口自动化

乐却思蜀 2023-03-28 原文

一般对于自动化的理解,有两种方式的自动化。

第一,不需要写代码,完全由工具实现,这种方式的工具一般是公司自己研发的,方便黑盒测试人员使用。这种工具的特点是学习成本低,方便使用,但是通用性不强,也就是换了一家公司,就很有可能无法使用之前的工具。

第二,需要自己写代码,在别人的框架下编写代码测试,或者是需要自己搭建自动化测试框架。这种方式对测试人员的代码要求高,学习成本高,但是通用性很强,去任何一家都可以用这套东西。
鉴于以上介绍,本文当然是介绍第二种方式了。

接口自动化测试,我们的目的是使用python进行接口测试,并完成输出测试报告。我们需要用到的东西有如下:python3,unittest,requests。

一、接口项目

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

接口文档如下:

 

 

 

二、接口用例 

软件测试都需要写测试用例,不管你做的性能,自动化还是其它任何的测试工作。

真实的工作写接口的测试用例,可能考虑很多场景,如接口的功能(正常场景),接口的边界等价,接口的异常场景,接口参数组合,接口的性能等等。本文采用输出法分析,根据出参的不同设计出测试用例。详细用例参考如下:(用例太小看不清楚,可以查看原图,然后放大) 

三、代码阶段

3.1 框架的设计

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

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。

2)我们使用python的requests测试接口,这个库大名鼎鼎,而且官网还有中文。

官网网址:

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

添加发布会接口代码文件: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

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

请点查看公告处加入测试行业圈方式,(←可点击查看公告处),在线解答哦

最后的测试结果:

日志结果如下:

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

看完这篇内容后,相信以下两件事,也会对你的个人提升有所帮助:

 - 将来的你定会感谢现在拼命努力的自己 

 

> > > 学习路线+测试实用干货精选汇总,需要可以关注一下我的公众号,公众号长期做「有价值的输出」是最低标准,感谢您的阅读。

> > > 一起探讨交流,共同学习软件测试技术、进测试Q群,里面有我们收集的配套教程和技术文档提供给自学的伙伴。请点此处查看公告处加入社区方式,(←可点击查看公告处)

> > > 荐语:与其花时间吃点点点苦,不如学学自动化,提高薪资然后摸鱼,不香么?

? 往期技术文章推荐

测试开发技能(1):持续集成这样做,App自动化测试效率提高50%

测试开发技能(2):性能测试知识学习路线(看看这篇,好好学习)

测试开发技能(3):十分钟弄懂最快的APP自动化工具uiautomator2 

有关Python+Requests实现接口自动化的更多相关文章

  1. ruby-on-rails - 使用 Ruby on Rails 进行自动化测试 - 最佳实践 - 2

    很好奇,就使用ruby​​onrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提

  2. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

    关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

  3. ruby - RuntimeError(自动加载常量 Apps 多线程时检测到循环依赖 - 2

    我收到这个错误:RuntimeError(自动加载常量Apps时检测到循环依赖当我使用多线程时。下面是我的代码。为什么会这样?我尝试多线程的原因是因为我正在编写一个HTML抓取应用程序。对Nokogiri::HTML(open())的调用是一个同步阻塞调用,需要1秒才能返回,我有100,000多个页面要访问,所以我试图运行多个线程来解决这个问题。有更好的方法吗?classToolsController0)app.website=array.join(',')putsapp.websiteelseapp.website="NONE"endapp.saveapps=Apps.order("

  4. ruby - 如何根据特征实现 FactoryGirl 的条件行为 - 2

    我有一个用户工厂。我希望默认情况下确认用户。但是鉴于unconfirmed特征,我不希望它们被确认。虽然我有一个基于实现细节而不是抽象的工作实现,但我想知道如何正确地做到这一点。factory:userdoafter(:create)do|user,evaluator|#unwantedimplementationdetailshereunlessFactoryGirl.factories[:user].defined_traits.map(&:name).include?(:unconfirmed)user.confirm!endendtrait:unconfirmeddoenden

  5. Python 相当于 Perl/Ruby ||= - 2

    这个问题在这里已经有了答案:关闭10年前。PossibleDuplicate:Pythonconditionalassignmentoperator对于这样一个简单的问题表示歉意,但是谷歌搜索||=并不是很有帮助;)Python中是否有与Ruby和Perl中的||=语句等效的语句?例如:foo="hey"foo||="what"#assignfooifit'sundefined#fooisstill"hey"bar||="yeah"#baris"yeah"另外,类似这样的东西的通用术语是什么?条件分配是我的第一个猜测,但Wikipediapage跟我想的不太一样。

  6. java - 什么相当于 ruby​​ 的 rack 或 python 的 Java wsgi? - 2

    什么是ruby​​的rack或python的Java的wsgi?还有一个路由库。 最佳答案 来自Python标准PEP333:Bycontrast,althoughJavahasjustasmanywebapplicationframeworksavailable,Java's"servlet"APImakesitpossibleforapplicationswrittenwithanyJavawebapplicationframeworktoruninanywebserverthatsupportstheservletAPI.ht

  7. 华为OD机试用Python实现 -【明明的随机数】 2023Q1A - 2

    华为OD机试题本篇题目:明明的随机数题目输入描述输出描述:示例1输入输出说明代码编写思路最近更新的博客华为od2023|什么是华为od,od薪资待遇,od机试题清单华为OD机试真题大全,用Python解华为机试题|机试宝典【华为OD机试】全流程解析+经验分享,题型分享,防作弊指南华为o

  8. python - 如何读取 MIDI 文件、更改其乐器并将其写回? - 2

    我想解析一个已经存在的.mid文件,改变它的乐器,例如从“acousticgrandpiano”到“violin”,然后将它保存回去或作为另一个.mid文件。根据我在文档中看到的内容,该乐器通过program_change或patch_change指令进行了更改,但我找不到任何在已经存在的MIDI文件中执行此操作的库.他们似乎都只支持从头开始创建的MIDI文件。 最佳答案 MIDIpackage会为您完成此操作,但具体方法取决于midi文件的原始内容。一个MIDI文件由一个或多个音轨组成,每个音轨是十六个channel中任何一个上的

  9. 基于C#实现简易绘图工具【100010177】 - 2

    C#实现简易绘图工具一.引言实验目的:通过制作窗体应用程序(C#画图软件),熟悉基本的窗体设计过程以及控件设计,事件处理等,熟悉使用C#的winform窗体进行绘图的基本步骤,对于面向对象编程有更加深刻的体会.Tutorial任务设计一个具有基本功能的画图软件**·包括简单的新建文件,保存,重新绘图等功能**·实现一些基本图形的绘制,包括铅笔和基本形状等,学习橡皮工具的创建**·设计一个合理舒适的UI界面**注明:你可能需要先了解一些关于winform窗体应用程序绘图的基本知识,以及关于GDI+类和结构的知识二.实验环境Windows系统下的visualstudio2017C#窗体应用程序三.

  10. ruby-on-rails - 从应用程序中自定义文件夹内的命名空间自动加载 - 2

    我们目前正在为ROR3.2开发自定义cms引擎。在这个过程中,我们希望成为我们的rails应用程序中的一等公民的几个类类型起源,这意味着它们应该驻留在应用程序的app文件夹下,它是插件。目前我们有以下类型:数据源数据类型查看我在app文件夹下创建了多个目录来保存这些:应用/数据源应用/数据类型应用/View更多类型将随之而来,我有点担心应用程序文件夹被这么多目录污染。因此,我想将它们移动到一个子目录/模块中,该子目录/模块包含cms定义的所有类型。所有类都应位于MyCms命名空间内,目录布局应如下所示:应用程序/my_cms/data_source应用程序/my_cms/data_ty

随机推荐