2011-05-10 15:21:48 +04:00
|
|
|
"""
|
|
|
|
Renderers are used to serialize a View's output into specific media types.
|
2011-05-10 15:51:49 +04:00
|
|
|
|
|
|
|
Django REST framework also provides HTML and PlainText renderers that help self-document the API,
|
|
|
|
by serializing the output along with documentation regarding the View, output status and headers,
|
2011-12-29 17:31:12 +04:00
|
|
|
and providing forms and links depending on the allowed methods, renderers and parsers on the View.
|
2011-01-28 01:30:10 +03:00
|
|
|
"""
|
2011-04-26 23:20:31 +04:00
|
|
|
from django import forms
|
2011-01-24 02:08:16 +03:00
|
|
|
from django.template import RequestContext, loader
|
2011-04-25 07:50:28 +04:00
|
|
|
from django.utils import simplejson as json
|
2011-01-24 02:08:16 +03:00
|
|
|
|
2012-01-23 22:32:37 +04:00
|
|
|
from djangorestframework.compat import yaml
|
2012-09-07 13:41:16 +04:00
|
|
|
from djangorestframework.settings import api_settings
|
2012-02-25 22:45:17 +04:00
|
|
|
from djangorestframework.utils import dict2xml
|
2012-09-03 16:30:20 +04:00
|
|
|
from djangorestframework.utils import encoders
|
2011-04-29 17:32:56 +04:00
|
|
|
from djangorestframework.utils.breadcrumbs import get_breadcrumbs
|
2011-05-24 16:29:30 +04:00
|
|
|
from djangorestframework.utils.mediatypes import get_media_type_params, add_media_type_param, media_type_matches
|
2011-06-25 13:25:26 +04:00
|
|
|
from djangorestframework import VERSION
|
2012-09-08 20:18:05 +04:00
|
|
|
from djangorestframework.fields import FloatField, IntegerField, DateTimeField, DateField, EmailField, CharField, BooleanField
|
2011-02-19 13:26:27 +03:00
|
|
|
|
2011-05-10 15:21:48 +04:00
|
|
|
import string
|
2012-02-25 22:45:17 +04:00
|
|
|
|
2011-05-10 15:21:48 +04:00
|
|
|
|
|
|
|
__all__ = (
|
|
|
|
'BaseRenderer',
|
2011-05-10 19:01:58 +04:00
|
|
|
'TemplateRenderer',
|
2011-05-10 15:21:48 +04:00
|
|
|
'JSONRenderer',
|
2011-12-29 17:24:52 +04:00
|
|
|
'JSONPRenderer',
|
2011-05-10 15:21:48 +04:00
|
|
|
'DocumentingHTMLRenderer',
|
|
|
|
'DocumentingXHTMLRenderer',
|
|
|
|
'DocumentingPlainTextRenderer',
|
2011-06-26 03:34:52 +04:00
|
|
|
'XMLRenderer',
|
|
|
|
'YAMLRenderer'
|
2011-05-10 15:21:48 +04:00
|
|
|
)
|
2011-01-24 02:08:16 +03:00
|
|
|
|
2011-01-28 00:09:25 +03:00
|
|
|
|
2011-04-29 17:32:56 +04:00
|
|
|
class BaseRenderer(object):
|
2011-05-10 15:21:48 +04:00
|
|
|
"""
|
2011-05-19 00:13:48 +04:00
|
|
|
All renderers must extend this class, set the :attr:`media_type` attribute,
|
|
|
|
and override the :meth:`render` method.
|
2011-05-10 15:21:48 +04:00
|
|
|
"""
|
2011-12-29 17:31:12 +04:00
|
|
|
|
2011-06-26 18:03:36 +04:00
|
|
|
_FORMAT_QUERY_PARAM = 'format'
|
2011-05-24 16:29:30 +04:00
|
|
|
|
2011-01-24 02:08:16 +03:00
|
|
|
media_type = None
|
2011-06-26 18:03:36 +04:00
|
|
|
format = None
|
2011-01-24 02:08:16 +03:00
|
|
|
|
2012-02-07 15:15:30 +04:00
|
|
|
def __init__(self, view=None):
|
2011-05-10 13:49:28 +04:00
|
|
|
self.view = view
|
2011-01-24 02:08:16 +03:00
|
|
|
|
2012-09-15 01:42:29 +04:00
|
|
|
def can_handle_format(self, format):
|
|
|
|
return format == self.format
|
|
|
|
|
|
|
|
def can_handle_media_type(self, media_type):
|
2011-05-24 16:29:30 +04:00
|
|
|
"""
|
2012-09-15 01:42:29 +04:00
|
|
|
Returns `True` if this renderer is able to deal with the given
|
|
|
|
media type.
|
2011-05-24 16:29:30 +04:00
|
|
|
|
2012-09-15 01:42:29 +04:00
|
|
|
The default implementation for this function is to check the media type
|
|
|
|
argument against the media_type attribute set on the class to see if
|
2011-05-24 16:29:30 +04:00
|
|
|
they match.
|
|
|
|
|
2012-09-15 01:42:29 +04:00
|
|
|
This may be overridden to provide for other behavior, but typically
|
|
|
|
you'll instead want to just set the `media_type` attribute on the class.
|
2011-05-24 16:29:30 +04:00
|
|
|
"""
|
2012-09-15 01:42:29 +04:00
|
|
|
return media_type_matches(self.media_type, media_type)
|
2011-05-24 16:29:30 +04:00
|
|
|
|
2011-05-10 15:21:48 +04:00
|
|
|
def render(self, obj=None, media_type=None):
|
|
|
|
"""
|
2011-05-10 15:51:49 +04:00
|
|
|
Given an object render it into a string.
|
|
|
|
|
|
|
|
The requested media type is also passed to this method,
|
|
|
|
as it may contain parameters relevant to how the parser
|
|
|
|
should render the output.
|
2011-05-19 00:13:48 +04:00
|
|
|
EG: ``application/json; indent=4``
|
2011-05-10 15:51:49 +04:00
|
|
|
|
2011-05-12 18:11:14 +04:00
|
|
|
By default render simply returns the output as-is.
|
2011-05-10 15:21:48 +04:00
|
|
|
Override this method to provide for other behavior.
|
|
|
|
"""
|
|
|
|
if obj is None:
|
2011-01-26 11:58:09 +03:00
|
|
|
return ''
|
2011-12-29 17:31:12 +04:00
|
|
|
|
2011-05-10 15:51:49 +04:00
|
|
|
return str(obj)
|
2011-01-24 02:08:16 +03:00
|
|
|
|
2011-01-28 01:30:10 +03:00
|
|
|
|
2011-05-12 18:11:14 +04:00
|
|
|
class JSONRenderer(BaseRenderer):
|
|
|
|
"""
|
|
|
|
Renderer which serializes to JSON
|
|
|
|
"""
|
2011-05-24 16:29:30 +04:00
|
|
|
|
2011-05-12 18:11:14 +04:00
|
|
|
media_type = 'application/json'
|
2011-06-26 18:03:36 +04:00
|
|
|
format = 'json'
|
2012-09-03 16:30:20 +04:00
|
|
|
encoder_class = encoders.JSONEncoder
|
2011-05-12 18:11:14 +04:00
|
|
|
|
|
|
|
def render(self, obj=None, media_type=None):
|
2011-05-24 16:29:30 +04:00
|
|
|
"""
|
|
|
|
Renders *obj* into serialized JSON.
|
|
|
|
"""
|
2011-05-12 18:11:14 +04:00
|
|
|
if obj is None:
|
|
|
|
return ''
|
|
|
|
|
|
|
|
# If the media type looks like 'application/json; indent=4', then
|
|
|
|
# pretty print the result.
|
|
|
|
indent = get_media_type_params(media_type).get('indent', None)
|
|
|
|
sort_keys = False
|
|
|
|
try:
|
|
|
|
indent = max(min(int(indent), 8), 0)
|
|
|
|
sort_keys = True
|
|
|
|
except (ValueError, TypeError):
|
|
|
|
indent = None
|
|
|
|
|
2012-09-03 16:30:20 +04:00
|
|
|
return json.dumps(obj, cls=self.encoder_class, indent=indent, sort_keys=sort_keys)
|
2011-05-12 18:11:14 +04:00
|
|
|
|
|
|
|
|
2011-12-29 17:24:52 +04:00
|
|
|
class JSONPRenderer(JSONRenderer):
|
|
|
|
"""
|
|
|
|
Renderer which serializes to JSONP
|
|
|
|
"""
|
2011-12-29 17:31:12 +04:00
|
|
|
|
2012-09-06 00:26:26 +04:00
|
|
|
media_type = 'application/javascript'
|
|
|
|
format = 'jsonp'
|
2011-12-29 17:24:52 +04:00
|
|
|
renderer_class = JSONRenderer
|
|
|
|
callback_parameter = 'callback'
|
|
|
|
|
|
|
|
def _get_callback(self):
|
|
|
|
return self.view.request.GET.get(self.callback_parameter, self.callback_parameter)
|
|
|
|
|
|
|
|
def _get_renderer(self):
|
|
|
|
return self.renderer_class(self.view)
|
|
|
|
|
|
|
|
def render(self, obj=None, media_type=None):
|
|
|
|
callback = self._get_callback()
|
|
|
|
json = self._get_renderer().render(obj, media_type)
|
|
|
|
return "%s(%s);" % (callback, json)
|
|
|
|
|
|
|
|
|
2011-05-12 18:11:14 +04:00
|
|
|
class XMLRenderer(BaseRenderer):
|
|
|
|
"""
|
|
|
|
Renderer which serializes to XML.
|
|
|
|
"""
|
2011-06-22 02:01:41 +04:00
|
|
|
|
2011-05-12 18:11:14 +04:00
|
|
|
media_type = 'application/xml'
|
2011-06-26 18:03:36 +04:00
|
|
|
format = 'xml'
|
2011-05-12 18:11:14 +04:00
|
|
|
|
|
|
|
def render(self, obj=None, media_type=None):
|
2011-05-24 16:29:30 +04:00
|
|
|
"""
|
|
|
|
Renders *obj* into serialized XML.
|
|
|
|
"""
|
2011-05-12 18:11:14 +04:00
|
|
|
if obj is None:
|
|
|
|
return ''
|
|
|
|
return dict2xml(obj)
|
|
|
|
|
|
|
|
|
2012-02-25 22:45:17 +04:00
|
|
|
class YAMLRenderer(BaseRenderer):
|
|
|
|
"""
|
|
|
|
Renderer which serializes to YAML.
|
|
|
|
"""
|
2011-12-29 17:31:12 +04:00
|
|
|
|
2012-02-25 22:45:17 +04:00
|
|
|
media_type = 'application/yaml'
|
|
|
|
format = 'yaml'
|
2011-12-29 17:31:12 +04:00
|
|
|
|
2012-02-25 22:45:17 +04:00
|
|
|
def render(self, obj=None, media_type=None):
|
|
|
|
"""
|
|
|
|
Renders *obj* into serialized YAML.
|
|
|
|
"""
|
|
|
|
if obj is None:
|
|
|
|
return ''
|
2011-07-01 20:44:08 +04:00
|
|
|
|
2012-02-25 22:45:17 +04:00
|
|
|
return yaml.safe_dump(obj)
|
2011-07-01 20:44:08 +04:00
|
|
|
|
2011-05-12 18:11:14 +04:00
|
|
|
|
2011-04-29 17:32:56 +04:00
|
|
|
class TemplateRenderer(BaseRenderer):
|
2011-05-10 15:21:48 +04:00
|
|
|
"""
|
|
|
|
A Base class provided for convenience.
|
2011-05-10 13:49:28 +04:00
|
|
|
|
2011-05-10 15:21:48 +04:00
|
|
|
Render the object simply by using the given template.
|
2011-05-19 00:13:48 +04:00
|
|
|
To create a template renderer, subclass this class, and set
|
2011-05-24 16:29:30 +04:00
|
|
|
the :attr:`media_type` and :attr:`template` attributes.
|
2011-05-10 15:21:48 +04:00
|
|
|
"""
|
2011-05-24 16:29:30 +04:00
|
|
|
|
2011-01-26 11:58:09 +03:00
|
|
|
media_type = None
|
|
|
|
template = None
|
|
|
|
|
2011-05-10 15:21:48 +04:00
|
|
|
def render(self, obj=None, media_type=None):
|
2011-05-24 16:29:30 +04:00
|
|
|
"""
|
|
|
|
Renders *obj* using the :attr:`template` specified on the class.
|
|
|
|
"""
|
2011-05-10 15:21:48 +04:00
|
|
|
if obj is None:
|
2011-01-26 11:58:09 +03:00
|
|
|
return ''
|
|
|
|
|
2011-05-24 16:29:30 +04:00
|
|
|
template = loader.get_template(self.template)
|
|
|
|
context = RequestContext(self.view.request, {'object': obj})
|
|
|
|
return template.render(context)
|
2011-01-26 23:31:47 +03:00
|
|
|
|
2011-01-24 02:08:16 +03:00
|
|
|
|
2011-04-29 17:32:56 +04:00
|
|
|
class DocumentingTemplateRenderer(BaseRenderer):
|
2011-05-10 15:21:48 +04:00
|
|
|
"""
|
|
|
|
Base class for renderers used to self-document the API.
|
|
|
|
Implementing classes should extend this class and set the template attribute.
|
|
|
|
"""
|
2011-05-24 16:29:30 +04:00
|
|
|
|
2011-01-24 02:08:16 +03:00
|
|
|
template = None
|
|
|
|
|
2011-05-10 15:28:11 +04:00
|
|
|
def _get_content(self, view, request, obj, media_type):
|
2011-05-10 19:01:58 +04:00
|
|
|
"""
|
|
|
|
Get the content as if it had been rendered by a non-documenting renderer.
|
2011-01-26 23:31:47 +03:00
|
|
|
|
|
|
|
(Typically this will be the content as it would have been if the Resource had been
|
2011-05-10 19:01:58 +04:00
|
|
|
requested with an 'Accept: */*' header, although with verbose style formatting if appropriate.)
|
|
|
|
"""
|
2011-01-24 02:08:16 +03:00
|
|
|
|
2011-04-29 17:32:56 +04:00
|
|
|
# Find the first valid renderer and render the content. (Don't use another documenting renderer.)
|
2012-09-11 17:17:26 +04:00
|
|
|
renderers = [renderer for renderer in view.renderer_classes
|
2012-02-25 22:45:17 +04:00
|
|
|
if not issubclass(renderer, DocumentingTemplateRenderer)]
|
2011-04-29 17:32:56 +04:00
|
|
|
if not renderers:
|
|
|
|
return '[No renderers were found]'
|
2011-05-10 15:21:48 +04:00
|
|
|
|
|
|
|
media_type = add_media_type_param(media_type, 'indent', '4')
|
2011-05-10 15:28:11 +04:00
|
|
|
content = renderers[0](view).render(obj, media_type)
|
2011-01-26 23:31:47 +03:00
|
|
|
if not all(char in string.printable for char in content):
|
|
|
|
return '[%d bytes of binary content]'
|
2011-12-29 17:31:12 +04:00
|
|
|
|
2011-01-26 23:31:47 +03:00
|
|
|
return content
|
2011-05-10 19:01:58 +04:00
|
|
|
|
2011-05-27 17:40:19 +04:00
|
|
|
def _get_form_instance(self, view, method):
|
2011-05-10 19:01:58 +04:00
|
|
|
"""
|
|
|
|
Get a form, possibly bound to either the input or output data.
|
2011-01-28 01:30:10 +03:00
|
|
|
In the absence on of the Resource having an associated form then
|
2011-05-10 19:01:58 +04:00
|
|
|
provide a form that can be used to submit arbitrary content.
|
|
|
|
"""
|
2012-09-09 00:50:54 +04:00
|
|
|
if not hasattr(self.view, 'get_serializer'): # No serializer, no form.
|
|
|
|
return
|
2012-09-08 20:18:05 +04:00
|
|
|
# We need to map our Fields to Django's Fields.
|
|
|
|
field_mapping = dict([
|
|
|
|
[FloatField.__name__, forms.FloatField],
|
|
|
|
[IntegerField.__name__, forms.IntegerField],
|
|
|
|
[DateTimeField.__name__, forms.DateTimeField],
|
|
|
|
[DateField.__name__, forms.DateField],
|
|
|
|
[EmailField.__name__, forms.EmailField],
|
|
|
|
[CharField.__name__, forms.CharField],
|
|
|
|
[BooleanField.__name__, forms.BooleanField]
|
|
|
|
])
|
|
|
|
|
|
|
|
# Creating an on the fly form see: http://stackoverflow.com/questions/3915024/dynamically-creating-classes-python
|
|
|
|
fields = {}
|
2012-09-08 23:56:18 +04:00
|
|
|
object, data = None, None
|
|
|
|
if hasattr(self.view, 'object'):
|
|
|
|
object = self.view.object
|
|
|
|
serializer = self.view.get_serializer(instance=object)
|
|
|
|
for k, v in serializer.fields.items():
|
2012-09-08 20:18:05 +04:00
|
|
|
fields[k] = field_mapping[v.__class__.__name__]()
|
|
|
|
OnTheFlyForm = type("OnTheFlyForm", (forms.Form,), fields)
|
2012-09-09 00:01:12 +04:00
|
|
|
if object and not self.view.request.method == 'DELETE': # Don't fill in the form when the object is deleted
|
2012-09-08 23:56:18 +04:00
|
|
|
data = serializer.data
|
|
|
|
form_instance = OnTheFlyForm(data)
|
2011-01-26 23:31:47 +03:00
|
|
|
return form_instance
|
|
|
|
|
2011-05-10 15:28:11 +04:00
|
|
|
def _get_generic_content_form(self, view):
|
2011-05-10 19:01:58 +04:00
|
|
|
"""
|
|
|
|
Returns a form that allows for arbitrary content types to be tunneled via standard HTML forms
|
|
|
|
(Which are typically application/x-www-form-urlencoded)
|
|
|
|
"""
|
2011-01-26 23:31:47 +03:00
|
|
|
|
2011-02-19 13:26:27 +03:00
|
|
|
# If we're not using content overloading there's no point in supplying a generic form,
|
2011-05-10 15:28:11 +04:00
|
|
|
# as the view won't treat the form's value as the content of the request.
|
2012-01-22 23:28:34 +04:00
|
|
|
if not getattr(view.request, '_USE_FORM_OVERLOADING', False):
|
2011-02-19 13:26:27 +03:00
|
|
|
return None
|
|
|
|
|
2011-01-26 23:31:47 +03:00
|
|
|
# NB. http://jacobian.org/writing/dynamic-form-generation/
|
|
|
|
class GenericContentForm(forms.Form):
|
2012-02-25 22:45:17 +04:00
|
|
|
def __init__(self, view, request):
|
2011-01-26 23:31:47 +03:00
|
|
|
"""We don't know the names of the fields we want to set until the point the form is instantiated,
|
|
|
|
as they are determined by the Resource the form is being created against.
|
|
|
|
Add the fields dynamically."""
|
|
|
|
super(GenericContentForm, self).__init__()
|
|
|
|
|
2012-02-25 22:45:17 +04:00
|
|
|
contenttype_choices = [(media_type, media_type) for media_type in view._parsed_media_types]
|
|
|
|
initial_contenttype = view._default_parser.media_type
|
2011-01-26 23:31:47 +03:00
|
|
|
|
2012-01-22 23:28:34 +04:00
|
|
|
self.fields[request._CONTENTTYPE_PARAM] = forms.ChoiceField(label='Content Type',
|
2011-05-12 15:55:13 +04:00
|
|
|
choices=contenttype_choices,
|
|
|
|
initial=initial_contenttype)
|
2012-01-22 23:28:34 +04:00
|
|
|
self.fields[request._CONTENT_PARAM] = forms.CharField(label='Content',
|
2011-05-12 15:55:13 +04:00
|
|
|
widget=forms.Textarea)
|
2011-01-26 23:31:47 +03:00
|
|
|
|
|
|
|
# If either of these reserved parameters are turned off then content tunneling is not possible
|
2012-01-22 23:28:34 +04:00
|
|
|
if self.view.request._CONTENTTYPE_PARAM is None or self.view.request._CONTENT_PARAM is None:
|
2011-01-26 23:31:47 +03:00
|
|
|
return None
|
|
|
|
|
|
|
|
# Okey doke, let's do it
|
2012-02-25 22:45:17 +04:00
|
|
|
return GenericContentForm(view, view.request)
|
2011-01-26 23:31:47 +03:00
|
|
|
|
2012-01-23 22:32:37 +04:00
|
|
|
def get_name(self):
|
|
|
|
try:
|
|
|
|
return self.view.get_name()
|
|
|
|
except AttributeError:
|
|
|
|
return self.view.__doc__
|
|
|
|
|
|
|
|
def get_description(self, html=None):
|
|
|
|
if html is None:
|
|
|
|
html = bool('html' in self.format)
|
|
|
|
try:
|
|
|
|
return self.view.get_description(html)
|
|
|
|
except AttributeError:
|
|
|
|
return self.view.__doc__
|
|
|
|
|
2011-05-10 15:21:48 +04:00
|
|
|
def render(self, obj=None, media_type=None):
|
2011-05-24 16:29:30 +04:00
|
|
|
"""
|
|
|
|
Renders *obj* using the :attr:`template` set on the class.
|
|
|
|
|
|
|
|
The context used in the template contains all the information
|
|
|
|
needed to self-document the response to this request.
|
|
|
|
"""
|
2011-06-22 02:01:41 +04:00
|
|
|
|
2011-05-10 15:21:48 +04:00
|
|
|
content = self._get_content(self.view, self.view.request, obj, media_type)
|
2011-05-27 17:40:19 +04:00
|
|
|
|
|
|
|
put_form_instance = self._get_form_instance(self.view, 'put')
|
|
|
|
post_form_instance = self._get_form_instance(self.view, 'post')
|
2011-01-24 02:08:16 +03:00
|
|
|
|
2012-01-23 22:32:37 +04:00
|
|
|
name = self.get_name()
|
|
|
|
description = self.get_description()
|
2011-02-19 13:26:27 +03:00
|
|
|
|
2011-05-10 13:49:28 +04:00
|
|
|
breadcrumb_list = get_breadcrumbs(self.view.request.path)
|
2011-02-19 13:26:27 +03:00
|
|
|
|
2011-01-24 02:08:16 +03:00
|
|
|
template = loader.get_template(self.template)
|
2011-05-10 13:49:28 +04:00
|
|
|
context = RequestContext(self.view.request, {
|
2011-01-24 02:08:16 +03:00
|
|
|
'content': content,
|
2011-05-12 18:11:14 +04:00
|
|
|
'view': self.view,
|
2012-02-25 22:45:17 +04:00
|
|
|
'request': self.view.request,
|
2011-05-10 13:49:28 +04:00
|
|
|
'response': self.view.response,
|
2011-02-19 13:26:27 +03:00
|
|
|
'description': description,
|
|
|
|
'name': name,
|
2011-06-25 13:25:26 +04:00
|
|
|
'version': VERSION,
|
2011-02-19 13:26:27 +03:00
|
|
|
'breadcrumblist': breadcrumb_list,
|
2012-02-25 22:45:17 +04:00
|
|
|
'allowed_methods': self.view.allowed_methods,
|
2011-06-26 18:03:36 +04:00
|
|
|
'available_formats': self.view._rendered_formats,
|
2011-05-27 17:40:19 +04:00
|
|
|
'put_form': put_form_instance,
|
|
|
|
'post_form': post_form_instance,
|
2011-06-26 18:03:36 +04:00
|
|
|
'FORMAT_PARAM': self._FORMAT_QUERY_PARAM,
|
2012-02-25 22:45:17 +04:00
|
|
|
'METHOD_PARAM': getattr(self.view, '_METHOD_PARAM', None),
|
2012-09-07 13:41:16 +04:00
|
|
|
'api_settings': api_settings
|
2011-01-24 02:08:16 +03:00
|
|
|
})
|
2011-12-29 17:31:12 +04:00
|
|
|
|
2011-01-24 02:08:16 +03:00
|
|
|
ret = template.render(context)
|
|
|
|
|
2011-05-10 15:28:11 +04:00
|
|
|
# Munge DELETE Response code to allow us to return content
|
|
|
|
# (Do this *after* we've rendered the template so that we include
|
|
|
|
# the normal deletion response code in the output)
|
2012-02-02 20:19:44 +04:00
|
|
|
if self.view.response.status_code == 204:
|
|
|
|
self.view.response.status_code = 200
|
2011-05-10 15:28:11 +04:00
|
|
|
|
2011-01-24 02:08:16 +03:00
|
|
|
return ret
|
|
|
|
|
|
|
|
|
2011-04-29 17:32:56 +04:00
|
|
|
class DocumentingHTMLRenderer(DocumentingTemplateRenderer):
|
2011-05-10 15:21:48 +04:00
|
|
|
"""
|
|
|
|
Renderer which provides a browsable HTML interface for an API.
|
|
|
|
See the examples at http://api.django-rest-framework.org to see this in action.
|
|
|
|
"""
|
2011-05-24 16:29:30 +04:00
|
|
|
|
2011-01-24 02:08:16 +03:00
|
|
|
media_type = 'text/html'
|
2011-06-26 18:03:36 +04:00
|
|
|
format = 'html'
|
2012-02-22 00:12:14 +04:00
|
|
|
template = 'djangorestframework/api.html'
|
2011-01-24 02:08:16 +03:00
|
|
|
|
|
|
|
|
2011-04-29 17:32:56 +04:00
|
|
|
class DocumentingXHTMLRenderer(DocumentingTemplateRenderer):
|
2011-05-10 15:21:48 +04:00
|
|
|
"""
|
|
|
|
Identical to DocumentingHTMLRenderer, except with an xhtml media type.
|
2011-01-28 01:30:10 +03:00
|
|
|
We need this to be listed in preference to xml in order to return HTML to WebKit based browsers,
|
2011-05-10 15:21:48 +04:00
|
|
|
given their Accept headers.
|
|
|
|
"""
|
2011-05-24 16:29:30 +04:00
|
|
|
|
2011-01-24 02:08:16 +03:00
|
|
|
media_type = 'application/xhtml+xml'
|
2011-06-26 18:03:36 +04:00
|
|
|
format = 'xhtml'
|
2012-02-22 00:12:14 +04:00
|
|
|
template = 'djangorestframework/api.html'
|
2011-01-24 02:08:16 +03:00
|
|
|
|
|
|
|
|
2011-04-29 17:32:56 +04:00
|
|
|
class DocumentingPlainTextRenderer(DocumentingTemplateRenderer):
|
2011-05-10 15:21:48 +04:00
|
|
|
"""
|
|
|
|
Renderer that serializes the object with the default renderer, but also provides plain-text
|
|
|
|
documentation of the returned status and headers, and of the resource's name and description.
|
|
|
|
Useful for browsing an API with command line tools.
|
|
|
|
"""
|
2011-05-24 16:29:30 +04:00
|
|
|
|
2011-01-24 02:08:16 +03:00
|
|
|
media_type = 'text/plain'
|
2011-06-26 18:03:36 +04:00
|
|
|
format = 'txt'
|
2012-02-22 00:12:14 +04:00
|
|
|
template = 'djangorestframework/api.txt'
|
2011-05-10 13:49:28 +04:00
|
|
|
|
|
|
|
|
2012-01-21 22:33:34 +04:00
|
|
|
DEFAULT_RENDERERS = (
|
|
|
|
JSONRenderer,
|
|
|
|
JSONPRenderer,
|
|
|
|
DocumentingHTMLRenderer,
|
|
|
|
DocumentingXHTMLRenderer,
|
|
|
|
DocumentingPlainTextRenderer,
|
|
|
|
XMLRenderer
|
|
|
|
)
|
2011-01-24 02:08:16 +03:00
|
|
|
|
2012-02-25 22:45:17 +04:00
|
|
|
if yaml:
|
|
|
|
DEFAULT_RENDERERS += (YAMLRenderer, )
|
|
|
|
else:
|
|
|
|
YAMLRenderer = None
|