2011-05-12 15:55:13 +04:00
|
|
|
"""
|
2011-12-09 17:37:53 +04:00
|
|
|
The :mod:`mixins` module provides a set of reusable `mixin`
|
2011-05-19 00:13:48 +04:00
|
|
|
classes that can be added to a `View`.
|
2011-05-12 15:55:13 +04:00
|
|
|
"""
|
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
|
2011-04-11 19:54:02 +04:00
|
|
|
from django.http import HttpResponse
|
2011-05-02 22:49:12 +04:00
|
|
|
|
2011-05-10 13:49:28 +04:00
|
|
|
from djangorestframework import status
|
2011-05-27 12:58:21 +04:00
|
|
|
from djangorestframework.resources import Resource, FormResource, ModelResource
|
2011-12-14 18:48:22 +04:00
|
|
|
from djangorestframework.response import Response, ErrorResponse
|
2011-12-14 18:45:59 +04:00
|
|
|
from djangorestframework.utils import MSIE_USER_AGENT_REGEX
|
2011-05-24 16:29:30 +04:00
|
|
|
from djangorestframework.utils.mediatypes import is_form_media_type, order_by_precedence
|
2011-05-10 13:49:28 +04:00
|
|
|
|
|
|
|
from StringIO import StringIO
|
2011-04-11 19:54:02 +04:00
|
|
|
|
|
|
|
|
2011-05-10 15:21:48 +04:00
|
|
|
__all__ = (
|
2011-05-12 15:55:13 +04:00
|
|
|
# Base behavior mixins
|
2011-05-10 15:21:48 +04:00
|
|
|
'RequestMixin',
|
|
|
|
'ResponseMixin',
|
|
|
|
'AuthMixin',
|
2011-05-12 15:55:13 +04:00
|
|
|
'ResourceMixin',
|
2011-05-19 11:36:55 +04:00
|
|
|
# Reverse URL lookup behavior
|
2011-05-13 20:19:12 +04:00
|
|
|
'InstanceMixin',
|
2011-05-12 15:55:13 +04:00
|
|
|
# Model behavior mixins
|
2012-01-03 21:17:25 +04:00
|
|
|
'GetResourceMixin',
|
|
|
|
'PostResourceMixin',
|
|
|
|
'PutResourceMixin',
|
|
|
|
'DeleteResourceMixin',
|
|
|
|
'ListResourceMixin',
|
2011-05-10 15:21:48 +04:00
|
|
|
)
|
2011-05-10 13:49:28 +04:00
|
|
|
|
2011-04-11 19:54:02 +04:00
|
|
|
|
|
|
|
########## Request Mixin ##########
|
2011-04-02 19:32:37 +04:00
|
|
|
|
|
|
|
class RequestMixin(object):
|
2011-05-10 13:49:28 +04:00
|
|
|
"""
|
2011-05-19 00:13:48 +04:00
|
|
|
`Mixin` class to provide request parsing behavior.
|
2011-05-10 13:49:28 +04:00
|
|
|
"""
|
2011-04-02 19:32:37 +04:00
|
|
|
|
2011-05-12 15:55:13 +04:00
|
|
|
_USE_FORM_OVERLOADING = True
|
|
|
|
_METHOD_PARAM = '_method'
|
|
|
|
_CONTENTTYPE_PARAM = '_content_type'
|
|
|
|
_CONTENT_PARAM = '_content'
|
2011-04-02 19:32:37 +04:00
|
|
|
|
2011-05-13 12:59:36 +04:00
|
|
|
"""
|
|
|
|
The set of request parsers that the view can handle.
|
2011-12-09 17:37:53 +04:00
|
|
|
|
2011-05-19 00:13:48 +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
|
|
|
parsers = ()
|
|
|
|
|
2011-05-12 18:11:14 +04:00
|
|
|
@property
|
|
|
|
def method(self):
|
2011-04-02 19:32:37 +04:00
|
|
|
"""
|
2011-05-12 18:11:14 +04:00
|
|
|
Returns the HTTP method.
|
2011-05-13 12:59:36 +04:00
|
|
|
|
2011-12-14 18:48:22 +04:00
|
|
|
This should be used instead of just reading :const:`request.method`, as
|
|
|
|
it allows the `method` to be overridden by using a hidden `form` field
|
|
|
|
on a form POST request.
|
2011-04-02 19:32:37 +04:00
|
|
|
"""
|
|
|
|
if not hasattr(self, '_method'):
|
2011-05-12 19:03:14 +04:00
|
|
|
self._load_method_and_content_type()
|
2011-04-02 19:32:37 +04:00
|
|
|
return self._method
|
|
|
|
|
2011-05-12 18:11:14 +04:00
|
|
|
@property
|
|
|
|
def content_type(self):
|
2011-04-02 19:32:37 +04:00
|
|
|
"""
|
2011-05-10 13:49:28 +04:00
|
|
|
Returns the content type header.
|
2011-05-13 12:59:36 +04:00
|
|
|
|
|
|
|
This should be used instead of ``request.META.get('HTTP_CONTENT_TYPE')``,
|
|
|
|
as it allows the content type to be overridden by using a hidden form
|
|
|
|
field on a form POST request.
|
2011-04-02 19:32:37 +04:00
|
|
|
"""
|
|
|
|
if not hasattr(self, '_content_type'):
|
2011-05-12 19:03:14 +04:00
|
|
|
self._load_method_and_content_type()
|
2011-04-02 19:32:37 +04:00
|
|
|
return self._content_type
|
|
|
|
|
2011-05-12 18:11:14 +04:00
|
|
|
@property
|
|
|
|
def DATA(self):
|
2011-04-02 19:32:37 +04:00
|
|
|
"""
|
2011-05-13 12:59:36 +04:00
|
|
|
Parses the request body and returns the data.
|
|
|
|
|
|
|
|
Similar to ``request.POST``, except that it handles arbitrary parsers,
|
|
|
|
and also works on methods other than POST (eg PUT).
|
2011-04-02 19:32:37 +04:00
|
|
|
"""
|
2011-05-12 18:11:14 +04:00
|
|
|
if not hasattr(self, '_data'):
|
|
|
|
self._load_data_and_files()
|
|
|
|
return self._data
|
|
|
|
|
|
|
|
@property
|
|
|
|
def FILES(self):
|
|
|
|
"""
|
2011-05-13 12:59:36 +04:00
|
|
|
Parses the request body and returns the files.
|
2011-05-19 00:13:48 +04:00
|
|
|
Similar to ``request.FILES``, except that it handles arbitrary parsers,
|
2011-05-13 12:59:36 +04:00
|
|
|
and also works on methods other than POST (eg PUT).
|
2011-05-12 18:11:14 +04:00
|
|
|
"""
|
|
|
|
if not hasattr(self, '_files'):
|
|
|
|
self._load_data_and_files()
|
|
|
|
return self._files
|
|
|
|
|
|
|
|
def _load_data_and_files(self):
|
|
|
|
"""
|
|
|
|
Parse the request content into self.DATA and self.FILES.
|
|
|
|
"""
|
2011-05-12 19:03:14 +04:00
|
|
|
if not hasattr(self, '_content_type'):
|
|
|
|
self._load_method_and_content_type()
|
|
|
|
|
|
|
|
if not hasattr(self, '_data'):
|
2011-12-14 18:48:22 +04:00
|
|
|
(self._data, self._files) = self._parse(self._get_stream(),
|
|
|
|
self._content_type)
|
2011-05-12 19:03:14 +04:00
|
|
|
|
|
|
|
def _load_method_and_content_type(self):
|
2011-05-12 18:14:22 +04:00
|
|
|
"""
|
2011-12-14 18:48:22 +04:00
|
|
|
Set the method and content_type, and then check if they've been
|
|
|
|
overridden.
|
2011-05-12 18:14:22 +04:00
|
|
|
"""
|
|
|
|
self._method = self.request.method
|
2011-12-14 18:48:22 +04:00
|
|
|
self._content_type = self.request.META.get('HTTP_CONTENT_TYPE',
|
|
|
|
self.request.META.get('CONTENT_TYPE', ''))
|
2011-05-12 18:14:22 +04:00
|
|
|
self._perform_form_overloading()
|
|
|
|
|
2011-04-02 19:32:37 +04:00
|
|
|
def _get_stream(self):
|
|
|
|
"""
|
|
|
|
Returns an object that may be used to stream the request content.
|
|
|
|
"""
|
2011-05-12 19:03:14 +04:00
|
|
|
request = self.request
|
|
|
|
|
|
|
|
try:
|
2011-12-14 18:48:22 +04:00
|
|
|
content_length = int(request.META.get('CONTENT_LENGTH',
|
|
|
|
request.META.get('HTTP_CONTENT_LENGTH')))
|
2011-05-12 19:03:14 +04:00
|
|
|
except (ValueError, TypeError):
|
|
|
|
content_length = 0
|
|
|
|
|
|
|
|
# TODO: Add 1.3's LimitedStream to compat and use that.
|
2011-05-13 20:19:12 +04:00
|
|
|
# NOTE: Currently only supports parsing request body as a stream with 1.3
|
2011-05-12 19:03:14 +04:00
|
|
|
if content_length == 0:
|
|
|
|
return None
|
|
|
|
elif hasattr(request, 'read'):
|
2011-12-14 18:48:22 +04:00
|
|
|
return request
|
2011-05-12 19:03:14 +04:00
|
|
|
return StringIO(request.raw_post_data)
|
|
|
|
|
2011-05-12 15:55:13 +04:00
|
|
|
def _perform_form_overloading(self):
|
2011-04-02 19:32:37 +04:00
|
|
|
"""
|
2011-12-14 18:48:22 +04:00
|
|
|
If this is a form POST request, then we need to check if the method and
|
|
|
|
content/content_type have been overridden by setting them in hidden
|
|
|
|
form fields or not.
|
2011-04-02 19:32:37 +04:00
|
|
|
"""
|
2011-05-12 18:11:14 +04:00
|
|
|
|
2011-05-12 19:03:14 +04:00
|
|
|
# We only need to use form overloading on form POST requests.
|
2011-12-14 18:48:22 +04:00
|
|
|
if (not self._USE_FORM_OVERLOADING
|
|
|
|
or self._method != 'POST'
|
|
|
|
or not is_form_media_type(self._content_type)):
|
2011-04-02 19:32:37 +04:00
|
|
|
return
|
2011-12-09 17:37:53 +04:00
|
|
|
|
2011-05-12 19:03:14 +04:00
|
|
|
# At this point we're committed to parsing the request as form data.
|
2011-07-06 15:05:57 +04:00
|
|
|
self._data = data = self.request.POST.copy()
|
2011-05-12 19:03:14 +04:00
|
|
|
self._files = self.request.FILES
|
|
|
|
|
|
|
|
# Method overloading - change the method and remove the param from the content.
|
|
|
|
if self._METHOD_PARAM in data:
|
2011-07-06 15:05:57 +04:00
|
|
|
# NOTE: unlike `get`, `pop` on a `QueryDict` seems to return a list of values.
|
|
|
|
self._method = self._data.pop(self._METHOD_PARAM)[0].upper()
|
2011-05-12 19:03:14 +04:00
|
|
|
|
|
|
|
# Content overloading - modify the content type, and re-parse.
|
|
|
|
if self._CONTENT_PARAM in data and self._CONTENTTYPE_PARAM in data:
|
2011-07-06 15:05:57 +04:00
|
|
|
self._content_type = self._data.pop(self._CONTENTTYPE_PARAM)[0]
|
|
|
|
stream = StringIO(self._data.pop(self._CONTENT_PARAM)[0])
|
2011-05-12 19:03:14 +04:00
|
|
|
(self._data, self._files) = self._parse(stream, self._content_type)
|
2011-04-02 19:32:37 +04:00
|
|
|
|
2011-05-12 15:55:13 +04:00
|
|
|
def _parse(self, stream, content_type):
|
2011-04-11 14:24:14 +04:00
|
|
|
"""
|
|
|
|
Parse the request content.
|
|
|
|
|
2011-12-14 18:48:22 +04:00
|
|
|
May raise a 415 ErrorResponse (Unsupported Media Type), or a 400
|
|
|
|
ErrorResponse (Bad Request).
|
2011-04-11 14:24:14 +04:00
|
|
|
"""
|
2011-04-11 14:47:22 +04:00
|
|
|
if stream is None or content_type is None:
|
2011-05-12 15:55:13 +04:00
|
|
|
return (None, None)
|
2011-04-11 14:47:22 +04:00
|
|
|
|
2011-12-14 18:45:59 +04:00
|
|
|
for parser_cls in self.parsers:
|
2011-05-10 13:49:28 +04:00
|
|
|
parser = parser_cls(self)
|
|
|
|
if parser.can_handle_request(content_type):
|
|
|
|
return parser.parse(stream)
|
2011-04-11 14:24:14 +04:00
|
|
|
|
2011-12-14 18:48:22 +04:00
|
|
|
error = {'error':
|
|
|
|
"Unsupported media type in request '%s'." % content_type}
|
|
|
|
raise ErrorResponse(status.HTTP_415_UNSUPPORTED_MEDIA_TYPE, error)
|
2011-04-11 18:03:49 +04:00
|
|
|
|
2011-04-11 14:24:14 +04:00
|
|
|
@property
|
2011-05-12 18:11:14 +04:00
|
|
|
def _parsed_media_types(self):
|
2011-05-12 15:55:13 +04:00
|
|
|
"""
|
2011-05-12 18:11:14 +04:00
|
|
|
Return a list of all the media types that this view can parse.
|
2011-05-12 15:55:13 +04:00
|
|
|
"""
|
2011-04-11 14:24:14 +04:00
|
|
|
return [parser.media_type for parser in self.parsers]
|
2011-04-11 19:54:02 +04:00
|
|
|
|
2011-04-11 14:24:14 +04:00
|
|
|
@property
|
2011-05-12 18:11:14 +04:00
|
|
|
def _default_parser(self):
|
2011-05-12 15:55:13 +04:00
|
|
|
"""
|
2011-05-24 16:29:30 +04:00
|
|
|
Return the view's default parser class.
|
2011-12-09 17:37:53 +04:00
|
|
|
"""
|
2011-04-11 14:24:14 +04:00
|
|
|
return self.parsers[0]
|
|
|
|
|
2011-04-11 19:54:02 +04:00
|
|
|
|
|
|
|
########## ResponseMixin ##########
|
|
|
|
|
2011-12-14 18:48:22 +04:00
|
|
|
|
2011-04-11 19:54:02 +04:00
|
|
|
class ResponseMixin(object):
|
2011-05-10 13:49:28 +04:00
|
|
|
"""
|
2011-05-24 13:27:24 +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.
|
2011-12-14 18:48:22 +04:00
|
|
|
|
|
|
|
Also supports overriding the content type by specifying an ``_accept=``
|
|
|
|
parameter in the URL.
|
|
|
|
|
|
|
|
Ignores Accept headers from Internet Explorer user agents and uses a
|
|
|
|
sensible browser Accept header instead.
|
2011-05-10 13:49:28 +04:00
|
|
|
"""
|
2011-05-12 18:11:14 +04:00
|
|
|
|
2011-12-14 18:48:22 +04:00
|
|
|
# Allow override of Accept header in URL query params
|
|
|
|
_ACCEPT_QUERY_PARAM = '_accept'
|
2011-05-12 18:11:14 +04:00
|
|
|
_IGNORE_IE_ACCEPT_HEADER = True
|
2011-04-11 19:54:02 +04:00
|
|
|
|
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
|
|
|
|
2011-12-14 18:48:22 +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-04-28 22:54:30 +04:00
|
|
|
renderers = ()
|
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
|
|
|
"""
|
2011-05-19 00:13:48 +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:
|
2011-05-24 19:31:17 +04:00
|
|
|
renderer, media_type = self._determine_renderer(self.request)
|
2011-04-11 19:54:02 +04:00
|
|
|
except ErrorResponse, exc:
|
2011-05-24 16:29:30 +04:00
|
|
|
renderer = self._default_renderer(self)
|
2011-05-24 19:31:17 +04:00
|
|
|
media_type = renderer.media_type
|
2011-04-11 19:54:02 +04:00
|
|
|
response = exc.response
|
2011-05-24 19:31:17 +04:00
|
|
|
|
|
|
|
# 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:
|
2011-05-24 19:31:17 +04:00
|
|
|
content = renderer.render(response.cleaned_content, media_type)
|
2011-04-11 19:54:02 +04:00
|
|
|
else:
|
2011-05-24 16:29:30 +04:00
|
|
|
content = renderer.render()
|
2011-04-11 19:54:02 +04:00
|
|
|
|
|
|
|
# Build the HTTP Response
|
2011-12-14 18:48:22 +04:00
|
|
|
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
|
|
|
"""
|
2011-12-14 14:48:19 +04:00
|
|
|
Determines the appropriate renderer for the output, given the client's
|
|
|
|
'Accept' header, and the :attr:`renderers` set on this class.
|
2011-05-24 19:31:17 +04:00
|
|
|
|
|
|
|
Returns a 2-tuple of `(renderer, media_type)`
|
2011-05-10 19:01:58 +04:00
|
|
|
|
2011-12-14 14:48:19 +04:00
|
|
|
See: RFC 2616, Section 14
|
|
|
|
http://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html
|
2011-05-10 15:51:49 +04:00
|
|
|
"""
|
2011-04-11 19:54:02 +04:00
|
|
|
|
2011-12-14 14:48:19 +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)]
|
2011-12-14 14:48:19 +04:00
|
|
|
|
2011-05-12 18:11:14 +04:00
|
|
|
elif (self._IGNORE_IE_ACCEPT_HEADER and
|
2011-12-14 14:48:19 +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'])):
|
2011-12-14 14:48:19 +04:00
|
|
|
# Ignore MSIE's broken accept behavior and do something sensible
|
|
|
|
# instead.
|
2011-04-11 19:54:02 +04:00
|
|
|
accept_list = ['text/html', '*/*']
|
2011-12-14 14:48:19 +04:00
|
|
|
|
2011-12-14 18:48:22 +04:00
|
|
|
elif 'HTTP_ACCEPT' in request.META:
|
2011-04-11 19:54:02 +04:00
|
|
|
# Use standard HTTP Accept negotiation
|
2011-12-14 14:48:19 +04:00
|
|
|
accept_list = [token.strip() for token in
|
2011-12-14 18:48:22 +04:00
|
|
|
request.META['HTTP_ACCEPT'].split(',')]
|
2011-12-14 14:48:19 +04:00
|
|
|
|
2011-04-11 19:54:02 +04:00
|
|
|
else:
|
|
|
|
# No accept header specified
|
2011-06-26 18:03:36 +04:00
|
|
|
accept_list = ['*/*']
|
2011-05-24 16:29:30 +04:00
|
|
|
|
|
|
|
# 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 :)
|
2011-12-14 18:48:22 +04:00
|
|
|
# Worst case is: len(accept_list) * len(self.renderers)
|
2011-05-24 16:29:30 +04:00
|
|
|
renderers = [renderer_cls(self) for renderer_cls in self.renderers]
|
|
|
|
|
2011-06-26 18:03:36 +04:00
|
|
|
for accepted_media_type_lst in order_by_precedence(accept_list):
|
2011-05-24 16:29:30 +04:00
|
|
|
for renderer in renderers:
|
2011-06-26 18:03:36 +04:00
|
|
|
for accepted_media_type in accepted_media_type_lst:
|
|
|
|
if renderer.can_handle_response(accepted_media_type):
|
|
|
|
return renderer, accepted_media_type
|
|
|
|
|
2011-05-24 16:29:30 +04:00
|
|
|
# No acceptable renderers were found
|
2011-12-14 18:48:22 +04:00
|
|
|
error = {'detail': "Could not satisfy the client's Accept header",
|
|
|
|
'available_types': self._rendered_media_types}
|
|
|
|
raise ErrorResponse(status.HTTP_406_NOT_ACCEPTABLE, error)
|
2011-05-24 16:29:30 +04:00
|
|
|
|
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
|
|
|
|
2011-06-26 18:03:36 +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
|
|
|
"""
|
2011-05-24 16:29:30 +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 ##########
|
2011-04-02 19:32:37 +04:00
|
|
|
|
2011-04-11 16:45:38 +04:00
|
|
|
class AuthMixin(object):
|
2011-05-10 13:49:28 +04:00
|
|
|
"""
|
2011-12-14 18:48:22 +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
|
|
|
|
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
|
|
|
|
2011-12-14 18:48:22 +04:00
|
|
|
Should be a tuple/list of classes as described in the :mod:`authentication`
|
|
|
|
module.
|
2011-05-13 12:59:36 +04:00
|
|
|
"""
|
2011-04-29 17:32:56 +04:00
|
|
|
authentication = ()
|
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
|
|
|
|
2011-12-14 18:48:22 +04:00
|
|
|
Should be a tuple/list of classes as described in the :mod:`permissions`
|
|
|
|
module.
|
2011-05-13 12:59:36 +04:00
|
|
|
"""
|
2011-04-25 04:03:23 +04:00
|
|
|
permissions = ()
|
2011-04-11 16:45:38 +04:00
|
|
|
|
|
|
|
@property
|
2011-05-10 13:49:28 +04:00
|
|
|
def user(self):
|
2011-05-13 12:59:36 +04:00
|
|
|
"""
|
2011-12-14 18:48:22 +04:00
|
|
|
Returns the :obj:`user` for the current request, as determined by the
|
|
|
|
set of :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
|
2011-05-13 20:19:12 +04:00
|
|
|
|
2011-04-25 04:03:23 +04:00
|
|
|
def _authenticate(self):
|
2011-05-10 13:49:28 +04:00
|
|
|
"""
|
2011-12-14 18:48:22 +04:00
|
|
|
Attempt to authenticate the request using each authentication class in
|
|
|
|
turn. Returns a ``User`` object, which may be ``AnonymousUser``.
|
2011-05-10 13:49:28 +04:00
|
|
|
"""
|
2011-04-29 17:32:56 +04:00
|
|
|
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-04-11 16:45:38 +04:00
|
|
|
|
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
|
2011-04-25 04:03:23 +04:00
|
|
|
for permission_cls in self.permissions:
|
|
|
|
permission = permission_cls(self)
|
2011-05-13 12:59:36 +04:00
|
|
|
permission.check_permission(user)
|
2011-04-25 04:03:23 +04:00
|
|
|
|
2011-04-11 16:45:38 +04:00
|
|
|
|
2011-05-12 15:55:13 +04:00
|
|
|
########## Resource Mixin ##########
|
|
|
|
|
2011-05-12 18:11:14 +04:00
|
|
|
class ResourceMixin(object):
|
2011-05-13 20:19:12 +04:00
|
|
|
"""
|
|
|
|
Provides request validation and response filtering behavior.
|
|
|
|
|
2011-05-19 00:13:48 +04:00
|
|
|
Should be a class as described in the :mod:`resources` module.
|
2011-05-13 20:19:12 +04:00
|
|
|
|
2011-12-14 18:48:22 +04:00
|
|
|
The :obj:`resource` is an object that maps a view onto it's representation
|
|
|
|
on the server.
|
2011-05-13 20:19:12 +04:00
|
|
|
|
|
|
|
It provides validation on the content of incoming requests,
|
2011-12-14 18:48:22 +04:00
|
|
|
and filters the object representation into a serializable object for the
|
|
|
|
response.
|
2011-05-13 20:19:12 +04:00
|
|
|
"""
|
2012-01-03 13:59:56 +04:00
|
|
|
resource_class = None
|
2011-05-13 20:19:12 +04:00
|
|
|
|
2011-05-12 15:55:13 +04:00
|
|
|
@property
|
|
|
|
def CONTENT(self):
|
2011-05-17 12:15:35 +04:00
|
|
|
"""
|
|
|
|
Returns the cleaned, validated request content.
|
2011-05-27 17:40:19 +04:00
|
|
|
|
2011-12-14 18:48:22 +04:00
|
|
|
May raise an :class:`response.ErrorResponse` with status code 400
|
|
|
|
(Bad Request).
|
2011-05-17 12:15:35 +04:00
|
|
|
"""
|
2011-05-12 15:55:13 +04:00
|
|
|
if not hasattr(self, '_content'):
|
2011-05-13 20:19:12 +04:00
|
|
|
self._content = self.validate_request(self.DATA, self.FILES)
|
2011-05-12 15:55:13 +04:00
|
|
|
return self._content
|
|
|
|
|
2011-05-27 17:40:19 +04:00
|
|
|
@property
|
|
|
|
def PARAMS(self):
|
|
|
|
"""
|
|
|
|
Returns the cleaned, validated query parameters.
|
|
|
|
|
2011-12-14 18:48:22 +04:00
|
|
|
May raise an :class:`response.ErrorResponse` with status code 400
|
|
|
|
(Bad Request).
|
2011-05-27 17:40:19 +04:00
|
|
|
"""
|
|
|
|
return self.validate_request(self.request.GET)
|
|
|
|
|
2012-01-03 21:17:25 +04:00
|
|
|
def get_resource_class(self):
|
2012-01-03 13:59:56 +04:00
|
|
|
if self.resource_class:
|
2012-01-03 21:17:25 +04:00
|
|
|
return self.resource_class
|
2011-05-27 13:35:43 +04:00
|
|
|
elif getattr(self, 'model', None):
|
2012-01-03 21:17:25 +04:00
|
|
|
return ModelResource
|
2011-05-27 13:35:43 +04:00
|
|
|
elif getattr(self, 'form', None):
|
2012-01-03 21:17:25 +04:00
|
|
|
return FormResource
|
|
|
|
elif hasattr(self, 'request') and getattr(self, '%s_form' % self.method.lower(), None):
|
|
|
|
return FormResource
|
|
|
|
else:
|
|
|
|
return Resource
|
|
|
|
|
|
|
|
@property
|
|
|
|
def resource(self):
|
|
|
|
if not hasattr(self, '_resource'):
|
|
|
|
resource_class = self.get_resource_class()
|
|
|
|
self._resource = resource_class(view=self)
|
|
|
|
return self._resource
|
2011-05-27 12:58:21 +04:00
|
|
|
|
2011-05-27 17:40:19 +04:00
|
|
|
def validate_request(self, data, files=None):
|
2011-05-13 20:19:12 +04:00
|
|
|
"""
|
2011-12-14 18:48:22 +04:00
|
|
|
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.
|
2011-05-13 20:19:12 +04:00
|
|
|
"""
|
2012-01-03 21:17:25 +04:00
|
|
|
return self.resource.validate_request(data, files)
|
2011-05-13 20:19:12 +04:00
|
|
|
|
|
|
|
def filter_response(self, obj):
|
|
|
|
"""
|
|
|
|
Given the response content, filter it into a serializable object.
|
|
|
|
"""
|
2012-01-03 21:17:25 +04:00
|
|
|
return self.resource.filter_response(obj)
|
2011-05-12 15:55:13 +04:00
|
|
|
|
2011-05-27 17:40:19 +04:00
|
|
|
def get_bound_form(self, content=None, method=None):
|
2012-01-03 21:17:25 +04:00
|
|
|
if hasattr(self.resource, 'get_bound_form'):
|
|
|
|
return self.resource.get_bound_form(content, method=method)
|
2011-07-31 00:23:53 +04:00
|
|
|
else:
|
|
|
|
return None
|
2011-05-12 15:55:13 +04:00
|
|
|
|
2011-05-13 20:19:12 +04:00
|
|
|
|
2011-05-17 12:15:35 +04:00
|
|
|
##########
|
|
|
|
|
2011-12-14 18:48:22 +04:00
|
|
|
|
2011-05-17 12:15:35 +04:00
|
|
|
class InstanceMixin(object):
|
|
|
|
"""
|
2011-12-14 18:48:22 +04:00
|
|
|
`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):
|
|
|
|
"""
|
2011-12-14 18:48:22 +04:00
|
|
|
Store the callable object on the resource class that has been
|
|
|
|
associated with this view.
|
2011-05-17 12:15:35 +04:00
|
|
|
"""
|
|
|
|
view = super(InstanceMixin, cls).as_view(**initkwargs)
|
2012-01-03 21:17:25 +04:00
|
|
|
resource_class = getattr(cls(**initkwargs), 'resource_class', None)
|
|
|
|
if resource_class:
|
2011-05-17 12:15:35 +04:00
|
|
|
# We do a little dance when we store the view callable...
|
2011-12-14 18:48:22 +04:00
|
|
|
# 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).
|
2011-06-15 17:09:01 +04:00
|
|
|
# This makes sure our URL reversing works ok.
|
2012-01-03 21:17:25 +04:00
|
|
|
resource_class.view_callable = (view,)
|
2011-05-17 12:15:35 +04:00
|
|
|
return view
|
|
|
|
|
2011-05-12 15:55:13 +04:00
|
|
|
|
2012-01-03 13:59:56 +04:00
|
|
|
########## Resource operation Mixins ##########
|
2011-05-02 22:49:12 +04:00
|
|
|
|
2012-01-03 21:17:25 +04:00
|
|
|
class GetResourceMixin(object):
|
2011-12-14 18:48:22 +04:00
|
|
|
|
2012-01-03 13:59:56 +04:00
|
|
|
def get(self, request, *args, **kwargs):
|
2011-05-02 22:49:12 +04:00
|
|
|
try:
|
2012-01-03 21:17:25 +04:00
|
|
|
self.resource.retrieve(request, *args, **kwargs)
|
|
|
|
except self.resource.DoesNotExist:
|
2012-01-03 13:59:56 +04:00
|
|
|
raise ErrorResponse(status.HTTP_404_NOT_FOUND)
|
2012-01-03 21:17:25 +04:00
|
|
|
return self.resource.instance
|
2011-12-09 17:37:53 +04:00
|
|
|
|
|
|
|
|
2012-01-03 21:17:25 +04:00
|
|
|
class PostResourceMixin(object):
|
2011-07-05 09:25:39 +04:00
|
|
|
|
2012-01-03 13:59:56 +04:00
|
|
|
def post(self, request, *args, **kwargs):
|
2012-01-03 21:17:25 +04:00
|
|
|
self.resource.create(request, *args, **kwargs)
|
|
|
|
self.resource.update(self.CONTENT, request, *args, **kwargs)
|
|
|
|
headers = {'Location': self.resource.get_url()}
|
|
|
|
return Response(status.HTTP_201_CREATED, self.resource.instance, headers)
|
2011-05-02 22:49:12 +04:00
|
|
|
|
|
|
|
|
2012-01-03 21:17:25 +04:00
|
|
|
class PutResourceMixin(object):
|
2011-12-14 18:48:22 +04:00
|
|
|
|
2012-01-03 13:59:56 +04:00
|
|
|
def put(self, request, *args, **kwargs):
|
2011-12-14 18:48:22 +04:00
|
|
|
headers = {}
|
|
|
|
try:
|
2012-01-03 21:17:25 +04:00
|
|
|
self.resource.retrieve(request, *args, **kwargs)
|
2012-01-03 13:59:56 +04:00
|
|
|
status_code = status.HTTP_204_NO_CONTENT
|
2012-01-03 21:17:25 +04:00
|
|
|
except self.resource.DoesNotExist:
|
|
|
|
self.resource.create(request, *args, **kwargs)
|
2012-01-03 13:59:56 +04:00
|
|
|
status_code = status.HTTP_201_CREATED
|
2012-01-03 21:17:25 +04:00
|
|
|
self.resource.update(self.CONTENT, request, *args, **kwargs)
|
|
|
|
return Response(status_code, self.resource.instance, {})
|
2011-12-14 18:48:22 +04:00
|
|
|
|
|
|
|
|
2012-01-03 13:59:56 +04:00
|
|
|
class DeleteResourceMixin(object):
|
2011-12-14 18:48:22 +04:00
|
|
|
|
2012-01-03 13:59:56 +04:00
|
|
|
def delete(self, request, *args, **kwargs):
|
|
|
|
try:
|
2012-01-03 21:17:25 +04:00
|
|
|
self.resource.retrieve(request, *args, **kwargs)
|
|
|
|
except self.resource.DoesNotExist:
|
2012-01-03 13:59:56 +04:00
|
|
|
raise ErrorResponse(status.HTTP_404_NOT_FOUND)
|
2012-01-03 21:17:25 +04:00
|
|
|
self.resource.delete(request, *args, **kwargs)
|
2012-01-03 13:59:56 +04:00
|
|
|
return
|
2011-12-14 18:48:22 +04:00
|
|
|
|
2011-05-02 22:49:12 +04:00
|
|
|
|
2012-01-03 21:17:25 +04:00
|
|
|
class ListResourceMixin(object):
|
|
|
|
|
|
|
|
def get(self, request, *args, **kwargs):
|
|
|
|
return self.resource.list(request, *args, **kwargs)
|
|
|
|
|
|
|
|
|
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):
|
|
|
|
""" Helper method to determine what the `limit` should be """
|
|
|
|
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):
|
2011-12-14 18:48:22 +04:00
|
|
|
"""Constructs a url used for getting the next/previous urls."""
|
2011-12-09 17:37:53 +04:00
|
|
|
url = "%s?page=%d" % (self.request.path, page_number)
|
|
|
|
|
|
|
|
limit = self.get_limit()
|
|
|
|
if limit != self.limit:
|
|
|
|
url = "%s&limit=%d" % (url, limit)
|
|
|
|
|
|
|
|
return url
|
|
|
|
|
|
|
|
def next(self, page):
|
2011-12-14 18:48:22 +04:00
|
|
|
"""Returns a url to the next page of results. (If any exists.)"""
|
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):
|
2011-12-14 18:48:22 +04:00
|
|
|
"""Returns a url to the previous page of results. (If any exists.)"""
|
2011-12-09 17:37:53 +04:00
|
|
|
if not page.has_previous():
|
|
|
|
return None
|
|
|
|
|
|
|
|
return self.url_with_page_number(page.previous_page_number())
|
|
|
|
|
|
|
|
def serialize_page_info(self, page):
|
2011-12-14 18:48:22 +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.
|
|
|
|
|
2011-12-14 18:48:22 +04:00
|
|
|
The response is modified to include to useful data relating to the
|
|
|
|
number of objects, number of pages, next/previous urls etc. etc.
|
2011-12-09 17:37:53 +04:00
|
|
|
|
|
|
|
The serialised objects are put into `results` on this new, modified
|
|
|
|
response
|
|
|
|
"""
|
|
|
|
|
2011-12-14 18:48:22 +04:00
|
|
|
# We don't want to paginate responses for anything other than GET
|
|
|
|
# requests
|
2011-12-09 17:37:53 +04:00
|
|
|
if self.method.upper() != 'GET':
|
2012-01-03 21:17:25 +04:00
|
|
|
return self.resource.filter_response(obj)
|
2011-12-09 17:37:53 +04:00
|
|
|
|
|
|
|
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)
|
|
|
|
|
2012-01-03 21:17:25 +04:00
|
|
|
serialized_object_list = self.resource.filter_response(page.object_list)
|
2011-12-09 17:37:53 +04:00
|
|
|
serialized_page_info = self.serialize_page_info(page)
|
|
|
|
|
|
|
|
serialized_page_info['results'] = serialized_object_list
|
|
|
|
|
|
|
|
return serialized_page_info
|