什么是参数关联?
参数关联,也叫接口关联,即接口之间存在参数的联系或依赖。在完成某一功能业务时,有时需要按顺序请求多个接口,此时在某些接口之间可能会存在关联关系。比如:B接口的某个或某些请求参数是通过调用A接口获取的,即需要先请求A接口,从A接口的返回数据中拿到需要的字段值,在请求B接口时作为请求参数传入。
有哪些场景?
最常见的场景之一:
请求登录接口后获取到token值,后续其他接口请求时需要将token作为请求参数传入。
再比如下单–>支付场景,调用下单接口生成订单后会返回订单号,订单号传给支付接口进行支付。
1、参数关联场景
以网上购物为例,对应的场景及请求我们可以大致简化如下(可联想某宝购物流程):
用户在购物车选择商品点击【去结算】进入订单确认页,订单确认页点击【提交订单】这时会先请求下单接口创建订单
紧接着会拿创建的订单去请求获取支付凭证接口,这个接口会调起支付页面,也就是输入密码的支付界面
输入支付密码后会请求支付服务的支付接口进行实际支付,支付的结果会返回给请求方,告知是否支付成功
这个流程中涉及到的接口其实都是有关联的,我们要对整个流程进行测试那么就需要按顺序调用所有这些涉及到的接口。
但这里我们只需要弄明白参数关联,那么以下单接口和获取支付凭证接口为例,举例说明就够了,即先请求下单接口生成订单号,再拿这个订单号去请求获取支付凭证接口,才能调起支付界面并进行支付。
下单接口如下:
接口地址:<服务器>/trade/order/purchase
请求类型:post
请求参数:
{
"goodsId": 10, //商品id
"goodsSkuId": 33, //sku id
"num": 2, //购买数量
"tradePromotion": { //选择的优惠项
"type": 1, //类型<1:优惠券>
"promotionId": 1 //优惠id
}
}
返回值 data:
{
"code": 0,
"msg": "成功",
"data": {
"tradeNo": "0020220116204344962706666" //交易订单号
},
"t": 1639658625474
}
获取支付凭证接口如下:
接口地址:<服务器>/pay/pre/consum
请求类型:post
请求参数:
{
"orderNo":"0020220116204344962706666", //交易订单号
"product":"alipayWapClient" //支付渠道
}
返回值 data:
{
"code": 0,
"msg": "成功",
"data": {
"payNo":"123213213219379213",
"certificate": "\n"
},
"t": 1639659171031
}
其中orderNo字段使这两个接口关联了起来。因为每次生成的订单号都不一样,所以在测试这个场景的时候,就需要使这两个接口的参数进行关联,才能走通。
2、脚本编写
那么在pytest框架的自动化测试中,参数关联可以怎样处理呢?
两种思路如下:
根据业务场景的调用时序,在用例中按顺序调用接口
将依赖的接口编写成fixture函数,并使用yield返回下个接口需要的参数
1)在用例中按顺序调用
代码示例如下:
import requests
import json
import pytest
def test_order_pay():
'''
创建订单->获取支付凭证,调起支付界面
:return:
'''
# 先调用下单接口生成订单
url_order = "https://gouwu.com/trade/order/purchase"
data_order = {
"goodsId": 10,
"goodsSkuId": 33,
"num": 2,
"tradePromotion": {
"type": 1,
"promotionId": 1
},
"tradeDirectionArticle": {
"articleId": 1
}
}
res_order = requests.post(url=url_order, json=data_order).text
tradeNo = json.loads(res_order)["tradeNo"]
# 再请求获取支付凭证接口
url_pay = "https://gouwu.com/pay/pre/consum"
data_pay = {
"orderNo": tradeNo, # tradeNo通过下单接口获取
"product": "alipayWapClient"
}
res_pay = requests.post(url=url_pay, json=data_pay).text
res_pay = json.loads(res_pay)
# 断言
assert res_pay["code"]==0
assert res_pay["data"]["payNo"]
assert res_pay["data"]["certificate"]
if __name__ == '__main__':
pytest.main()
上面的代码只是流水式的进行调用,我们还可以先将每个接口请求封装成单独的函数,在测试用例中只需按照顺序调用这些函数即可,这个我们会在后续的文章中进行说明。
2)使用Fixture函数
pytest中Fixture函数可以参考我之前的文章pytest-Fixture(固件)
定义Fixture函数,代码示例如下:
@pytest.fixture()
def get_order():
'''
请求下单接口,创建订单
:return:
'''
url_order = "https://gouwu.com/trade/order/purchase"
data_order = {
"goodsId": 10,
"goodsSkuId": 33,
"num": 2,
"tradePromotion": {
"type": 1,
"promotionId": 1
},
"tradeDirectionArticle": {
"articleId": 1
}
}
res_order = requests.post(url=url_order, json=data_order).text
tradeNo = json.loads(res_order)["tradeNo"]
yield tradeNo
在测试函数中调用上面定义的fixture函数,代码示例如下:
def test_pay(get_order):
'''
下单->支付场景校验
:param get_order: 调用上面的Fixture函数,函数名get_order即返回的tradeNo
:return:
'''
url_pay = "https://gouwu.com/pay/pre/consum"
data_pay = {
"orderNo": get_order, # get_order即为上面定义的fixture函数返回值
"product": "alipayWapClient"
}
res_pay = requests.post(url=url_pay, json=data_pay).text
res_pay = json.loads(res_pay)
# 断言
assert res_pay["code"] == 0
assert res_pay["data"]["payNo"]
assert res_pay["data"]["certificate"]
参数关联在接口自动化测试中是必然会遇到的场景,设计关联参数的用例会直接影响到用例的维护,当然这也是在接口自动化项目的架构设计时就需要考虑的问题。
下面是我整理的2023年最全的软件测试工程师学习知识架构体系图 |
不经历风雨,怎能见彩虹;不经历努力,难有成长。踏上奋斗之路,背负勇气与激情,展翅翱翔,绽放光芒。披荆斩棘,冲破困境,唯有坚持不懈,方能书写辉煌篇章。
生命如一朵绽放的花朵,奋斗是它的花蕊。执着的努力,化作无限可能;坚定的信念,引领人生航向。不畏困难,放飞梦想的翅膀,让奋斗的火焰照亮前行的路途。
努力是人生的密码,奋斗是成功的钥匙。不管困境多么艰巨,只要坚持不懈,勇往直前,就一定能跨过障碍,迎接辉煌。相信自己,敢于拼搏,每一次奋斗都离成功更近一步,让梦想的火焰在心中燃烧!