Python接口测试- requests 文件上传

前面我们通过 post 请求发送的是文本内容,也就是 ASCII 字符。如果需要发送文件到服务器,比如上传图片、视频等,就需要发送二进制数据。

一般上传文件使用的都是Content-Type: multipart/form-data;数据类型,可以发送文件,也可以发送相关的消息体数据。

使用 requests 上传文件的基本步骤:

  1. 构造文件数据,通过 open 函数以二进制方式打开文件
  2. 构造相关数据
  3. 发送请求,将文件数据以 files 参数传入,其他消息体数据通过 data或 json 传入

requests 官方文档上给出的示例如下:

>>> url = 'http://httpbin.org/post'
>>> files = {'file': open('report.xls', 'rb')}  # => 直接通过open函数打开文件并将文件对象存在字典中

>>> r = requests.post(url, files=files)
>>> r.text
{
  ...
  "files": {
    "file": ""
  },
  ...
}

# 你可以显式地设置文件名,文件类型和请求头:
>>> url = 'http://httpbin.org/post'
>>> files = {'file': ('report.xls', open('report.xls', 'rb'), 'application/vnd.ms-excel', {'Expires': '0'})} # => 打开上传文件并且加入文件相关参数

>>> r = requests.post(url, files=files)
>>> r.text
{
  ...
  "files": {
    "file": ""
  },
  ...
}

我想大家虽然能看懂,但是依然会略显懵逼。这里我还是以 showdoc 中的文件上传为例为大家演示一下:

showdoc 中的文件上传位于接口文档的编辑页面。



我们任意选择一张图片上传,并抓包查看:



在原始中看到了相关的内容,包括 Content-Type,图片信息等(切换到 Raw 的时候会比较卡)。Raw 中始终是查看请求内容最全面的地方,只是这里会显示二进制的图片内容,会比较卡。所以第一次我们看一下 Raw 即可。以后直接在 WebForms 中查看,需要的数据在那里都可以看到。

注意其中的 name="editormd-image-file",需要用此 name 为服务器指定文件对象。

好,接下来我们参考官方文档上的方式模拟:

from urllib import parse
from pprint import pprint
import requests

host = 'http://127.0.0.1'
user = {
    'username': 'showdoc',
    'password':'123456'
}

s = requests.Session() # => 会话对象

# 登录
login_url = parse.urljoin(host, '/showdoc/server/index.php?s=/api/user/login')
lr = s.post(login_url, data=user)
pprint(lr.json())

# 上传图片
upload_url = parse.urljoin(host,'http://127.0.0.1/showdoc/server/index.php?s=/api/page/uploadImg')
# 构造图片数据,这里必须要填上图片相关参数
file = {
    'editormd-image-file': open(r'D:\data\1.png', 'rb'),   # => 用name指定文件
    'Content-Disposition': 'form-data', 
    'Content-Type': 'image/png', 
    'filename':'1.png'
    }
ur = s.post(upload_url, files=file)  # => 注意这里,参数名是 files
pprint(ur.json())

如果你遇到其他情况需要上传文件的同时传输其他表单数据,直接和之前的 post 请求一样,通过字典构造数据并通过 data 参数传递即可。

>>> r = s.post(upload_url, files=file, data=data)  # => 包括url参数,headers,cookies都可以一起传递

警告
我们强烈建议你用二进制模式(binary mode)打开文件。这是因为 Requests 可能会试图为你提供 Content-Length header,在它这样做的时候,这个值会被设为文件的字节数(bytes)。如果用文本模式(text mode)打开文件,就可能会发生错误。

你可能感兴趣的:(Python接口测试- requests 文件上传)