2013-02-05 00:55:35 +04:00
|
|
|
from __future__ import unicode_literals
|
2015-06-11 01:45:23 +03:00
|
|
|
from django.conf.urls import url, include
|
2012-09-20 16:06:27 +04:00
|
|
|
from django.test import TestCase
|
2014-08-19 20:06:55 +04:00
|
|
|
from django.utils import six
|
2014-09-18 14:20:56 +04:00
|
|
|
from tests.models import BasicModel
|
2012-09-20 16:06:27 +04:00
|
|
|
from rest_framework.response import Response
|
|
|
|
from rest_framework.views import APIView
|
2013-05-18 18:44:40 +04:00
|
|
|
from rest_framework import generics
|
|
|
|
from rest_framework import routers
|
2014-09-18 14:20:56 +04:00
|
|
|
from rest_framework import serializers
|
2012-09-20 16:06:27 +04:00
|
|
|
from rest_framework import status
|
|
|
|
from rest_framework.renderers import (
|
|
|
|
BaseRenderer,
|
|
|
|
JSONRenderer,
|
2012-10-09 18:58:48 +04:00
|
|
|
BrowsableAPIRenderer
|
2012-09-20 16:06:27 +04:00
|
|
|
)
|
2013-05-18 18:44:40 +04:00
|
|
|
from rest_framework import viewsets
|
2012-10-02 18:24:42 +04:00
|
|
|
from rest_framework.settings import api_settings
|
2012-09-20 16:06:27 +04:00
|
|
|
|
2013-05-18 20:21:43 +04:00
|
|
|
|
2014-09-18 14:20:56 +04:00
|
|
|
# Serializer used to test BasicModel
|
|
|
|
class BasicModelSerializer(serializers.ModelSerializer):
|
|
|
|
class Meta:
|
|
|
|
model = BasicModel
|
|
|
|
|
|
|
|
|
2012-09-20 16:06:27 +04:00
|
|
|
class MockPickleRenderer(BaseRenderer):
|
|
|
|
media_type = 'application/pickle'
|
|
|
|
|
|
|
|
|
|
|
|
class MockJsonRenderer(BaseRenderer):
|
|
|
|
media_type = 'application/json'
|
|
|
|
|
2013-05-18 20:21:43 +04:00
|
|
|
|
2013-05-18 18:45:05 +04:00
|
|
|
class MockTextMediaRenderer(BaseRenderer):
|
|
|
|
media_type = 'text/html'
|
2012-09-20 16:06:27 +04:00
|
|
|
|
|
|
|
DUMMYSTATUS = status.HTTP_200_OK
|
|
|
|
DUMMYCONTENT = 'dummycontent'
|
|
|
|
|
2015-02-09 20:43:20 +03:00
|
|
|
|
|
|
|
def RENDERER_A_SERIALIZER(x):
|
|
|
|
return ('Renderer A: %s' % x).encode('ascii')
|
|
|
|
|
|
|
|
|
|
|
|
def RENDERER_B_SERIALIZER(x):
|
|
|
|
return ('Renderer B: %s' % x).encode('ascii')
|
2012-09-20 16:06:27 +04:00
|
|
|
|
|
|
|
|
|
|
|
class RendererA(BaseRenderer):
|
|
|
|
media_type = 'mock/renderera'
|
|
|
|
format = "formata"
|
|
|
|
|
2012-10-10 15:15:18 +04:00
|
|
|
def render(self, data, media_type=None, renderer_context=None):
|
|
|
|
return RENDERER_A_SERIALIZER(data)
|
2012-09-20 16:06:27 +04:00
|
|
|
|
|
|
|
|
|
|
|
class RendererB(BaseRenderer):
|
|
|
|
media_type = 'mock/rendererb'
|
|
|
|
format = "formatb"
|
|
|
|
|
2012-10-10 15:15:18 +04:00
|
|
|
def render(self, data, media_type=None, renderer_context=None):
|
|
|
|
return RENDERER_B_SERIALIZER(data)
|
2012-09-20 16:06:27 +04:00
|
|
|
|
2013-05-18 20:21:43 +04:00
|
|
|
|
2013-05-18 18:45:05 +04:00
|
|
|
class RendererC(RendererB):
|
|
|
|
media_type = 'mock/rendererc'
|
|
|
|
format = 'formatc'
|
|
|
|
charset = "rendererc"
|
|
|
|
|
2012-09-20 16:06:27 +04:00
|
|
|
|
|
|
|
class MockView(APIView):
|
2013-05-18 18:45:05 +04:00
|
|
|
renderer_classes = (RendererA, RendererB, RendererC)
|
2012-09-20 16:06:27 +04:00
|
|
|
|
|
|
|
def get(self, request, **kwargs):
|
|
|
|
return Response(DUMMYCONTENT, status=DUMMYSTATUS)
|
|
|
|
|
2013-05-18 20:21:43 +04:00
|
|
|
|
2013-05-21 00:18:17 +04:00
|
|
|
class MockViewSettingContentType(APIView):
|
2013-05-18 20:21:43 +04:00
|
|
|
renderer_classes = (RendererA, RendererB, RendererC)
|
|
|
|
|
|
|
|
def get(self, request, **kwargs):
|
2013-05-21 00:18:17 +04:00
|
|
|
return Response(DUMMYCONTENT, status=DUMMYSTATUS, content_type='setbyview')
|
2013-05-18 20:21:43 +04:00
|
|
|
|
|
|
|
|
2012-09-20 16:06:27 +04:00
|
|
|
class HTMLView(APIView):
|
2012-10-09 18:58:48 +04:00
|
|
|
renderer_classes = (BrowsableAPIRenderer, )
|
2012-09-20 16:06:27 +04:00
|
|
|
|
|
|
|
def get(self, request, **kwargs):
|
|
|
|
return Response('text')
|
|
|
|
|
|
|
|
|
|
|
|
class HTMLView1(APIView):
|
2012-10-09 18:58:48 +04:00
|
|
|
renderer_classes = (BrowsableAPIRenderer, JSONRenderer)
|
2012-09-20 16:06:27 +04:00
|
|
|
|
|
|
|
def get(self, request, **kwargs):
|
|
|
|
return Response('text')
|
|
|
|
|
|
|
|
|
2013-05-18 18:44:40 +04:00
|
|
|
class HTMLNewModelViewSet(viewsets.ModelViewSet):
|
2014-08-29 15:35:53 +04:00
|
|
|
serializer_class = BasicModelSerializer
|
|
|
|
queryset = BasicModel.objects.all()
|
2013-05-18 18:44:40 +04:00
|
|
|
|
|
|
|
|
|
|
|
class HTMLNewModelView(generics.ListCreateAPIView):
|
|
|
|
renderer_classes = (BrowsableAPIRenderer,)
|
|
|
|
permission_classes = []
|
|
|
|
serializer_class = BasicModelSerializer
|
2014-08-29 15:35:53 +04:00
|
|
|
queryset = BasicModel.objects.all()
|
2013-05-18 18:44:40 +04:00
|
|
|
|
|
|
|
|
|
|
|
new_model_viewset_router = routers.DefaultRouter()
|
|
|
|
new_model_viewset_router.register(r'', HTMLNewModelViewSet)
|
|
|
|
|
2013-05-21 15:01:56 +04:00
|
|
|
|
2015-06-11 01:45:23 +03:00
|
|
|
urlpatterns = [
|
2013-05-21 00:18:17 +04:00
|
|
|
url(r'^setbyview$', MockViewSettingContentType.as_view(renderer_classes=[RendererA, RendererB, RendererC])),
|
2013-05-18 18:45:05 +04:00
|
|
|
url(r'^.*\.(?P<format>.+)$', MockView.as_view(renderer_classes=[RendererA, RendererB, RendererC])),
|
|
|
|
url(r'^$', MockView.as_view(renderer_classes=[RendererA, RendererB, RendererC])),
|
2012-09-20 16:06:27 +04:00
|
|
|
url(r'^html$', HTMLView.as_view()),
|
|
|
|
url(r'^html1$', HTMLView1.as_view()),
|
2013-05-18 18:44:40 +04:00
|
|
|
url(r'^html_new_model$', HTMLNewModelView.as_view()),
|
|
|
|
url(r'^html_new_model_viewset', include(new_model_viewset_router.urls)),
|
2012-09-20 16:06:27 +04:00
|
|
|
url(r'^restframework', include('rest_framework.urls', namespace='rest_framework'))
|
2015-06-11 01:45:23 +03:00
|
|
|
]
|
2012-09-20 16:06:27 +04:00
|
|
|
|
|
|
|
|
|
|
|
# TODO: Clean tests bellow - remove duplicates with above, better unit testing, ...
|
|
|
|
class RendererIntegrationTests(TestCase):
|
|
|
|
"""
|
|
|
|
End-to-end testing of renderers using an ResponseMixin on a generic view.
|
|
|
|
"""
|
|
|
|
|
2014-03-02 15:40:30 +04:00
|
|
|
urls = 'tests.test_response'
|
2012-09-20 16:06:27 +04:00
|
|
|
|
|
|
|
def test_default_renderer_serializes_content(self):
|
|
|
|
"""If the Accept header is not set the default renderer should serialize the response."""
|
|
|
|
resp = self.client.get('/')
|
2013-05-21 00:00:56 +04:00
|
|
|
self.assertEqual(resp['Content-Type'], RendererA.media_type + '; charset=utf-8')
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(resp.content, RENDERER_A_SERIALIZER(DUMMYCONTENT))
|
|
|
|
self.assertEqual(resp.status_code, DUMMYSTATUS)
|
2012-09-20 16:06:27 +04:00
|
|
|
|
|
|
|
def test_head_method_serializes_no_content(self):
|
|
|
|
"""No response must be included in HEAD requests."""
|
|
|
|
resp = self.client.head('/')
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(resp.status_code, DUMMYSTATUS)
|
2013-05-21 00:00:56 +04:00
|
|
|
self.assertEqual(resp['Content-Type'], RendererA.media_type + '; charset=utf-8')
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(resp.content, six.b(''))
|
2012-09-20 16:06:27 +04:00
|
|
|
|
|
|
|
def test_default_renderer_serializes_content_on_accept_any(self):
|
|
|
|
"""If the Accept header is set to */* the default renderer should serialize the response."""
|
|
|
|
resp = self.client.get('/', HTTP_ACCEPT='*/*')
|
2013-05-21 00:00:56 +04:00
|
|
|
self.assertEqual(resp['Content-Type'], RendererA.media_type + '; charset=utf-8')
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(resp.content, RENDERER_A_SERIALIZER(DUMMYCONTENT))
|
|
|
|
self.assertEqual(resp.status_code, DUMMYSTATUS)
|
2012-09-20 16:06:27 +04:00
|
|
|
|
|
|
|
def test_specified_renderer_serializes_content_default_case(self):
|
|
|
|
"""If the Accept header is set the specified renderer should serialize the response.
|
|
|
|
(In this case we check that works for the default renderer)"""
|
|
|
|
resp = self.client.get('/', HTTP_ACCEPT=RendererA.media_type)
|
2013-05-21 00:00:56 +04:00
|
|
|
self.assertEqual(resp['Content-Type'], RendererA.media_type + '; charset=utf-8')
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(resp.content, RENDERER_A_SERIALIZER(DUMMYCONTENT))
|
|
|
|
self.assertEqual(resp.status_code, DUMMYSTATUS)
|
2012-09-20 16:06:27 +04:00
|
|
|
|
|
|
|
def test_specified_renderer_serializes_content_non_default_case(self):
|
|
|
|
"""If the Accept header is set the specified renderer should serialize the response.
|
|
|
|
(In this case we check that works for a non-default renderer)"""
|
|
|
|
resp = self.client.get('/', HTTP_ACCEPT=RendererB.media_type)
|
2013-05-21 00:00:56 +04:00
|
|
|
self.assertEqual(resp['Content-Type'], RendererB.media_type + '; charset=utf-8')
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(resp.content, RENDERER_B_SERIALIZER(DUMMYCONTENT))
|
|
|
|
self.assertEqual(resp.status_code, DUMMYSTATUS)
|
2012-09-20 16:06:27 +04:00
|
|
|
|
|
|
|
def test_specified_renderer_serializes_content_on_accept_query(self):
|
|
|
|
"""The '_accept' query string should behave in the same way as the Accept header."""
|
2012-10-02 18:24:42 +04:00
|
|
|
param = '?%s=%s' % (
|
|
|
|
api_settings.URL_ACCEPT_OVERRIDE,
|
|
|
|
RendererB.media_type
|
|
|
|
)
|
|
|
|
resp = self.client.get('/' + param)
|
2013-05-21 00:00:56 +04:00
|
|
|
self.assertEqual(resp['Content-Type'], RendererB.media_type + '; charset=utf-8')
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(resp.content, RENDERER_B_SERIALIZER(DUMMYCONTENT))
|
|
|
|
self.assertEqual(resp.status_code, DUMMYSTATUS)
|
2012-09-20 16:06:27 +04:00
|
|
|
|
|
|
|
def test_specified_renderer_serializes_content_on_format_query(self):
|
|
|
|
"""If a 'format' query is specified, the renderer with the matching
|
|
|
|
format attribute should serialize the response."""
|
|
|
|
resp = self.client.get('/?format=%s' % RendererB.format)
|
2013-05-21 00:00:56 +04:00
|
|
|
self.assertEqual(resp['Content-Type'], RendererB.media_type + '; charset=utf-8')
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(resp.content, RENDERER_B_SERIALIZER(DUMMYCONTENT))
|
|
|
|
self.assertEqual(resp.status_code, DUMMYSTATUS)
|
2012-09-20 16:06:27 +04:00
|
|
|
|
|
|
|
def test_specified_renderer_serializes_content_on_format_kwargs(self):
|
|
|
|
"""If a 'format' keyword arg is specified, the renderer with the matching
|
|
|
|
format attribute should serialize the response."""
|
|
|
|
resp = self.client.get('/something.formatb')
|
2013-05-21 00:00:56 +04:00
|
|
|
self.assertEqual(resp['Content-Type'], RendererB.media_type + '; charset=utf-8')
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(resp.content, RENDERER_B_SERIALIZER(DUMMYCONTENT))
|
|
|
|
self.assertEqual(resp.status_code, DUMMYSTATUS)
|
2012-09-20 16:06:27 +04:00
|
|
|
|
|
|
|
def test_specified_renderer_is_used_on_format_query_with_matching_accept(self):
|
|
|
|
"""If both a 'format' query and a matching Accept header specified,
|
|
|
|
the renderer with the matching format attribute should serialize the response."""
|
|
|
|
resp = self.client.get('/?format=%s' % RendererB.format,
|
|
|
|
HTTP_ACCEPT=RendererB.media_type)
|
2013-05-21 00:00:56 +04:00
|
|
|
self.assertEqual(resp['Content-Type'], RendererB.media_type + '; charset=utf-8')
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(resp.content, RENDERER_B_SERIALIZER(DUMMYCONTENT))
|
|
|
|
self.assertEqual(resp.status_code, DUMMYSTATUS)
|
2012-09-20 16:06:27 +04:00
|
|
|
|
|
|
|
|
|
|
|
class Issue122Tests(TestCase):
|
|
|
|
"""
|
|
|
|
Tests that covers #122.
|
|
|
|
"""
|
2014-03-02 15:40:30 +04:00
|
|
|
urls = 'tests.test_response'
|
2012-09-20 16:06:27 +04:00
|
|
|
|
|
|
|
def test_only_html_renderer(self):
|
|
|
|
"""
|
|
|
|
Test if no infinite recursion occurs.
|
|
|
|
"""
|
|
|
|
self.client.get('/html')
|
|
|
|
|
|
|
|
def test_html_renderer_is_first(self):
|
|
|
|
"""
|
|
|
|
Test if no infinite recursion occurs.
|
|
|
|
"""
|
|
|
|
self.client.get('/html1')
|
2013-05-18 18:45:05 +04:00
|
|
|
|
2013-05-18 20:21:43 +04:00
|
|
|
|
2013-05-18 18:44:40 +04:00
|
|
|
class Issue467Tests(TestCase):
|
|
|
|
"""
|
|
|
|
Tests for #467
|
|
|
|
"""
|
|
|
|
|
2014-03-02 15:40:30 +04:00
|
|
|
urls = 'tests.test_response'
|
2013-05-18 18:44:40 +04:00
|
|
|
|
2013-05-21 15:01:56 +04:00
|
|
|
def test_form_has_label_and_help_text(self):
|
|
|
|
resp = self.client.get('/html_new_model')
|
|
|
|
self.assertEqual(resp['Content-Type'], 'text/html; charset=utf-8')
|
2014-09-08 17:24:05 +04:00
|
|
|
# self.assertContains(resp, 'Text comes here')
|
|
|
|
# self.assertContains(resp, 'Text description.')
|
2013-05-21 15:01:56 +04:00
|
|
|
|
|
|
|
|
|
|
|
class Issue807Tests(TestCase):
|
2013-05-18 18:45:05 +04:00
|
|
|
"""
|
|
|
|
Covers #807
|
|
|
|
"""
|
2013-05-18 20:21:43 +04:00
|
|
|
|
2014-03-02 15:40:30 +04:00
|
|
|
urls = 'tests.test_response'
|
2013-05-18 20:21:43 +04:00
|
|
|
|
2013-05-18 18:45:05 +04:00
|
|
|
def test_does_not_append_charset_by_default(self):
|
|
|
|
"""
|
2013-05-18 20:21:43 +04:00
|
|
|
Renderers don't include a charset unless set explicitly.
|
2013-05-18 18:45:05 +04:00
|
|
|
"""
|
|
|
|
headers = {"HTTP_ACCEPT": RendererA.media_type}
|
|
|
|
resp = self.client.get('/', **headers)
|
2013-05-21 00:00:56 +04:00
|
|
|
expected = "{0}; charset={1}".format(RendererA.media_type, 'utf-8')
|
|
|
|
self.assertEqual(expected, resp['Content-Type'])
|
2013-05-18 20:21:43 +04:00
|
|
|
|
2013-05-18 18:45:05 +04:00
|
|
|
def test_if_there_is_charset_specified_on_renderer_it_gets_appended(self):
|
|
|
|
"""
|
|
|
|
If renderer class has charset attribute declared, it gets appended
|
|
|
|
to Response's Content-Type
|
|
|
|
"""
|
2013-05-18 20:21:43 +04:00
|
|
|
headers = {"HTTP_ACCEPT": RendererC.media_type}
|
|
|
|
resp = self.client.get('/', **headers)
|
2013-05-18 18:45:05 +04:00
|
|
|
expected = "{0}; charset={1}".format(RendererC.media_type, RendererC.charset)
|
2013-05-18 20:21:43 +04:00
|
|
|
self.assertEqual(expected, resp['Content-Type'])
|
|
|
|
|
2014-10-03 00:00:33 +04:00
|
|
|
def test_content_type_set_explicitly_on_response(self):
|
2013-05-18 18:45:05 +04:00
|
|
|
"""
|
2014-10-03 00:00:33 +04:00
|
|
|
The content type may be set explicitly on the response.
|
2013-05-18 18:45:05 +04:00
|
|
|
"""
|
2013-05-18 20:21:43 +04:00
|
|
|
headers = {"HTTP_ACCEPT": RendererC.media_type}
|
|
|
|
resp = self.client.get('/setbyview', **headers)
|
2013-05-21 00:18:17 +04:00
|
|
|
self.assertEqual('setbyview', resp['Content-Type'])
|
2013-05-21 15:01:56 +04:00
|
|
|
|
2013-05-18 18:44:40 +04:00
|
|
|
def test_viewset_label_help_text(self):
|
|
|
|
param = '?%s=%s' % (
|
|
|
|
api_settings.URL_ACCEPT_OVERRIDE,
|
|
|
|
'text/html'
|
|
|
|
)
|
|
|
|
resp = self.client.get('/html_new_model_viewset/' + param)
|
2013-05-21 15:01:56 +04:00
|
|
|
self.assertEqual(resp['Content-Type'], 'text/html; charset=utf-8')
|
2014-09-08 17:24:05 +04:00
|
|
|
# self.assertContains(resp, 'Text comes here')
|
|
|
|
# self.assertContains(resp, 'Text description.')
|
2013-05-18 18:44:40 +04:00
|
|
|
|
|
|
|
def test_form_has_label_and_help_text(self):
|
|
|
|
resp = self.client.get('/html_new_model')
|
2013-05-21 15:01:56 +04:00
|
|
|
self.assertEqual(resp['Content-Type'], 'text/html; charset=utf-8')
|
2014-09-08 17:24:05 +04:00
|
|
|
# self.assertContains(resp, 'Text comes here')
|
|
|
|
# self.assertContains(resp, 'Text description.')
|