python+pytest接口自动化-请求参数格式的确定

我们在做接口测试之前,先需要根据接口文档或抓包接口数据,搞清楚被测接口的详细内容,其中就包含请求参数的编码格式,从而使用对应的参数格式发送请求。例如某个接口规定的请求主体的编码方式为 application/json,那么在请求该接口时,请求参数格式必须是 json 格式,使用其他的编码方式请求不会成功。

那么,在http请求中,请求主体常用的编码方式有哪些?每种编码方式在python中需要对应使用什么样格式的请求参数?这便是咱们在本篇博客要弄明白的地方。

content-type

在POST请求中,由content-type指定请求参数的格式,所以,为了搞清楚请求参数的编码格式,我们有必要了解HTTP请求头信息中的 content-type 字段。

content-type的作用

在HTTP协议中,报文通常包括两个部分 请求头部(head)、请求主体(body),其中 body 可以为空,如 GET 请求是将请求参数放在请求URL中而不是放在 body 中。

而POST请求中的请求参数则是放在 body 中,接收请求的一端(也就是服务器)需要知道传过来的 body 是什么类型的数据,采用怎样的编码方式,才能对数据进行对应的解析,这时就需要在请求头中使用 content-type 来指明 body 的媒体类型

媒体类型

媒体类型(通常称为 Multipurpose Internet Mail Extensions 或 MIME 类型 )是一种标准,用来表示文档、文件或字节流的性质和格式。它在IETF RFC 6838中进行了定义和标准化。

浏览器通常使用MIME类型(而不是文件扩展名)来确定如何处理URL,因此Web服务器在响应头中添加正确的MIME类型非常重要。如果配置不正确,浏览器可能会曲解文件内容,网站将无法正常工作,并且下载的文件也会被错误处理。

而浏览器传过来的内容也需要指定媒体类型,服务器才能根据媒体类型做对应的数据解析。

常见的媒体类型如下:

另外还有 multipart 类型,表示细分领域的文件类型的种类,经常对应不同的 MIME 类型,用于文件的上传,分为如下两种:

  1. multipart/form-data

  2. multipart/byteranges

这里均只做简单说明,具体每种类型的使用详情可自行查找相关资料。

总之,在接口测试中,请求头中的 Content-Type 作用就是,用来告知服务端请求body的编码方式。GET请求因为body为空,所以在GET请求中没有Content-Type字段。

get请求参数格式

我们已经知道GET请求的请求参数是直接放在URL中的,且不需要content-type指定媒体类型。而GET请求中的请求参数的编码格式为query string params。

query string params

说明

query string params 格式,参数会以 url string 的形式进行传递,即?后的字符串则为其请求参数,并以&作为分隔符,参数编写方式为?key=value&key=value,拼接在 url 后面。通常用于GET请求,除此之外其他有些请求方式也可以使用这种格式。

在浏览器中打开百度,搜索给你一页白纸-博客园,通过F12抓包也可以看到 Payload 中请求参数的格式为 Query String Parameters,如下图:

即打开URL链接https://www.baidu.com/s?ie=utf-8&wd=给你一页白纸-博客园

python代码发送请求

如果使用python中的 requests.get() 对上图示例发送get请求,则需使用参数 params,参数值为dict(字典)格式即可,示例如下:


import requests


url = "http://www.baidu.com/s"

params = {"wd": "给你一页白纸-博客园", "ie": "utf-8"}

res = requests.get(url=url, params=params)

print(res.text)

GET请求中请求主体编码格式固定,在做接口测试时基本无需确定其编码格式,较为简单。

post请求参数格式

对于HTTP协议中的POST请求,其请求参数有不同的编码格式。

服务端通常是根据请求头(headers)中的 Content-Type 字段来获知请求中的消息主体是用何种方式编码 (即媒体类型),再对请求参数进行对应方式的解析。

post请求常见的编码方式有四种:

  1. application/x-www-form-urlencoded

  2. multipart/form-data

  3. application/json

  4. text/xml

