优质文章,第一时间送达!
本项目实现接口自动化的技术选型:Python+Requests+Pytest+YAML+Allure,主要是针对之前开发的一个接口项目来进行学习,通过 Python+Requests 来发送和处理 来生成测试报告。
接口项目开发学习:
使用Flask开发简单接口(1)--GET请求接口()
使用Flask开发简单接口(2)--POST请求接口()
使用Flask开发简单接口(3)--引入MySQL ()
使用Flask开发简单接口(4)--借助Redis实现token验证()
使用Flask开发简单接口(5)--数据加密处理()
项目说明本项目在实现过程中,把整个项目拆分成请求方法封装、HTTP接口封装、关键字封装、测试用例等模块。
首先利用Python把输出测试报告。
当然,如果感兴趣的话,还可以再对接口自动化进行Jenkins持续集成。
GitHub项目源码地址:
项目结构api ====>> 接口封装层,如封装接口
common ====>> 各种工具类
core ====>> requests请求方法封装、关键字返回结果类
config ====>> 配置文件
data ====>> 测试数据文件管理
operation ====>> 关键字封装层,如把多个Python接口封装为关键字
pytest.ini ====>> pytest配置文件
requirements.txt ====>> 相关依赖包文件
testcases ====>> 测试用例
请求方法封装在 core/rest_client.py文件中,对Requests库下一些常见的请求方法进行了简单封装,以便调用起来更加方便。
class RestClient:def __init__(self, api_root_url):self.api_root_url = api_root_urlself.session = requests.sessiondef get(self, url, **kwargs):return self.request(url, "GET", **kwargs)def post(self, url, data=None, json=None, **kwargs):return self.request(url, "POST", data, json, **kwargs)def put(self, url, data=None, **kwargs):return self.request(url, "PUT", data, **kwargs)def delete(self, url, **kwargs):return self.request(url, "DELETE", **kwargs)def patch(self, url, data=None, **kwargs):return self.request(url, "PATCH", data, **kwargs)def request(self, url, method, data=None, json=None, **kwargs):url = self.api_root_url + urlheaders = dict(**kwargs).get("headers")params = dict(**kwargs).get("params")files = dict(**kwargs).get("params")cookies = dict(**kwargs).get("params")self.request_log(url, method, data, json, params, headers, files, cookies)if method == "GET":return self.session.get(url, **kwargs)if method == "POST":return requests.post(url, data, json, **kwargs)if method == "PUT":if json:# PUT 和 PATCH 中没有提供直接使用json参数的方法,因此需要用data来传入data = complexjson.dumps(json)return self.session.put(url, data, **kwargs)if method == "DELETE":return self.session.delete(url, **kwargs)if method == "PATCH":if json:data = complexjson.dumps(json)return self.session.patch(url, data, **kwargs)
下的请求。
class User(RestClient):def __init__(self, api_root_url, **kwargs):super(User, self).__init__(api_root_url, **kwargs)def list_all_users(self, **kwargs):return self.get("/users", **kwargs)def list_one_user(self, username, **kwargs):return self.get("/users/{}".format(username), **kwargs)def register(self, **kwargs):return self.post("/register", **kwargs)def login(self, **kwargs):return self.post("/login", **kwargs)def update(self, user_id, **kwargs):return self.put("/update/user/{}".format(user_id), **kwargs)def delete(self, name, **kwargs):return self.post("/delete/user/{}".format(name), **kwargs)
关键字返回结果类在 core/result_base.py下,定义了一个空类ResultBase,该类主要用于自定义关键字返回结果。
class ResultBase:pass"""自定义示例:result = ResultBaseresult.success = Falseresult.msg = res.json["msg"]result.response = res"""`在多流程的业务场景测试下,通过自定义期望保存的返回数据值,以便更好的进行断言。
关键字封装关键字应该是具有一定业务意义的,在封装关键字的时候,可以通过调用多个Python接口来完成。在某些情况下,比如测试一个充值接口的时候,在充值后可能需要调用查询接口得到最新账户余额,来判断查询结果与预期结果是否一致,那么可以这样来进行测试:
1, 首先,可以把 充值-查询的操作封装为一个关键字,在这个关键字中依次调用充值和查询的接口,并可以自定义关键字的返回结果。
2, 接着,在编写测试用例的时候,直接调用关键字来进行测试,这时就可以拿到关键字返回的结果,那么断言的时候,就可以直接对关键字返回结果进行断言。
测试用例层根据用例名分配测试数据
测试数据位于 data文件夹下,在这里使用YAML来管理测试数据,同时要求测试数据中第一层的名称,需要与测试用例的方法名保持一致,如test_get_all_user_info、test_delete_user。
test_get_all_user_info:# 期望结果,期望返回码,期望返回信息# except_result, except_code, except_msg- [True, 0, "查询成功"]省略test_delete_user:# 删除的用户名,期望结果,期望返回码,期望返回信息# username, except_result, except_code, except_msg- ["测试test", True, 0, "删除用户信息成功"]- ["wintest3", False, 3006, "该用户不允许删除"]这里借助 fixture方法,我们就能够通过request.function.__name__自动获取到当前执行用例的函数名testcase_name,当我们传入测试数据api_data之后,接着便可以使用api_data.get(testcase_name)来获取到对应用例的测试数据。
import pytestfrom testcases.conftest import api_data@pytest.fixture(scope="function")def testcase_data(request):testcase_name = request.function.__name__return api_data.get(testcase_name)`数据准备和清理
在接口自动化中,为了保证用例可稳定、重复地执行,我们还需要有测试前置操作和后置操作,即数据准备和数据清理工作。
@pytest.fixture(scope="function")def delete_register_user:"""注册用户前,先删除数据,用例执行之后,再次删除以清理数据"""del_sql = base_data["init_sql"]["delete_register_user"]db.execute_db(del_sql)logger.info("注册用户操作:清理用户--准备注册新用户")logger.info("执行前置SQL:{}".format(del_sql))yield # 用于唤醒 teardown 操作db.execute_db(del_sql)logger.info("注册用户操作:删除注册的用户")logger.info("执行后置SQL:{}".format(del_sql))在这里,以用户注册用例为例。对于前置操作,我们应该准备一条删除SQL,用于将数据库中已存在的相同用户删除,对于后置操作,我们应该再执行删除SQL,确保该测试数据正常完成清理工作。
在测试用例中,我们只需要在用例上传入 fixture的函数参数名delete_register_user,这样就可以调用fixture实现测试前置及后置操作。当然,也可以使用pytest装饰器@pytest.mark.usefixtures来完成,如:
@pytest.mark.usefixtures("delete_register_user")` Allure用例描述
在这里,我们结合 Allure 来实现输出测试报告,同时我们可以使用其装饰器来添加一些用例描述并显示到测试报告中,以便报告内容更加清晰、直观、可读。如使用 @allure.title自定义报告中显示的用例标题,使用@allure.description自定义用例的描述内容,使用@allure.step可在报告中显示操作步骤,使用@allure.issue可在报告中显示缺陷及其链接等。
@allure.step("步骤1 ==>> 注册用户")def step_1(username, password, telephone, sex, address):logger.info("步骤1 ==>> 注册用户 ==>> {}, {}, {}, {}, {}".format(username, password, telephone, sex, address))@allure.severity(allure.severity_level.NORMAL)@allure.epic("针对单个接口的测试")@allure.feature("用户注册模块")class TestUserRegister:"""用户注册"""@allure.story("用例--注册用户信息")@allure.description("该用例是针对获取用户注册接口的测试")@allure.issue("):省略`
项目部署首先,下载项目源码后,在根目录下找到 requirements.txt文件,然后通过 pip 工具安装 requirements.txt 依赖,执行命令:
pip3 install -r requirements.txt` 接着,修改 config/setting.ini配置文件,在Windows环境下,安装相应依赖之后,在命令行窗口执行命令:
pytest注意:因为我这里是针对自己的接口项目进行测试,如果想直接执行我的测试用例来查看效果,需要提前部署上面提到的接口项目。
测试报告效果展示在命令行执行命令:pytest运行用例后,会得到一个测试报告的原始文件,但这个时候还不能打开成HTML的报告,还需要在项目根目录下,执行命令启动allure服务:
# 需要提前配置allure环境,才可以直接使用命令行allure serve ./report`最终,可以看到测试报告的效果图如下:
来源:
回复下方「关键词」,获取优质资源
回复关键词「 pybook03」,立即获取主页君与小伙伴一起翻译的《Think Python 2e》电子版
回复关键词「入门资料」,立即获取主页君整理的 10 本 Python 入门书的电子版
回复关键词「m」,立即获取Python精选优质文章合集
回复关键词「」,将数字替换成 0 及以上数字,有惊喜好礼哦~
题图:pexels,CC0 授权。