python urllib2 文件发送问题
我想通过Python把一个文件上传到服务器,为了让服务器能识别这个输入,我需要把这个文件命名为“xmlfile”。
import urllib2
url = "http://somedomain"
to_send = open('test.xml').read()
data = {}
data['xmlfile'] = to_send
f = urllib2.urlopen(url, data)
这样做不行,另外,我该怎么获取服务器的响应并保存到某个地方呢?
换句话说,我想做的事情和我用Curl命令时一样:
curl.exe http://somedomain -F xmlfile=@test.xml -o response.html
2 个回答
0
你可以看看这个 StackOverflow上的问题,或者直接查看它提到的代码 链接。
通常情况下,urllib
发送数据时使用的是 application/x-www-form-urlencoded
这种内容类型,但你需要的是 multipart/form-data
。而提到的那个库会按照需要对数据进行编码。
1
我刚刚看了nimrodm提到的问题。里面有一个回答提到了poster模块。这个模块可以处理multipart/form-data
这种编码方式,所以如果你不介意在项目中添加一个新的依赖,我建议使用poster模块。
这事情没有想象中那么简单。网上有一个代码片段,我用过,效果不错。你可能需要根据自己的需求对它进行一些调整。
class RequestWithMethod(urllib2.Request):
def __init__(self, method, *args, **kwargs):
self._method = method
urllib2.Request.__init__(self, *args, **kwargs)
def get_method(self):
return self._method
class RestRequest(object):
def __init__(self, base_url):
self.base_url = base_url
def request(self, url, method, headers={"Accept" : "application/json"}, data=None, json_response=True):
request = RequestWithMethod(url='{0}{1}{2}'.format(self.base_url, root_url(), url),
method=method,
headers=headers)
if data != None:
data = urllib.urlencode(data)
response = urllib2.urlopen(request, data=data).read()
if json_response:
return from_json(response)
else:
return response
def GET(self, url, **kwargs):
return self.request(url, 'GET', **kwargs)
def POST(self, url, **kwargs):
return self.request(url, 'POST', **kwargs)
def POST_FILE(self, url, file, headers={"Accept" : "application/json"}, data={}, **kwargs):
content_type, body = encode_multipart_formdata(data, file)
headers['Content-type'] = content_type
headers['Content-length'] = str(len(body))
request = RequestWithMethod(url='{0}{1}{2}'.format(self.base_url, root_url(), url),
data=body,
method='POST',
headers=headers)
return from_json(urllib2.urlopen(request).read())
def PUT(self, url, **kwargs):
return self.request(url, 'PUT', **kwargs)
def DELETE(self, url, **kwargs):
return self.request(url, 'DELETE', **kwargs)
def encode_multipart_formdata(data, file):
boundary = '----------ThIs_Is_tHe_bouNdaRY_$'
L = []
for key, value in data.items():
L.append('--' + boundary)
L.append('Content-Disposition: form-data; name="{0}"'.format(key))
L.append('')
L.append(value)
key, filename, value = file
L.append('--' + boundary)
L.append('Content-Disposition: form-data; name="{0}"; filename="{1}"'.format(key, filename))
content_type = mimetypes.guess_type(filename)[0] or 'application/octet-stream'
L.append('Content-Type: {0}'.format(content_type))
L.append('')
L.append(value)
L.append('--' + boundary + '--')
L.append('')
body = '\r\n'.join(L)
content_type = 'multipart/form-data; boundary={0}'.format(boundary)
return content_type, body