利用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 相关文章推荐
一则python3的简单爬虫代码
May 26 Python
Python基础语法(Python基础知识点)
Feb 28 Python
Linux下通过python访问MySQL、Oracle、SQL Server数据库的方法
Apr 23 Python
浅析Python中的for 循环
Jun 09 Python
Python中用字符串调用函数或方法示例代码
Aug 04 Python
对pandas中to_dict的用法详解
Jun 05 Python
python判断字符串或者集合是否为空的实例
Jan 23 Python
python3+PyQt5 数据库编程--增删改实例
Jun 17 Python
Python matplotlib生成图片背景透明的示例代码
Aug 30 Python
Python Django中间件,中间件函数,全局异常处理操作示例
Nov 08 Python
Pytorch数据拼接与拆分操作实现图解
Apr 30 Python
Django解决frame拒绝问题的方法
Dec 18 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 表单提交给自己
2008/07/24 PHP
php实现基于微信公众平台开发SDK(demo)扩展的方法
2014/12/22 PHP
PHP从FLV文件获取视频预览图的方法
2015/03/12 PHP
php简单日历函数
2015/10/28 PHP
ajax上传时参数提交不更新等相关问题
2012/12/11 Javascript
DWR实现模拟Google搜索效果实现原理及代码
2013/01/30 Javascript
一张表格告诉你windows.onload()与$(document).ready()的区别
2014/05/16 Javascript
详谈jQuery中的this和$(this)
2014/11/13 Javascript
js实现点击链接后窗口缩小并居中的方法
2015/03/02 Javascript
javascript实现可全选、反选及删除表格的方法
2015/05/15 Javascript
简单谈谈Javascript中类型的判断
2015/10/19 Javascript
JSON遍历方式实例总结
2015/12/07 Javascript
JSON简介以及用法汇总
2016/02/21 Javascript
第五章之BootStrap 栅格系统
2016/04/25 Javascript
bootstrap常用组件之头部导航实现代码
2017/04/20 Javascript
webpack组织模块打包Library的原理及实现
2018/03/10 Javascript
vue 过滤器filter实例详解
2018/03/14 Javascript
Vue实现仿iPhone悬浮球的示例代码
2020/03/13 Javascript
微信小程序整个页面的自动适应布局的实现
2020/07/12 Javascript
python实现倒计时的示例
2014/02/14 Python
用python 制作图片转pdf工具
2015/01/30 Python
python3序列化与反序列化用法实例
2015/05/26 Python
python itchat实现微信自动回复的示例代码
2017/08/14 Python
python实现随机梯度下降(SGD)
2020/03/24 Python
Django 多语言教程的实现(i18n)
2018/07/07 Python
python分块读取大数据,避免内存不足的方法
2018/12/10 Python
Django中Middleware中的函数详解
2019/07/18 Python
python tkinter组件摆放方式详解
2019/09/16 Python
python为什么要安装到c盘
2020/07/20 Python
CSS3实现粒子旋转伸缩加载动画
2016/04/22 HTML / CSS
浅谈three.js中的needsUpdate的应用
2012/11/12 HTML / CSS
英国领先的在线礼品店:Getting Personal
2019/09/24 全球购物
运动会入场词50字
2014/02/20 职场文书
个人担保书范文
2014/05/20 职场文书
应届生求职信范文
2014/06/30 职场文书
创业计划书之校园跑腿公司
2019/09/24 职场文书