利用python模拟实现POST请求提交图片的方法


Posted in Python onJuly 25, 2017

本文主要给大家介绍的是关于利用python模拟实现POST请求提交图片的方法,分享出来供大家参考学习,下面来一看看详细的介绍:

使用requests来模拟HTTP请求本来是一件非常轻松的事情,比如上传图片来说,简单的几行代码即可:

import requests
files = {'attachment_file': ('1.png', open('1.png', 'rb'), 'image/png', {})}
values = {'next':"http://www.xxxx.com/xxxx"}
r = requests.post('http://www.xxxx.com/upload', files=files, data=values) # 成功
r = requests.post('http://www.xxxx.com/upload', files=files, data=values) # 失败
r = requests.post('http://www.xxxx.com/upload', files=files, data=values) # 失败
r = requests.post('http://www.xxxx.com/upload', files=files, data=values) # 失败
r = requests.post('http://www.xxxx.com/upload', files=files, data=values) # 失败
...

不过我今天在调试一个django程序的时候却遇到了大坑————为了偷懒,我直接在ipython中执行了上述代码,第一次提交的时候一切正常,但第二次之后提交就怎么也通过不了django的form验证。

验证部分的代码很简单:

......
form = AttachmentForm(request.POST, request.FILES)
if form.is_valid():
 form.save(request, obj)
 messages.success(request,_('Your attachment was uploaded.'))
 return HttpResponseRedirect(next)
......

什么鬼!?怎么只有第一次成功提交???后面全失败??只好一步一步的跟进到django源码中,发现问题出在django/forms/fields.py文件中:

def to_python(self, data):
 if data in validators.EMPTY_VALUES:
 return None
 # UploadedFile objects should have name and size attributes.
 try:
 file_name = data.name
 file_size = data.size
 except AttributeError:
 raise ValidationError(self.error_messages['invalid'])
 if self.max_length is not None and len(file_name) > self.max_length:
 error_values = {'max': self.max_length, 'length': len(file_name)}
 raise ValidationError(self.error_messages['max_length'] % error_values)
 if not file_name:
 raise ValidationError(self.error_messages['invalid'])
 if not self.allow_empty_file and not file_size:
 raise ValidationError(self.error_messages['empty'])
 return data

在第一次执行的时候,一切正常,这个data即InMemoryUploadFile文件类型,name、size就是我们上传的图片名、大小,而第二次执行post请求时候,发现data.size居然变成了0?!怪不得直接引发了if not self.allow_empty_file and not file_size这个判断的异常呢!

由此可知,问题的核心并不出现在django对于表单验证的部分,而是出自发送请求的部分。不过发请求的部分代码很简单啊?分别输出了正常情况和错误情况requests发出的请求包,发现区别了:

#正常情况
In [28]: r = requests.post('http://www.xxxx.com/upload', files=files, data=values)
In [29]: r.request.body
#错误情况
In [33]: r = requests.post('http://www.xxxx.com/upload', files=files, data=values)
In [34]: r.request.body
Out[34]: '--155322d3e780432bb06e58135e041c8f\r\nContent-Disposition: form-data; name="next"\r\n\r\nhttp://www.xxxx.com/upload\r\n--155322d3e780432bb06e58135e041c8f\r\nContent-Disposition: form-data; name="attachment_file"; filename="1.png"\r\nContent-Type: image/png\r\n\r\n\r\n--155322d3e780432bb06e58135e041c8f--\r\n'

正常情况没输出,错误情况反而看着像正常情况下的输出?这不科学啊?

结合以上2点,我隐约感觉问题出在数据的构造上,关键在于files = {'attachment_file': ('1.png', open('1.png', 'rb') , 'image/png', {})}这里,首先关于字典、列表这种可变类型作为函数的参数传递时候就需要特别注意,其次open函数打开了一个文件,那么哪里关闭文件了呢?

带着这个怀疑,我把代码改写成:

fl = open('1.png','rb')
files = {'attachment_file': ('1.png', fl, 'image/png', {})}
r1 = requests.post('http://www.xxxx.com/upload', files=files, data=values)
fl.close()
fl = open('1.png','rb')
files = {'attachment_file': ('1.png', fl, 'image/png', {})}
r2 = requests.post('http://www.xxxx.com/upload', files=files, data=values)

然后再执行,果然成功上传了2张图片。其实按照正常情况不会出现测试时候这种打开一张图片不停上传的情形,不过也正因为这样才会遇到如此有意思的问题。关于requests中files对象的处理代码在models.py文件中,有兴趣的读者可以自行调试。