application/x-www-form-urlencoded

application/x-www-form-urlencoded 是最常见的 POST 提交数据的方式,用于表单数据的提交。浏览器的原生form表单,如果不设置enctype属性,那么最终就会默认以 application/x-www-form-urlencoded 方式提交数据。

说明

POST请求使用 application/x-www-form-urlencoded 对请求参数进行编码时,有以下特点:

  • 请求头header中content-type的值为:application/x-www-form-urlencoded
  • 请求参数会按照 key1=value1&key2=value2 的方式进行编码,且 key 和 value 都进行了 URL 转码
  • 服务器收到请求后,会对应的方式对这种编码格式的请求参数进行解析
  • 虽然都将请求参数进行了 key1=value1&key2=value2 形式编码,但GET请求中会将编码后的内容拼接在URL后。

以请求TesterHome网登陆接口为例:

Request Headers中的 content-type 为 application/x-www-form-urlencoded; charset=UTF-8。

python代码发送请求

使用python中的 requests.post() 请求上面图片中的接口时,需使用参数data,参数值为dict(字典)格式即可,代码如下:

 

import requests


'''

请求头的content-type为application/x-www-form-urlencoded

'''


data = {

"user[login]": "账号",

"user[password]": "密码",

"user[remember_me]": 0,

"commit": "登录"

}

headers = {

"user-agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/98.0.4758.82 Safari/537.36",

"content-type": "application/x-www-form-urlencoded; charset=UTF-8"

}

url = "https://testerhome.com/account/sign_in"


# 编码格式为application/x-www-form-urlencoded;charset=UTF-8,使用data参数,参数值为dict,

res = requests.post(url=url, headers=headers, data=data)

print(res.text)
multipart/form-data

multipart/form-data 也是一个常见的 POST 数据提交的方式,用于上传文件。我们使用表单上传文件时,必须让 form 表单的enctype等于 multipart/form-data。

说明

POST 请求使用 multipart/form-data 对请求参数进行编码时,有以下特点:

  1. 使用 boundary 用于分割不同的字段
  2. 消息主体中按照字段个数又分为多个结构类似的部分,每部分都以--boundary开始,紧接着下一行是内容描述信息,再下一行是字段具体内容(文本或二进制)。如果传输的是文件,还要包含文件名和文件类型信息
  3. 消息主体最后以 --boundary-- 标示结束

示例如下:


POST http://www.example.com HTTP/1.1

Content-Type:multipart/form-data; boundary=----WebKitFormBoundary8G1vtgT1pXWqqHzV


------WebKitFormBoundary8G1vtgT1pXWqqHzV

Content-Disposition: form-data; name="txt"


title

------WebKitFormBoundary8G1vtgT1pXWqqHzV

Content-Disposition: form-data; name="file"; filename="blog.png"

Content-Type: image/png


PNG ... content of blog.png ...

------WebKitFormBoundary8G1vtgT1pXWqqHzV--

python代码发送请求

例如,我们在请求牛图网的上传图片的接口时,Requests Headers 中为content-type: multipart/form-data; boundary=----WebKitFormBoundary4aA3ZrkOVwUIvmx0,如下图所示:

在python中使用requests.post()请求该接口,代码示例如下:

 

import requests


'''

请求头的content-type为multipart/form-data

'''


def post_mulitpart_form_data():

headers = {

"user-agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/98.0.4758.82 Safari/537.36",

"content-type": "multipart/form-data; boundary=----WebKitFormBoundary4aA3ZrkOVwUIvmx0"

}

url = "https://www.niupic.com/api/upload"


filepath = "./dianzan.jpg"

# 读取上传文件的内容

files = {"file": open(filepath, "rb")}

# 使用files参数接收请求内容,即读取的上传文件内容

res = requests.post(url=url, headers=headers, files=files).content

print(json.loads(res))



if __name__ == '__main__':

