Estoy tratando de escribir algún código python que pueda crear solicitudes http mime de varias partes en el cliente, y luego interpretarlas adecuadamente en el servidor. Tengo, creo que, en parte logrado en el cliente final con esto:Crear y analizar solicitudes HTTP de múltiples partes en Python
from email.mime.multipart import MIMEMultipart, MIMEBase
import httplib
h1 = httplib.HTTPConnection('localhost:8080')
msg = MIMEMultipart()
fp = open('myfile.zip', 'rb')
base = MIMEBase("application", "octet-stream")
base.set_payload(fp.read())
msg.attach(base)
h1.request("POST", "http://localhost:8080/server", msg.as_string())
El único problema con esto es que la biblioteca de correo electrónico también incluye el tipo de contenido y las cabeceras MIME-Version, y no estoy seguro cómo van a estar relacionados con las cabeceras HTTP incluidos por httplib:
Content-Type: multipart/mixed; boundary="===============2050792481=="
MIME-Version: 1.0
--===============2050792481==
Content-Type: application/octet-stream
MIME-Version: 1.0
ésta puede ser la razón por la que cuando esta solicitud es recibida por mi solicitud web.py, acabo de recibir un mensaje de error. El controlador de la POST web.py:
class MultipartServer:
def POST(self, collection):
print web.input()
lanza este error:
Traceback (most recent call last):
File "/usr/local/lib/python2.6/dist-packages/web.py-0.34-py2.6.egg/web/application.py", line 242, in process
return self.handle()
File "/usr/local/lib/python2.6/dist-packages/web.py-0.34-py2.6.egg/web/application.py", line 233, in handle
return self._delegate(fn, self.fvars, args)
File "/usr/local/lib/python2.6/dist-packages/web.py-0.34-py2.6.egg/web/application.py", line 415, in _delegate
return handle_class(cls)
File "/usr/local/lib/python2.6/dist-packages/web.py-0.34-py2.6.egg/web/application.py", line 390, in handle_class
return tocall(*args)
File "/home/richard/Development/server/webservice.py", line 31, in POST
print web.input()
File "/usr/local/lib/python2.6/dist-packages/web.py-0.34-py2.6.egg/web/webapi.py", line 279, in input
return storify(out, *requireds, **defaults)
File "/usr/local/lib/python2.6/dist-packages/web.py-0.34-py2.6.egg/web/utils.py", line 150, in storify
value = getvalue(value)
File "/usr/local/lib/python2.6/dist-packages/web.py-0.34-py2.6.egg/web/utils.py", line 139, in getvalue
return unicodify(x)
File "/usr/local/lib/python2.6/dist-packages/web.py-0.34-py2.6.egg/web/utils.py", line 130, in unicodify
if _unicode and isinstance(s, str): return safeunicode(s)
File "/usr/local/lib/python2.6/dist-packages/web.py-0.34-py2.6.egg/web/utils.py", line 326, in safeunicode
return obj.decode(encoding)
File "/usr/lib/python2.6/encodings/utf_8.py", line 16, in decode
return codecs.utf_8_decode(input, errors, True)
UnicodeDecodeError: 'utf8' codec can't decode bytes in position 137-138: invalid data
Mi línea de código está representado por la línea de error a mitad de camino:
File "/home/richard/Development/server/webservice.py", line 31, in POST
print web.input()
Está quedando , pero no estoy seguro de a dónde ir desde aquí. ¿Es esto un problema con mi código de cliente o una limitación de web.py (quizás simplemente no puede admitir solicitudes de varias partes)? Cualquier sugerencia o sugerencia de bibliotecas de códigos alternativos sería gratamente recibida.
EDITAR
El error anterior fue causado por los datos no siendo automáticamente Base64 codificados. Agregar
encoders.encode_base64(base)
Deshace este error y ahora el problema es claro. solicitud HTTP no se interpreta correctamente en el servidor, presumiblemente debido a la biblioteca de correo electrónico está incluyendo cuáles deben ser las cabeceras HTTP en el cuerpo en su lugar:
<Storage {'Content-Type: multipart/mixed': u'',
' boundary': u'"===============1342637378=="\n'
'MIME-Version: 1.0\n\n--===============1342637378==\n'
'Content-Type: application/octet-stream\n'
'MIME-Version: 1.0\n'
'Content-Transfer-Encoding: base64\n'
'\n0fINCs PBk1jAAAAAAAAA.... etc
Así que algo no está bien allí.
Gracias
Richard
¿Qué diablos es una solicitud http multiparte? ¿Este concepto realmente se usa? – SingleNegationElimination
@TokenMacGuy - sí. sí lo es. –