data flattening needs to go into resource

This commit is contained in:
Tom Christie 2011-05-19 08:36:55 +01:00
parent 49d4e50342
commit 8c3280f9c0
6 changed files with 171 additions and 210 deletions

View File

@ -26,7 +26,7 @@ __all__ = (
'ResponseMixin', 'ResponseMixin',
'AuthMixin', 'AuthMixin',
'ResourceMixin', 'ResourceMixin',
# # Reverse URL lookup behavior
'InstanceMixin', 'InstanceMixin',
# Model behavior mixins # Model behavior mixins
'ReadModelMixin', 'ReadModelMixin',
@ -360,7 +360,6 @@ class AuthMixin(object):
""" """
The set of authentication types that this view can handle. The set of authentication types that this view can handle.
Should be a tuple/list of classes as described in the ``authentication`` module. Should be a tuple/list of classes as described in the ``authentication`` module.
""" """
authentication = () authentication = ()

View File

@ -63,9 +63,18 @@ class BaseParser(object):
class JSONParser(BaseParser): class JSONParser(BaseParser):
"""
JSON parser.
"""
media_type = 'application/json' media_type = 'application/json'
def parse(self, stream): def parse(self, stream):
"""
Returns a 2-tuple of `(data, files)`.
`data` will be an object which is the parsed content of the response.
`files` will always be `None`.
"""
try: try:
return (json.load(stream), None) return (json.load(stream), None)
except ValueError, exc: except ValueError, exc:
@ -73,103 +82,55 @@ class JSONParser(BaseParser):
{'detail': 'JSON parse error - %s' % unicode(exc)}) {'detail': 'JSON parse error - %s' % unicode(exc)})
class DataFlatener(object):
"""Utility object for flattening dictionaries of lists. Useful for "urlencoded" decoded data."""
def flatten_data(self, data):
"""Given a data dictionary {<key>: <value_list>}, returns a flattened dictionary
with information provided by the method "is_a_list"."""
flatdata = dict()
for key, val_list in data.items():
if self.is_a_list(key, val_list):
flatdata[key] = val_list
else:
if val_list:
flatdata[key] = val_list[0]
else:
# If the list is empty, but the parameter is not a list,
# we strip this parameter.
data.pop(key)
return flatdata
def is_a_list(self, key, val_list):
"""Returns True if the parameter with name *key* is expected to be a list, or False otherwise.
*val_list* which is the received value for parameter *key* can be used to guess the answer."""
return False
class PlainTextParser(BaseParser): class PlainTextParser(BaseParser):
""" """
Plain text parser. Plain text parser.
Simply returns the content of the stream.
""" """
media_type = 'text/plain' media_type = 'text/plain'
def parse(self, stream): def parse(self, stream):
"""
Returns a 2-tuple of `(data, files)`.
`data` will simply be a string representing the body of the request.
`files` will always be `None`.
"""
return (stream.read(), None) return (stream.read(), None)
class FormParser(BaseParser, DataFlatener): class FormParser(BaseParser):
"""
Parser for form data.
""" """
The default parser for form data.
Return a dict containing a single value for each non-reserved parameter.
In order to handle select multiple (and having possibly more than a single value for each parameter),
you can customize the output by subclassing the method 'is_a_list'."""
media_type = 'application/x-www-form-urlencoded' media_type = 'application/x-www-form-urlencoded'
"""The value of the parameter when the select multiple is empty.
Browsers are usually stripping the select multiple that have no option selected from the parameters sent.
A common hack to avoid this is to send the parameter with a value specifying that the list is empty.
This value will always be stripped before the data is returned.
"""
EMPTY_VALUE = '_empty'
RESERVED_FORM_PARAMS = ('csrfmiddlewaretoken',)
def parse(self, stream): def parse(self, stream):
"""
Returns a 2-tuple of `(data, files)`.
`data` will be a `QueryDict` containing all the form parameters.
`files` will always be `None`.
"""
data = parse_qs(stream.read(), keep_blank_values=True) data = parse_qs(stream.read(), keep_blank_values=True)
# removing EMPTY_VALUEs from the lists and flatening the data
for key, val_list in data.items():
self.remove_empty_val(val_list)
data = self.flatten_data(data)
# Strip any parameters that we are treating as reserved
for key in data.keys():
if key in self.RESERVED_FORM_PARAMS:
data.pop(key)
return (data, None) return (data, None)
def remove_empty_val(self, val_list):
""" """
while(1): # Because there might be several times EMPTY_VALUE in the list
try:
ind = val_list.index(self.EMPTY_VALUE)
except ValueError:
break
else:
val_list.pop(ind)
class MultiPartParser(BaseParser):
"""
Parser for multipart form data, which may include file data.
"""
class MultiPartParser(BaseParser, DataFlatener):
media_type = 'multipart/form-data' media_type = 'multipart/form-data'
RESERVED_FORM_PARAMS = ('csrfmiddlewaretoken',)
def parse(self, stream): def parse(self, stream):
"""
Returns a 2-tuple of `(data, files)`.
`data` will be a `QueryDict` containing all the form parameters.
`files` will be a `QueryDict` containing all the form files.
"""
upload_handlers = self.view.request._get_upload_handlers() upload_handlers = self.view.request._get_upload_handlers()
django_parser = DjangoMultiPartParser(self.view.request.META, stream, upload_handlers) django_parser = DjangoMultiPartParser(self.view.request.META, stream, upload_handlers)
data, files = django_parser.parse() return django_parser.parse()
# Flatening data, files and combining them
data = self.flatten_data(dict(data.iterlists()))
files = self.flatten_data(dict(files.iterlists()))
# Strip any parameters that we are treating as reserved
for key in data.keys():
if key in self.RESERVED_FORM_PARAMS:
data.pop(key)
return (data, files)

View File

@ -59,6 +59,7 @@ class IsAuthenticated(BasePermission):
if not user.is_authenticated(): if not user.is_authenticated():
raise _403_FORBIDDEN_RESPONSE raise _403_FORBIDDEN_RESPONSE
class IsAdminUser(): class IsAdminUser():
""" """
Allows access only to admin users. Allows access only to admin users.

View File

@ -36,7 +36,7 @@ class TestContentParsing(TestCase):
form_data = {'qwerty': 'uiop'} form_data = {'qwerty': 'uiop'}
view.parsers = (FormParser, MultiPartParser) view.parsers = (FormParser, MultiPartParser)
view.request = self.req.put('/', data=form_data) view.request = self.req.put('/', data=form_data)
self.assertEqual(view.DATA, form_data) self.assertEqual(view.DATA.items(), form_data.items())
def ensure_determines_non_form_content_PUT(self, view): def ensure_determines_non_form_content_PUT(self, view):
"""Ensure view.RAW_CONTENT returns content for PUT request with non-form content.""" """Ensure view.RAW_CONTENT returns content for PUT request with non-form content."""

View File

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

View File

@ -8,7 +8,7 @@ class MyModelResource(ModelResource):
fields = ('foo', 'bar', 'baz', 'url') fields = ('foo', 'bar', 'baz', 'url')
ordering = ('created',) ordering = ('created',)
urlpatterns = patterns('modelresourceexample.views', urlpatterns = patterns('',
url(r'^$', ListOrCreateModelView.as_view(resource=MyModelResource), name='model-resource-root'), url(r'^$', ListOrCreateModelView.as_view(resource=MyModelResource), name='model-resource-root'),
url(r'^([0-9]+)/$', InstanceModelView.as_view(resource=MyModelResource)), url(r'^([0-9]+)/$', InstanceModelView.as_view(resource=MyModelResource)),
) )