在没有请求的情况下将 cURL POST 转换为 urllib2
Converting cURL POST into urllib2 without Requests
我目前有一个适用于我的环境的 cURL 命令。我需要将其转换为 python 2.7 urllib 等效项并需要一些帮助
有问题的 cURL 命令:
curl -k -v -XPOST -H "Authorization: Bearer $Token" -F file=@${local_filename} https://interesting_url.com/
我需要同时支持 Header (-H) 和 form/file (-F) 的能力,但尚未成功。
This post includes a discussion on headers,但我无法让“-F”等效项工作
使用 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 和第三方包的长答案
您可以实现自定义 class 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()
尝试查看请求库。您可以使用 pip install requests
安装它,之后只需在线搜索一些示例代码。根据您的应用程序,它会有所不同。 requests 库允许你做很多事情。有很多文档。我强烈建议您检查一下。
我目前有一个适用于我的环境的 cURL 命令。我需要将其转换为 python 2.7 urllib 等效项并需要一些帮助
有问题的 cURL 命令:
curl -k -v -XPOST -H "Authorization: Bearer $Token" -F file=@${local_filename} https://interesting_url.com/
我需要同时支持 Header (-H) 和 form/file (-F) 的能力,但尚未成功。
This post includes a discussion on headers,但我无法让“-F”等效项工作
使用 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 和第三方包的长答案
您可以实现自定义 class 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()
尝试查看请求库。您可以使用 pip install requests
安装它,之后只需在线搜索一些示例代码。根据您的应用程序,它会有所不同。 requests 库允许你做很多事情。有很多文档。我强烈建议您检查一下。