post_mulitpart_form_data()

运行结果如下:


C:\Users\xiaoqq\AppData\Local\Programs\Python\Python37\python.exe E:/blog/python接口自动化/flask_demo/test.py

{'status': 'success', 'code': 200, 'data': 'https://i.niupic.com/images/2022/02/20/9V3n.jpg', 'msg': '上传成功!'}


Process finished with exit code 0

application/x-www-form-urlencoded 与 multipart/form-data 这两种 POST 请求的数据编码方式,都是浏览器原生支持的,且现阶段标准中原生form表单也只支持这两种方式(通过form元素的enctype属性指定,默认为 application/x-www-form-urlencoded。其实enctype还支持text/plain,不过用得非常少)。

application/json

在一般公司的普通业务场景中,application/json 很常见,用来告诉服务端,消息主体是序列化后的json字符串,即前端传给服务端的数据是json格式的。

也就是说,如果请求头中 content-type 为 application/json,那么我们在使用工具如postman或python脚本模拟请求接口时,请求参数也需要先转换成json格式,然后才能发送请求。代码示例如下:

 

import requests

import json


'''

请求头的content-type为application/json

'''


headers = {"Content-Type": "application/json;charset=utf8"}

url = "http://127.0.0.1:5000/login"

_data = {

"username": "lilei",

"password": "123456"

}


# 这里使用json参数,即json=_data

res = requests.post(url=url, headers=headers, json=_data).text

# 当然还可以使用data参数,但需先将_data转换为json格式,即data=json.dumps(_data)

# json.dumps()将dict格式转换成json格式

res = requests.post(url=url, headers=headers, data=json.dumps(_data)).text

print(res)
text/xml

post请求中,有些请求主体的编码格式为 text/xml,即请求头中content-type 字段对应值为 text/xml,对于这样的接口,我们需要使用xml格式的参数去发送请求。

使用 requests.post() 发送请求参数为xml格式的post请求时,只需要将xml文件中的body部分写成一个字符串类型就行,遇到换行时在后面加个反斜杠,并将这个字符串赋值给data参数。代码示例如下:

 

import requests


'''

请求头的content-type为text/xml

'''


def post_text_xml():

headers = {"Content-Type": "text/xml"}

url = "http://httpbin.org/post"


body = '<?xml version="1.0" encoding = "UTF-8"?>' \

'<COM>' \

'<REQ name="给你一页白纸">' \

'<USER_ID></USER_ID>' \

'<COMMODITY_ID>111111</COMMODITY_ID>' \

'<SESSION_ID>asdfghjklfr0123</SESSION_ID>' \

'</REQ>' \

'</COM>'


res = requests.post(url=url, headers=headers, data=body.encode("utf-8")).text

print(res)



if __name__ == '__main__':

post_text_xml()

运行结果如下:

 

C:\Users\xiaoqq\AppData\Local\Programs\Python\Python37\python.exe E:/blog/python接口自动化/flask_demo/test.py

{

"args": {},

"data": "<?xml version=\"1.0\" encoding = \"UTF-8\"?><COM><REQ name=\"\u7ed9\u4f60\u4e00\u9875\u767d\u7eb8\"><USER_ID></USER_ID><COMMODITY_ID>111111</COMMODITY_ID><SESSION_ID>asdfghjklfr0123</SESSION_ID></REQ></COM>",

"files": {},

"form": {},

"headers": {

"Accept": "*/*",

"Accept-Encoding": "gzip, deflate",

"Content-Length": "182",

"Content-Type": "text/xml",

"Host": "httpbin.org",

"User-Agent": "python-requests/2.24.0",

"X-Amzn-Trace-Id": "Root=1-6211ebd3-2cc90293777649ba01e50b08"

},

"json": null,

"origin": "101.71.37.212",

"url": "http://httpbin.org/post"

}



Process finished with exit code 0
总结

这里只介绍了HTTP协议中 GET请求 和 POST请求 常见的请求参数编码格式,怎样确定请求参数的编码格式,以及在python代码中使用requests发送请求时,需要使用怎样格式的参数。

我们总结如下:

  1. GET请求,请求参数编码格式为query string params,requests.get() 发送请求时使用params参数,params的值 (即请求参数) 为字典格式
  2. POST请求主体的编码格式需要根据请求头中的content-type字段确定
  3. content-type: application/x-www-form-urlencoded,requests.post() 发送请求时使用data参数,data的值 (即请求参数) 为字典格式
  4. content-type: multipart/form-data,requests.post() 发送请求时使用files参数,files的值 (即请求参数) 即为读取的上传文件的内容
  5. content-type: application/json,requests.post() 发送请求时使用json参数,json的值 (即请求参数) 为字典格式,或者也可以使用 data 参数,但此时需要先将请求参数转换为json格式
  6. content-type: text/xml,requests.post() 发送请求时使用data参数,data的值 (即请求参数) 为 xml 中的body部分内容

具体的脚本编写方法参考示例代码。

最后感谢每一个认真阅读我文章的人,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:

这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你!

### 回答1: 在进行接口测试时,使用自动化框架可以提高工作效率并减少出错的可能性。Python中的requests库可以方便地发送HTTP请求并获取响应,而pytest则是一个功能强大的测试框架,可以支持接口测试自动化。 1. requests库是Python中的HTTP客户端库,可以方便地构建和发送HTTP请求,并解析返回的响应数据。requests库的文档详细且易于理解,可以快速掌握该库的使用方法,例如发送基本的GET、POST、PUT等HTTP请求,设置请求头、请求体、cookies等,也可以进行文件上传和下载。 2. pytest是一个基于Python的测试框架,它提供了完整的测试和夹具API,并支持并行化测试和分布式测试。pytest的测试用例可以非常灵活和高效地管理和维护,可以根据需要添加更多的测试用例,而不会破坏原有的测试逻辑。 3. 接口自动化测试主要包括如下几个步骤:构建请求参数、发送HTTP请求、解析响应数据、对返回结果进行断言。使用requests和pytest库可以方便地完成以上操作,例如使用pytest的fixture、参数化和mark机制可以让测试用例更容易编写和维护,而requests则可以方便地发送HTTP请求和获取响应。 4. 使用Python、requests以及pytest可以轻松构建接口自动化测试框架,并集成到CI/CD流程中实现持续集成和自动化部署,从而提高软件质量、减少故障率和提高开发效率。同时,Python作为一门便于阅读的语言,使得代码易于理解和维护,也更适合于多人协作开发。 ### 回答2: Python requests pytest 接口自动化框架是一种基于 Python 语言的测试框架,主要用于对接口进行自动化测试,旨在提高测试效率,降低测试成本。该框架由 Python requests 库和 pytest 测试框架组成,可以快速、方便地进行接口测试。 Python requests 库是 Python 中的一个 HTTP 库,用于发送 HTTP 请求,处理 HTTP 响应并支持 RESTful API。该库简化了 HTTP 请求的过程,提供了易于使用的方法和参数。 而 pytest 是一个功能强大的 Python 测试框架,可以自动化执行测试用例,提供了易于编写和扩展的测试样式。与传统的测试框架相比,pytest 拥有更高的执行效率和更好的测试结果报告。 在使用该框架进行接口自动化测试时,首先需要使用 pytest 编写测试用例,然后使用 Python requests 库发送 HTTP 请求,并进行断言和验证响应结果。该框架支持多种请求方法、参数、Cookie、Headers 和响应格式等,可根据需要进行自定义。 另外,该框架还支持测试用例的参数化、Fixture 和模块化等功能,可以实现重复测试用例的复用、测试环境的配置和共享等。 总之,Python requests pytest 接口自动化框架是一种功能强大、易学易用的自动化测试工具,可以有效提高测试效率和质量。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值