2011-03-11 15:34:39 +03:00
|
|
|
"""
|
|
|
|
..
|
|
|
|
>>> from djangorestframework.parsers import FormParser
|
|
|
|
>>> from djangorestframework.compat import RequestFactory
|
2011-05-04 12:21:17 +04:00
|
|
|
>>> from djangorestframework.views import BaseView
|
2011-04-02 19:32:37 +04:00
|
|
|
>>> from StringIO import StringIO
|
2011-03-11 15:34:39 +03:00
|
|
|
>>> from urllib import urlencode
|
|
|
|
>>> req = RequestFactory().get('/')
|
2011-05-04 12:21:17 +04:00
|
|
|
>>> some_view = BaseView()
|
|
|
|
>>> some_view.request = req # Make as if this request had been dispatched
|
2011-03-11 15:34:39 +03:00
|
|
|
|
2011-03-11 16:05:35 +03:00
|
|
|
FormParser
|
|
|
|
============
|
|
|
|
|
2011-03-11 15:34:39 +03:00
|
|
|
Data flatening
|
|
|
|
----------------
|
|
|
|
|
|
|
|
Here is some example data, which would eventually be sent along with a post request :
|
|
|
|
|
|
|
|
>>> inpt = urlencode([
|
|
|
|
... ('key1', 'bla1'),
|
|
|
|
... ('key2', 'blo1'), ('key2', 'blo2'),
|
|
|
|
... ])
|
|
|
|
|
|
|
|
Default behaviour for :class:`parsers.FormParser`, is to return a single value for each parameter :
|
|
|
|
|
2011-05-04 12:21:17 +04:00
|
|
|
>>> FormParser(some_view).parse(StringIO(inpt)) == {'key1': 'bla1', 'key2': 'blo1'}
|
2011-03-11 15:34:39 +03:00
|
|
|
True
|
|
|
|
|
|
|
|
However, you can customize this behaviour by subclassing :class:`parsers.FormParser`, and overriding :meth:`parsers.FormParser.is_a_list` :
|
|
|
|
|
|
|
|
>>> class MyFormParser(FormParser):
|
|
|
|
...
|
|
|
|
... def is_a_list(self, key, val_list):
|
|
|
|
... return len(val_list) > 1
|
|
|
|
|
|
|
|
This new parser only flattens the lists of parameters that contain a single value.
|
|
|
|
|
2011-05-04 12:21:17 +04:00
|
|
|
>>> MyFormParser(some_view).parse(StringIO(inpt)) == {'key1': 'bla1', 'key2': ['blo1', 'blo2']}
|
2011-03-11 15:34:39 +03:00
|
|
|
True
|
|
|
|
|
2011-05-10 13:49:28 +04:00
|
|
|
.. note:: The same functionality is available for :class:`parsers.MultiPartParser`.
|
2011-03-11 16:05:35 +03:00
|
|
|
|
2011-03-11 15:34:39 +03:00
|
|
|
Submitting an empty list
|
|
|
|
--------------------------
|
|
|
|
|
|
|
|
When submitting an empty select multiple, like this one ::
|
|
|
|
|
|
|
|
<select multiple="multiple" name="key2"></select>
|
|
|
|
|
|
|
|
The browsers usually strip the parameter completely. A hack to avoid this, and therefore being able to submit an empty select multiple, is to submit a value that tells the server that the list is empty ::
|
|
|
|
|
|
|
|
<select multiple="multiple" name="key2"><option value="_empty"></select>
|
|
|
|
|
|
|
|
:class:`parsers.FormParser` provides the server-side implementation for this hack. Considering the following posted data :
|
|
|
|
|
|
|
|
>>> inpt = urlencode([
|
|
|
|
... ('key1', 'blo1'), ('key1', '_empty'),
|
|
|
|
... ('key2', '_empty'),
|
|
|
|
... ])
|
|
|
|
|
|
|
|
:class:`parsers.FormParser` strips the values ``_empty`` from all the lists.
|
|
|
|
|
2011-05-04 12:21:17 +04:00
|
|
|
>>> MyFormParser(some_view).parse(StringIO(inpt)) == {'key1': 'blo1'}
|
2011-03-11 15:34:39 +03:00
|
|
|
True
|
|
|
|
|
|
|
|
Oh ... but wait a second, the parameter ``key2`` isn't even supposed to be a list, so the parser just stripped it.
|
|
|
|
|
|
|
|
>>> class MyFormParser(FormParser):
|
|
|
|
...
|
|
|
|
... def is_a_list(self, key, val_list):
|
|
|
|
... return key == 'key2'
|
|
|
|
...
|
2011-05-04 12:21:17 +04:00
|
|
|
>>> MyFormParser(some_view).parse(StringIO(inpt)) == {'key1': 'blo1', 'key2': []}
|
2011-03-11 15:34:39 +03:00
|
|
|
True
|
|
|
|
|
2011-03-11 16:05:35 +03:00
|
|
|
Better like that. Note that you can configure something else than ``_empty`` for the empty value by setting :attr:`parsers.FormParser.EMPTY_VALUE`.
|
2011-03-11 15:34:39 +03:00
|
|
|
"""
|
2011-03-11 16:05:35 +03:00
|
|
|
import httplib, mimetypes
|
|
|
|
from tempfile import TemporaryFile
|
|
|
|
from django.test import TestCase
|
|
|
|
from djangorestframework.compat import RequestFactory
|
2011-05-10 13:49:28 +04:00
|
|
|
from djangorestframework.parsers import MultiPartParser
|
2011-05-04 12:21:17 +04:00
|
|
|
from djangorestframework.views import BaseView
|
2011-04-02 19:32:37 +04:00
|
|
|
from StringIO import StringIO
|
2011-03-11 16:05:35 +03:00
|
|
|
|
|
|
|
def encode_multipart_formdata(fields, files):
|
|
|
|
"""For testing multipart parser.
|
|
|
|
fields is a sequence of (name, value) elements for regular form fields.
|
|
|
|
files is a sequence of (name, filename, value) elements for data to be uploaded as files
|
|
|
|
Return (content_type, body)."""
|
|
|
|
BOUNDARY = '----------ThIs_Is_tHe_bouNdaRY_$'
|
|
|
|
CRLF = '\r\n'
|
|
|
|
L = []
|
|
|
|
for (key, value) in fields:
|
|
|
|
L.append('--' + BOUNDARY)
|
|
|
|
L.append('Content-Disposition: form-data; name="%s"' % key)
|
|
|
|
L.append('')
|
|
|
|
L.append(value)
|
|
|
|
for (key, filename, value) in files:
|
|
|
|
L.append('--' + BOUNDARY)
|
|
|
|
L.append('Content-Disposition: form-data; name="%s"; filename="%s"' % (key, filename))
|
|
|
|
L.append('Content-Type: %s' % get_content_type(filename))
|
|
|
|
L.append('')
|
|
|
|
L.append(value)
|
|
|
|
L.append('--' + BOUNDARY + '--')
|
|
|
|
L.append('')
|
|
|
|
body = CRLF.join(L)
|
|
|
|
content_type = 'multipart/form-data; boundary=%s' % BOUNDARY
|
|
|
|
return content_type, body
|
|
|
|
|
|
|
|
def get_content_type(filename):
|
|
|
|
return mimetypes.guess_type(filename)[0] or 'application/octet-stream'
|
|
|
|
|
2011-05-10 13:49:28 +04:00
|
|
|
class TestMultiPartParser(TestCase):
|
2011-03-11 16:05:35 +03:00
|
|
|
def setUp(self):
|
|
|
|
self.req = RequestFactory()
|
|
|
|
self.content_type, self.body = encode_multipart_formdata([('key1', 'val1'), ('key1', 'val2')],
|
|
|
|
[('file1', 'pic.jpg', 'blablabla'), ('file1', 't.txt', 'blobloblo')])
|
|
|
|
|
|
|
|
def test_multipartparser(self):
|
2011-05-10 13:49:28 +04:00
|
|
|
"""Ensure that MultiPartParser can parse multipart/form-data that contains a mix of several files and parameters."""
|
2011-03-11 16:05:35 +03:00
|
|
|
post_req = RequestFactory().post('/', self.body, content_type=self.content_type)
|
2011-05-04 12:21:17 +04:00
|
|
|
view = BaseView()
|
|
|
|
view.request = post_req
|
2011-05-10 13:49:28 +04:00
|
|
|
parsed = MultiPartParser(view).parse(StringIO(self.body))
|
2011-03-11 16:05:35 +03:00
|
|
|
self.assertEqual(parsed['key1'], 'val1')
|
2011-04-02 19:32:37 +04:00
|
|
|
self.assertEqual(parsed.FILES['file1'].read(), 'blablabla')
|
2011-03-11 16:05:35 +03:00
|
|
|
|