另外,requests调用时上传文件名中不能包含中文,否则也不能通过django表单验证,这里也不深究原因了。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对三水点靠木的支持。

Python 相关文章推荐
用Python制作简单的朴素基数估计器的教程
Apr 01 Python
在Debian下配置Python+Django+Nginx+uWSGI+MySQL的教程
Apr 25 Python
Python 判断 有向图 是否有环的实例讲解
Feb 01 Python
matplotlib savefig 保存图片大小的实例
May 24 Python
python实现微信机器人: 登录微信、消息接收、自动回复功能
Apr 29 Python
pandas DataFrame索引行列的实现
Jun 04 Python
在python中用print()输出多个格式化参数的方法
Jul 16 Python
Flask使用Pyecharts在单个页面展示多个图表的方法
Aug 05 Python
Python爬虫:将headers请求头字符串转为字典的方法
Aug 21 Python
python 如何利用argparse解析命令行参数
Sep 11 Python
Python使用OpenCV和K-Means聚类对毕业照进行图像分割
Jun 11 Python
详解在OpenCV中如何使用图像像素
Mar 03 Python
利用django如何解析用户上传的excel文件
Jul 24 #Python
Python编程之变量赋值操作实例分析
Jul 24 #Python
Python模块结构与布局操作方法实例分析
Jul 24 #Python
Python列表list操作符实例分析【标准类型操作符、切片、连接字符、列表解析、重复操作等】
Jul 24 #Python
Python列表list内建函数用法实例分析【insert、remove、index、pop等】
Jul 24 #Python
python引入导入自定义模块和外部文件的实例
Jul 24 #Python
Python3.X 线程中信号量的使用方法示例
Jul 24 #Python
You might like
php环境配置 php5 MySQL5 apache2 phpmyadmin安装与配置图文教程
2007/03/16 PHP
PHP容易忘记的知识点分享
2013/04/30 PHP
详解在YII2框架中使用UEditor编辑器发布文章
2018/11/02 PHP
Yii支持多域名cors原理的实现
2018/12/05 PHP
TP5框架实现自定义分页样式的方法示例
2020/04/05 PHP
Using the TextRange Object
2006/10/14 Javascript
js jquery做的图片连续滚动代码
2008/01/06 Javascript
事件模型在各浏览器中存在差异
2010/10/20 Javascript
jQuery通过点击行来删除HTML表格行的实现示例
2014/09/10 Javascript
JavaScript中exec函数用法实例分析
2015/06/08 Javascript
简介JavaScript中的unshift()方法的使用
2015/06/09 Javascript
简介JavaScript中Math.cos()余弦方法的使用
2015/06/15 Javascript
javascript实现的闭包简单实例
2015/07/17 Javascript
Nodejs的express使用教程
2015/11/23 NodeJs
微信小程序的动画效果详解
2017/01/18 Javascript
jQuery点击导航栏选中更换样式的实现代码
2017/01/23 Javascript
详解微信小程序实现WebSocket心跳重连
2018/07/31 Javascript
Layui数据表格之获取表格中所有的数据方法
2018/08/20 Javascript
基于JavaScript实现一个简单的Vue
2018/09/26 Javascript
vue项目中实现图片预览的公用组件功能
2018/10/26 Javascript
ionic使用angularjs表单验证(模板验证)
2018/12/12 Javascript
Python 类与元类的深度挖掘 I【经验】
2016/05/06 Python
Python中sort和sorted函数代码解析
2018/01/25 Python
python提取图像的名字*.jpg到txt文本的方法
2018/05/10 Python
python写入并获取剪切板内容的实例
2018/05/31 Python
Python对接支付宝支付自实现功能
2019/10/10 Python
python实现井字棋小游戏
2020/03/04 Python
基于python requests selenium爬取excel vba过程解析
2020/08/12 Python
python向xls写入数据(包括合并,边框,对齐,列宽)
2021/02/02 Python
巴基斯坦购物网站:Goto
2019/03/11 全球购物
Jones New York官网:美国女装品牌,受白领女性欢迎
2019/11/26 全球购物
美国鲜花递送:UrbanStems
2021/01/04 全球购物
自荐信结尾
2013/10/27 职场文书
大学生的网上创业计划书
2013/12/31 职场文书
领导干部遵守党的政治纪律情况思想汇报
2014/09/14 职场文书
PyMongo 查询数据的实现
2021/06/28 Python