проблема отправки файла python urllib2 - PullRequest
3 голосов
/ 04 февраля 2011

Я хочу отправить файл на сервер через 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 ]

1 голос
/ 04 февраля 2011

Я только что прочитал вопрос, на который ссылается nimrodm. В одном ответе упоминается модуль плаката. Этот модуль может выполнять кодирование 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
0 голосов
/ 04 февраля 2011

Посмотрите на этот вопрос стекопотока или непосредственно на код, на который он ссылается .

urllib обычно отправляет данные, используя application/x-www-form-urlencoded тип содержимого, и пока вынужно multipart/form-data.Указанная библиотека кодирует данные по мере необходимости.

...