django-rest-framework/djangorestframework/mixins.py

631 lines
21 KiB
Python
Raw Normal View History

"""
2011-12-09 17:37:53 +04:00
The :mod:`mixins` module provides a set of reusable `mixin`
classes that can be added to a `View`.
"""
2011-04-11 14:47:22 +04:00
2011-05-10 13:49:28 +04:00
from django.contrib.auth.models import AnonymousUser
2011-12-09 17:37:53 +04:00
from django.core.paginator import Paginator
from django.db.models.fields.related import ForeignKey
2011-04-11 19:54:02 +04:00
from django.http import HttpResponse
from urlobject import URLObject
2011-05-02 22:49:12 +04:00
2011-05-10 13:49:28 +04:00
from djangorestframework import status
from djangorestframework.renderers import BaseRenderer
from djangorestframework.resources import Resource, FormResource, ModelResource
2011-05-10 13:49:28 +04:00
from djangorestframework.response import Response, ErrorResponse
from djangorestframework.request import request_class_factory
2011-05-10 13:49:28 +04:00
from djangorestframework.utils import as_tuple, MSIE_USER_AGENT_REGEX
from djangorestframework.utils.mediatypes import is_form_media_type, order_by_precedence
2011-05-10 13:49:28 +04:00
2011-04-11 19:54:02 +04:00
2011-05-10 15:21:48 +04:00
__all__ = (
# Base behavior mixins
2011-05-10 15:21:48 +04:00
'RequestMixin',
'ResponseMixin',
'AuthMixin',
'ResourceMixin',
# Reverse URL lookup behavior
'InstanceMixin',
# Model behavior mixins
2011-05-10 15:21:48 +04:00
'ReadModelMixin',
'CreateModelMixin',
'UpdateModelMixin',
'DeleteModelMixin',
'ListModelMixin'
)
2011-05-10 13:49:28 +04:00
2011-04-11 19:54:02 +04:00
########## Request Mixin ##########
class RequestMixin(object):
2011-05-10 13:49:28 +04:00
"""
2012-01-24 23:21:10 +04:00
`Mixin` class to enhance API of Django's standard `request`.
2011-05-10 13:49:28 +04:00
"""
_USE_FORM_OVERLOADING = True
_METHOD_PARAM = '_method'
_CONTENTTYPE_PARAM = '_content_type'
_CONTENT_PARAM = '_content'
2012-01-19 07:59:30 +04:00
parsers = ()
2011-05-13 12:59:36 +04:00
"""
2012-01-24 23:21:10 +04:00
The set of parsers that the request can handle.
2011-12-09 17:37:53 +04:00
Should be a tuple/list of classes as described in the :mod:`parsers` module.
2011-05-13 12:59:36 +04:00
"""
2011-04-11 14:47:22 +04:00
def get_request_class(self):
2011-05-12 18:11:14 +04:00
"""
2012-01-24 23:21:10 +04:00
Returns a subclass of Django's `HttpRequest` with a richer API,
as described in :mod:`request`.
2011-05-12 18:11:14 +04:00
"""
if not hasattr(self, '_request_class'):
self._request_class = request_class_factory(self.request)
self._request_class._USE_FORM_OVERLOADING = self._USE_FORM_OVERLOADING
self._request_class._METHOD_PARAM = self._METHOD_PARAM
self._request_class._CONTENTTYPE_PARAM = self._CONTENTTYPE_PARAM
self._request_class._CONTENT_PARAM = self._CONTENT_PARAM
self._request_class.parsers = self.parsers
return self._request_class
2011-05-12 18:11:14 +04:00
def get_request(self):
2011-05-12 18:11:14 +04:00
"""
Returns a custom request instance, with data and attributes copied from the
original request.
2011-05-12 18:14:22 +04:00
"""
request_class = self.get_request_class()
return request_class(self.request)
2011-04-11 19:54:02 +04:00
########## ResponseMixin ##########
class ResponseMixin(object):
2011-05-10 13:49:28 +04:00
"""
Adds behavior for pluggable `Renderers` to a :class:`views.View` class.
2011-12-09 17:37:53 +04:00
2011-05-10 15:51:49 +04:00
Default behavior is to use standard HTTP Accept header content negotiation.
Also supports overriding the content type by specifying an ``_accept=`` parameter in the URL.
2011-05-10 13:49:28 +04:00
Ignores Accept headers from Internet Explorer user agents and uses a sensible browser Accept header instead.
"""
2011-05-12 18:11:14 +04:00
_ACCEPT_QUERY_PARAM = '_accept' # Allow override of Accept header in URL query params
_IGNORE_IE_ACCEPT_HEADER = True
2011-04-11 19:54:02 +04:00
2012-01-19 07:59:30 +04:00
renderers = ()
2011-05-13 12:59:36 +04:00
"""
The set of response renderers that the view can handle.
2011-12-09 17:37:53 +04:00
Should be a tuple/list of classes as described in the :mod:`renderers` module.
2011-05-13 12:59:36 +04:00
"""
2011-05-10 15:59:13 +04:00
# TODO: wrap this behavior around dispatch(), ensuring it works
# out of the box with existing Django classes that use render_to_response.
2011-04-28 22:54:30 +04:00
def render(self, response):
2011-05-10 13:49:28 +04:00
"""
Takes a :obj:`Response` object and returns an :obj:`HttpResponse`.
2011-05-10 13:49:28 +04:00
"""
2011-04-11 19:54:02 +04:00
self.response = response
try:
renderer, media_type = self._determine_renderer(self.request)
2011-04-11 19:54:02 +04:00
except ErrorResponse, exc:
renderer = self._default_renderer(self)
media_type = renderer.media_type
2011-04-11 19:54:02 +04:00
response = exc.response
# Set the media type of the response
# Note that the renderer *could* override it in .render() if required.
response.media_type = renderer.media_type
2011-12-09 17:37:53 +04:00
2011-04-11 19:54:02 +04:00
# Serialize the response content
if response.has_content_body:
content = renderer.render(response.cleaned_content, media_type)
2011-04-11 19:54:02 +04:00
else:
content = renderer.render()
2011-04-11 19:54:02 +04:00
# Build the HTTP Response
resp = HttpResponse(content, mimetype=response.media_type, status=response.status)
2011-04-11 19:54:02 +04:00
for (key, val) in response.headers.items():
resp[key] = val
return resp
2011-04-28 22:54:30 +04:00
def _determine_renderer(self, request):
2011-05-10 15:51:49 +04:00
"""
Determines the appropriate renderer for the output, given the client's 'Accept' header,
and the :attr:`renderers` set on this class.
Returns a 2-tuple of `(renderer, media_type)`
2011-05-10 15:51:49 +04:00
See: RFC 2616, Section 14 - http://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html
"""
2011-04-11 19:54:02 +04:00
2011-05-12 18:11:14 +04:00
if self._ACCEPT_QUERY_PARAM and request.GET.get(self._ACCEPT_QUERY_PARAM, None):
2011-04-11 19:54:02 +04:00
# Use _accept parameter override
2011-05-12 18:11:14 +04:00
accept_list = [request.GET.get(self._ACCEPT_QUERY_PARAM)]
elif (self._IGNORE_IE_ACCEPT_HEADER and
2012-01-11 20:43:04 +04:00
'HTTP_USER_AGENT' in request.META and
2011-04-11 19:54:02 +04:00
MSIE_USER_AGENT_REGEX.match(request.META['HTTP_USER_AGENT'])):
# Ignore MSIE's broken accept behavior and do something sensible instead
2011-04-11 19:54:02 +04:00
accept_list = ['text/html', '*/*']
2012-01-11 20:43:04 +04:00
elif 'HTTP_ACCEPT' in request.META:
2011-04-11 19:54:02 +04:00
# Use standard HTTP Accept negotiation
2012-01-11 20:43:04 +04:00
accept_list = [token.strip() for token in request.META['HTTP_ACCEPT'].split(',')]
2011-04-11 19:54:02 +04:00
else:
# No accept header specified
accept_list = ['*/*']
# Check the acceptable media types against each renderer,
# attempting more specific media types first
# NB. The inner loop here isn't as bad as it first looks :)
# Worst case is we're looping over len(accept_list) * len(self.renderers)
renderers = [renderer_cls(self) for renderer_cls in self.renderers]
for accepted_media_type_lst in order_by_precedence(accept_list):
for renderer in renderers:
for accepted_media_type in accepted_media_type_lst:
if renderer.can_handle_response(accepted_media_type):
return renderer, accepted_media_type
# No acceptable renderers were found
2011-04-11 19:54:02 +04:00
raise ErrorResponse(status.HTTP_406_NOT_ACCEPTABLE,
{'detail': 'Could not satisfy the client\'s Accept header',
2011-05-12 18:11:14 +04:00
'available_types': self._rendered_media_types})
2011-04-11 19:54:02 +04:00
@property
2011-05-12 18:11:14 +04:00
def _rendered_media_types(self):
2011-05-10 15:51:49 +04:00
"""
2011-05-12 18:11:14 +04:00
Return an list of all the media types that this view can render.
2011-05-10 15:51:49 +04:00
"""
2011-04-28 22:54:30 +04:00
return [renderer.media_type for renderer in self.renderers]
2011-12-09 17:37:53 +04:00
@property
def _rendered_formats(self):
"""
Return a list of all the formats that this view can render.
"""
return [renderer.format for renderer in self.renderers]
2011-04-11 19:54:02 +04:00
@property
2011-05-12 18:11:14 +04:00
def _default_renderer(self):
2011-05-10 15:51:49 +04:00
"""
Return the view's default renderer class.
2011-05-10 15:51:49 +04:00
"""
2011-04-28 22:54:30 +04:00
return self.renderers[0]
2011-04-11 19:54:02 +04:00
########## Auth Mixin ##########
class AuthMixin(object):
2011-05-10 13:49:28 +04:00
"""
Simple :class:`mixin` class to add authentication and permission checking to a :class:`View` class.
2011-05-10 13:49:28 +04:00
"""
2011-12-09 17:37:53 +04:00
2012-01-19 07:59:30 +04:00
authentication = ()
2011-05-13 12:59:36 +04:00
"""
The set of authentication types that this view can handle.
2011-12-09 17:37:53 +04:00
Should be a tuple/list of classes as described in the :mod:`authentication` module.
2011-05-13 12:59:36 +04:00
"""
2012-01-19 07:59:30 +04:00
permissions = ()
2011-05-13 12:59:36 +04:00
"""
The set of permissions that will be enforced on this view.
2011-12-09 17:37:53 +04:00
Should be a tuple/list of classes as described in the :mod:`permissions` module.
2011-05-13 12:59:36 +04:00
"""
@property
2011-05-10 13:49:28 +04:00
def user(self):
2011-05-13 12:59:36 +04:00
"""
Returns the :obj:`user` for the current request, as determined by the set of
2011-12-09 17:37:53 +04:00
:class:`authentication` classes applied to the :class:`View`.
2011-05-13 12:59:36 +04:00
"""
2011-05-10 13:49:28 +04:00
if not hasattr(self, '_user'):
self._user = self._authenticate()
return self._user
def _authenticate(self):
2011-05-10 13:49:28 +04:00
"""
Attempt to authenticate the request using each authentication class in turn.
Returns a ``User`` object, which may be ``AnonymousUser``.
"""
for authentication_cls in self.authentication:
authentication = authentication_cls(self)
2011-05-10 13:49:28 +04:00
user = authentication.authenticate(self.request)
if user:
return user
return AnonymousUser()
2011-05-10 15:51:49 +04:00
# TODO: wrap this behavior around dispatch()
2011-05-10 13:49:28 +04:00
def _check_permissions(self):
"""
Check user permissions and either raise an ``ErrorResponse`` or return.
"""
user = self.user
for permission_cls in self.permissions:
permission = permission_cls(self)
2011-05-13 12:59:36 +04:00
permission.check_permission(user)
########## Resource Mixin ##########
2011-05-12 18:11:14 +04:00
class ResourceMixin(object):
"""
Provides request validation and response filtering behavior.
Should be a class as described in the :mod:`resources` module.
The :obj:`resource` is an object that maps a view onto it's representation on the server.
It provides validation on the content of incoming requests,
and filters the object representation into a serializable object for the response.
"""
resource = None
@property
def CONTENT(self):
2011-05-17 12:15:35 +04:00
"""
Returns the cleaned, validated request content.
May raise an :class:`response.ErrorResponse` with status code 400 (Bad Request).
2011-05-17 12:15:35 +04:00
"""
if not hasattr(self, '_content'):
self._content = self.validate_request(self.request.DATA, self.request.FILES)
return self._content
@property
def PARAMS(self):
"""
Returns the cleaned, validated query parameters.
May raise an :class:`response.ErrorResponse` with status code 400 (Bad Request).
"""
return self.validate_request(self.request.GET)
@property
def _resource(self):
if self.resource:
return self.resource(self)
elif getattr(self, 'model', None):
return ModelResource(self)
elif getattr(self, 'form', None):
return FormResource(self)
elif getattr(self, '%s_form' % self.request.method.lower(), None):
return FormResource(self)
return Resource(self)
def validate_request(self, data, files=None):
"""
Given the request *data* and optional *files*, return the cleaned, validated content.
May raise an :class:`response.ErrorResponse` with status code 400 (Bad Request) on failure.
"""
return self._resource.validate_request(data, files)
def filter_response(self, obj):
"""
Given the response content, filter it into a serializable object.
"""
return self._resource.filter_response(obj)
def get_bound_form(self, content=None, method=None):
if hasattr(self._resource, 'get_bound_form'):
return self._resource.get_bound_form(content, method=method)
else:
return None
2011-05-17 12:15:35 +04:00
##########
2012-01-21 22:24:10 +04:00
2011-05-17 12:15:35 +04:00
class InstanceMixin(object):
"""
`Mixin` class that is used to identify a `View` class as being the canonical identifier
for the resources it is mapped to.
2011-05-17 12:15:35 +04:00
"""
@classmethod
def as_view(cls, **initkwargs):
"""
Store the callable object on the resource class that has been associated with this view.
"""
view = super(InstanceMixin, cls).as_view(**initkwargs)
2011-05-23 20:07:31 +04:00
resource = getattr(cls(**initkwargs), 'resource', None)
if resource:
2011-05-17 12:15:35 +04:00
# We do a little dance when we store the view callable...
# we need to store it wrapped in a 1-tuple, so that inspect will treat it
# as a function when we later look it up (rather than turning it into a method).
# This makes sure our URL reversing works ok.
2011-05-23 20:07:31 +04:00
resource.view_callable = (view,)
2011-05-17 12:15:35 +04:00
return view
2011-05-02 22:49:12 +04:00
########## Model Mixins ##########
2012-01-11 00:42:50 +04:00
class ModelMixin(object):
""" Implements mechanisms used by other classes (like *ModelMixin group) to
define a query that represents Model instances the Mixin is working with.
If a *ModelMixin is going to retrive an instance (or queryset) using args and kwargs
passed by as URL arguments, it should provied arguments to objects.get and objects.filter
methods wrapped in by `build_query`
2012-01-11 18:07:33 +04:00
If a *ModelMixin is going to create/update an instance get_instance_data
handles the instance data creation/preaparation.
2012-01-11 00:42:50 +04:00
"""
2012-01-21 22:24:10 +04:00
queryset = None
2012-02-02 00:48:32 +04:00
def get_query_kwargs(self, *args, **kwargs):
"""
Return a dict of kwargs that will be used to build the
model instance retrieval or to filter querysets.
2012-01-11 00:42:50 +04:00
"""
2012-02-02 00:48:32 +04:00
kwargs = dict(kwargs)
2012-01-11 18:07:33 +04:00
# If the URLconf includes a .(?P<format>\w+) pattern to match against
# a .json, .xml suffix, then drop the 'format' kwarg before
# constructing the query.
2012-02-02 00:48:32 +04:00
if BaseRenderer._FORMAT_QUERY_PARAM in kwargs:
del kwargs[BaseRenderer._FORMAT_QUERY_PARAM]
2012-01-11 18:07:33 +04:00
2012-02-02 00:48:32 +04:00
return kwargs
2012-01-11 00:42:50 +04:00
def get_instance_data(self, model, content, **kwargs):
2012-01-11 18:07:33 +04:00
"""
2012-02-02 00:48:32 +04:00
Returns the dict with the data for model instance creation/update.
2012-01-11 00:42:50 +04:00
Arguments:
- model: model class (django.db.models.Model subclass) to work with
- content: a dictionary with instance data
- kwargs: a dict of URL provided keyword arguments
The create/update queries are created basicly with the contet provided
2012-01-11 18:07:33 +04:00
with POST/PUT HTML methods and kwargs passed in the URL. This methods
simply merges the URL data and the content preaparing the ready-to-use
data dictionary.
2012-01-11 00:42:50 +04:00
"""
tmp = dict(kwargs)
for field in model._meta.fields:
2012-01-11 18:07:33 +04:00
if isinstance(field, ForeignKey) and field.name in tmp:
2012-01-11 00:42:50 +04:00
# translate 'related_field' kwargs into 'related_field_id'
tmp[field.name + '_id'] = tmp[field.name]
del tmp[field.name]
all_kw_args = dict(content.items() + tmp.items())
return all_kw_args
2012-02-02 00:48:32 +04:00
def get_instance(self, **kwargs):
"""
2012-02-02 00:48:32 +04:00
Get a model instance for read/update/delete requests.
"""
2012-02-02 00:48:32 +04:00
return self.get_queryset().get(**kwargs)
2012-01-21 22:24:10 +04:00
def get_queryset(self):
"""
Return the queryset for this view.
"""
return getattr(self.resource, 'queryset',
self.resource.model.objects.all())
def get_ordering(self):
"""
Return the ordering for this view.
"""
return getattr(self.resource, 'ordering', None)
2012-01-11 00:42:50 +04:00
class ReadModelMixin(ModelMixin):
2011-05-10 13:49:28 +04:00
"""
Behavior to read a `model` instance on GET requests
2011-05-10 13:49:28 +04:00
"""
2011-05-02 22:49:12 +04:00
def get(self, request, *args, **kwargs):
2011-05-04 12:21:17 +04:00
model = self.resource.model
2012-02-02 00:48:32 +04:00
query_kwargs = self.get_query_kwargs(request, *args, **kwargs)
2011-05-02 22:49:12 +04:00
try:
2012-02-02 00:48:32 +04:00
self.model_instance = self.get_instance(**query_kwargs)
2011-05-04 12:21:17 +04:00
except model.DoesNotExist:
2011-05-02 22:49:12 +04:00
raise ErrorResponse(status.HTTP_404_NOT_FOUND)
return self.model_instance
2011-05-02 22:49:12 +04:00
2012-01-11 18:07:33 +04:00
2012-01-11 00:42:50 +04:00
class CreateModelMixin(ModelMixin):
2011-05-10 13:49:28 +04:00
"""
Behavior to create a `model` instance on POST requests
2011-05-10 13:49:28 +04:00
"""
def post(self, request, *args, **kwargs):
2011-05-04 12:21:17 +04:00
model = self.resource.model
# Copy the dict to keep self.CONTENT intact
content = dict(self.CONTENT)
m2m_data = {}
for field in model._meta.many_to_many:
2012-01-11 18:07:33 +04:00
if field.name in content:
m2m_data[field.name] = (
field.m2m_reverse_field_name(), content[field.name]
)
del content[field.name]
instance = model(**self.get_instance_data(model, content, *args, **kwargs))
2011-05-02 22:49:12 +04:00
instance.save()
for fieldname in m2m_data:
manager = getattr(instance, fieldname)
2011-12-09 17:37:53 +04:00
if hasattr(manager, 'add'):
manager.add(*m2m_data[fieldname][1])
else:
data = {}
data[manager.source_field_name] = instance
2011-12-09 17:37:53 +04:00
for related_item in m2m_data[fieldname][1]:
data[m2m_data[fieldname][0]] = related_item
manager.through(**data).save()
2011-05-02 22:49:12 +04:00
headers = {}
if hasattr(instance, 'get_absolute_url'):
headers['Location'] = self.resource(self).url(instance)
2011-05-02 22:49:12 +04:00
return Response(status.HTTP_201_CREATED, instance, headers)
2012-01-11 00:42:50 +04:00
class UpdateModelMixin(ModelMixin):
2011-05-10 13:49:28 +04:00
"""
Behavior to update a `model` instance on PUT requests
2011-05-10 13:49:28 +04:00
"""
2011-05-02 22:49:12 +04:00
def put(self, request, *args, **kwargs):
2011-05-04 12:21:17 +04:00
model = self.resource.model
2012-02-02 00:48:32 +04:00
query_kwargs = self.get_query_kwargs(request, *args, **kwargs)
2011-12-09 17:37:53 +04:00
2012-01-21 22:24:10 +04:00
# TODO: update on the url of a non-existing resource url doesn't work
# correctly at the moment - will end up with a new url
2011-05-02 22:49:12 +04:00
try:
2012-02-02 00:48:32 +04:00
self.model_instance = self.get_instance(*query_kwargs)
2011-05-02 22:49:12 +04:00
for (key, val) in self.CONTENT.items():
setattr(self.model_instance, key, val)
2011-05-04 12:21:17 +04:00
except model.DoesNotExist:
self.model_instance = model(**self.get_instance_data(model, self.CONTENT, *args, **kwargs))
self.model_instance.save()
return self.model_instance
2011-05-02 22:49:12 +04:00
2012-01-11 00:42:50 +04:00
class DeleteModelMixin(ModelMixin):
2011-05-10 13:49:28 +04:00
"""
Behavior to delete a `model` instance on DELETE requests
2011-05-10 13:49:28 +04:00
"""
2011-05-02 22:49:12 +04:00
def delete(self, request, *args, **kwargs):
2011-05-04 12:21:17 +04:00
model = self.resource.model
2012-02-02 00:48:32 +04:00
query_kwargs = self.get_query_kwargs(request, *args, **kwargs)
2011-05-02 22:49:12 +04:00
try:
2012-02-02 00:48:32 +04:00
instance = self.get_instance(**query_kwargs)
2011-05-04 12:21:17 +04:00
except model.DoesNotExist:
2011-05-02 22:49:12 +04:00
raise ErrorResponse(status.HTTP_404_NOT_FOUND, None, {})
instance.delete()
return
2012-01-11 00:42:50 +04:00
class ListModelMixin(ModelMixin):
2011-05-10 13:49:28 +04:00
"""
Behavior to list a set of `model` instances on GET requests
2011-05-10 13:49:28 +04:00
"""
2011-05-02 22:49:12 +04:00
def get(self, request, *args, **kwargs):
queryset = self.get_queryset()
2012-01-21 22:24:10 +04:00
ordering = self.get_ordering()
2012-02-02 00:48:32 +04:00
query_kwargs = self.get_query_kwargs(request, *args, **kwargs)
2012-02-02 00:48:32 +04:00
queryset = queryset.filter(**query_kwargs)
if ordering:
2012-01-21 22:24:10 +04:00
queryset = queryset.order_by(*ordering)
2011-05-02 22:49:12 +04:00
2012-01-21 22:24:10 +04:00
return queryset
2011-05-02 22:49:12 +04:00
2011-12-09 17:37:53 +04:00
########## Pagination Mixins ##########
class PaginatorMixin(object):
"""
Adds pagination support to GET requests
Obviously should only be used on lists :)
A default limit can be set by setting `limit` on the object. This will also
be used as the maximum if the client sets the `limit` GET param
"""
limit = 20
def get_limit(self):
2012-01-21 22:36:25 +04:00
"""
Helper method to determine what the `limit` should be
"""
2011-12-09 17:37:53 +04:00
try:
limit = int(self.request.GET.get('limit', self.limit))
return min(limit, self.limit)
except ValueError:
return self.limit
def url_with_page_number(self, page_number):
2012-01-21 22:36:25 +04:00
"""
Constructs a url used for getting the next/previous urls
"""
url = URLObject.parse(self.request.get_full_path())
url = url.add_query_param('page', page_number)
2011-12-09 17:37:53 +04:00
limit = self.get_limit()
if limit != self.limit:
url = url.add_query_param('limit', limit)
2011-12-09 17:37:53 +04:00
return url
def next(self, page):
2012-01-21 22:36:25 +04:00
"""
Returns a url to the next page of results (if any)
"""
2011-12-09 17:37:53 +04:00
if not page.has_next():
return None
return self.url_with_page_number(page.next_page_number())
def previous(self, page):
""" Returns a url to the previous page of results (if any) """
if not page.has_previous():
return None
return self.url_with_page_number(page.previous_page_number())
def serialize_page_info(self, page):
2012-01-21 22:36:25 +04:00
"""
This is some useful information that is added to the response
"""
2011-12-09 17:37:53 +04:00
return {
'next': self.next(page),
'page': page.number,
'pages': page.paginator.num_pages,
'per_page': self.get_limit(),
'previous': self.previous(page),
'total': page.paginator.count,
}
def filter_response(self, obj):
"""
Given the response content, paginate and then serialize.
The response is modified to include to useful data relating to the number
of objects, number of pages, next/previous urls etc. etc.
The serialised objects are put into `results` on this new, modified
response
"""
# We don't want to paginate responses for anything other than GET requests
if self.request.method.upper() != 'GET':
2011-12-09 17:37:53 +04:00
return self._resource.filter_response(obj)
paginator = Paginator(obj, self.get_limit())
try:
page_num = int(self.request.GET.get('page', '1'))
except ValueError:
raise ErrorResponse(status.HTTP_404_NOT_FOUND,
{'detail': 'That page contains no results'})
if page_num not in paginator.page_range:
raise ErrorResponse(status.HTTP_404_NOT_FOUND,
{'detail': 'That page contains no results'})
page = paginator.page(page_num)
serialized_object_list = self._resource.filter_response(page.object_list)
serialized_page_info = self.serialize_page_info(page)
serialized_page_info['results'] = serialized_object_list
return serialized_page_info