Стандартная библиотека Python для POST многочастных данных/данных формы
Вопрос
Я хотел бы отправить данные POST в формате multipart/form-data.Я нашел внешний модуль, который это делает: http://atlee.ca/software/poster/index.htmlоднако я бы предпочел избежать этой зависимости.Есть ли способ сделать это с помощью стандартных библиотек?
Спасибо
Решение
Стандартная библиотека на данный момент не поддерживает это.Есть рецепт из кулинарной книги это включает в себя довольно короткий фрагмент кода, который вы, возможно, захотите скопировать, а также долгое обсуждение альтернатив.
Другие советы
Это старая тема, но она все еще популярна, поэтому здесь я предлагаю использовать только стандартные модули.
Идея та же, что и здесь но поддерживает Python 2.x и Python 3.x.Он также имеет генератор тел, позволяющий избежать ненужного использования памяти.
import codecs
import mimetypes
import sys
import uuid
try:
import io
except ImportError:
pass # io is requiered in python3 but not available in python2
class MultipartFormdataEncoder(object):
def __init__(self):
self.boundary = uuid.uuid4().hex
self.content_type = 'multipart/form-data; boundary={}'.format(self.boundary)
@classmethod
def u(cls, s):
if sys.hexversion < 0x03000000 and isinstance(s, str):
s = s.decode('utf-8')
if sys.hexversion >= 0x03000000 and isinstance(s, bytes):
s = s.decode('utf-8')
return s
def iter(self, fields, files):
"""
fields is a sequence of (name, value) elements for regular form fields.
files is a sequence of (name, filename, file-type) elements for data to be uploaded as files
Yield body's chunk as bytes
"""
encoder = codecs.getencoder('utf-8')
for (key, value) in fields:
key = self.u(key)
yield encoder('--{}\r\n'.format(self.boundary))
yield encoder(self.u('Content-Disposition: form-data; name="{}"\r\n').format(key))
yield encoder('\r\n')
if isinstance(value, int) or isinstance(value, float):
value = str(value)
yield encoder(self.u(value))
yield encoder('\r\n')
for (key, filename, fd) in files:
key = self.u(key)
filename = self.u(filename)
yield encoder('--{}\r\n'.format(self.boundary))
yield encoder(self.u('Content-Disposition: form-data; name="{}"; filename="{}"\r\n').format(key, filename))
yield encoder('Content-Type: {}\r\n'.format(mimetypes.guess_type(filename)[0] or 'application/octet-stream'))
yield encoder('\r\n')
with fd:
buff = fd.read()
yield (buff, len(buff))
yield encoder('\r\n')
yield encoder('--{}--\r\n'.format(self.boundary))
def encode(self, fields, files):
body = io.BytesIO()
for chunk, chunk_len in self.iter(fields, files):
body.write(chunk)
return self.content_type, body.getvalue()
Демо
# some utf8 key/value pairs
fields = [('প্রায়', 42), ('bar', b'23'), ('foo', 'ން:')]
files = [('myfile', 'image.jpg', open('image.jpg', 'rb'))]
# iterate and write chunk in a socket
content_type, body = MultipartFormdataEncoder().encode(fields, files)
Вы не сможете сделать это с помощью stdlib быстро.Однако см. MultiPartForm
класс в этом PyMOTW.Вероятно, вы можете использовать или изменить это для достижения всего, что вам нужно: