在没有请求的情况下将cURL POST转换为urllib2

时间:2017-09-13 20:29:28

标签: python curl urllib2

我目前有一个在我的环境中工作的cURL命令。我需要将它转换为python 2.7 urllib等价物并需要一些帮助

有问题的cURL命令:

curl -k -v -XPOST -H "Authorization: Bearer $Token" -F file=@${local_filename} https://interesting_url.com/

我需要能够同时支持Header(-H)和表单/文件(-F)并且还没有成功。

This post includes a discussion on headers,但我无法让'-F'等效工作

2 个答案:

答案 0 :(得分:1)

使用python3,您可以使用requests执行此操作:

import requests

headers = {'Authorization': 'Bearer $Token'}

files = [('file', open('${local_filename}', 'rb'))]

requests.post('https://interesting_url.com/', headers=headers, files=files, verify=False)

使用urllib2上传文件是一项非常复杂的任务(example)。 所以我建议你requests

不使用pip和第三方软件包的长答案

您可以实现自定义类MultiPartForm,然后使用它来编码文件:

import itertools
import mimetools
import mimetypes
from cStringIO import StringIO
import urllib
import urllib2

class MultiPartForm(object):
    """Accumulate the data to be used when posting a form."""

    def __init__(self):
        self.form_fields = []
        self.files = []
        self.boundary = mimetools.choose_boundary()
        return

    def get_content_type(self):
        return 'multipart/form-data; boundary=%s' % self.boundary

    def add_field(self, name, value):
        """Add a simple field to the form data."""
        self.form_fields.append((name, value))
        return

    def add_file(self, fieldname, filename, fileHandle, mimetype=None):
        """Add a file to be uploaded."""
        body = fileHandle.read()
        if mimetype is None:
            mimetype = mimetypes.guess_type(filename)[0] or 'application/octet-stream'
        self.files.append((fieldname, filename, mimetype, body))
        return

    def __str__(self):
        """Return a string representing the form data, including attached files."""
        # Build a list of lists, each containing "lines" of the
        # request.  Each part is separated by a boundary string.
        # Once the list is built, return a string where each
        # line is separated by '\r\n'.  
        parts = []
        part_boundary = '--' + self.boundary

        # Add the form fields
        parts.extend(
            [ part_boundary,
              'Content-Disposition: form-data; name="%s"' % name,
              '',
              value,
            ]
            for name, value in self.form_fields
            )

        # Add the files to upload
        parts.extend(
            [ part_boundary,
              'Content-Disposition: file; name="%s"; filename="%s"' % \
                 (field_name, filename),
              'Content-Type: %s' % content_type,
              '',
              body,
            ]
            for field_name, filename, content_type, body in self.files
            )

        # Flatten the list and add closing boundary marker,
        # then return CR+LF separated data
        flattened = list(itertools.chain(*parts))
        flattened.append('--' + self.boundary + '--')
        flattened.append('')
        return '\r\n'.join(flattened)

with open(`local_file.txt`) as f:
    form = MultiPartForm()
    form.add_file('file', `local_file`, 
                  fileHandle=f)

    # Build the request
    request = urllib2.Request('https://interesting_url.com/')
    request.add_header('Authorization', 'Bearer $Token')
    body = str(form)
    request.add_header('Content-type', form.get_content_type())
    request.add_header('Content-length', len(body))
    request.add_data(body)

    print
    print 'OUTGOING DATA:'
    print request.get_data()

    print
    print 'SERVER RESPONSE:'
    print urllib2.urlopen(request).read()

The source

答案 1 :(得分:0)

尝试查看请求库。您可以使用pip install requests安装它,之后只需在线搜索一些示例代码即可。根据您的应用程序,它会有所不同。请求库允许您执行很多操作。有很多文档。我强烈建议您查看。