python urllib2 文件发送问题

3 投票
2 回答
2164 浏览
提问于 2025-04-16 11:13

我想通过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

撰写回答