2013-02-05 00:55:35 +04:00
|
|
|
from __future__ import unicode_literals
|
2015-06-25 23:55:51 +03:00
|
|
|
|
2018-01-25 11:40:49 +03:00
|
|
|
import pytest
|
2012-09-14 19:07:07 +04:00
|
|
|
from django.test import TestCase
|
2015-06-25 23:55:51 +03:00
|
|
|
|
2012-09-27 00:47:19 +04:00
|
|
|
from rest_framework import status
|
2013-06-28 20:17:39 +04:00
|
|
|
from rest_framework.authentication import BasicAuthentication
|
2015-06-25 23:55:51 +03:00
|
|
|
from rest_framework.decorators import (
|
2018-01-25 11:40:49 +03:00
|
|
|
action, api_view, authentication_classes, detail_route, list_route,
|
|
|
|
parser_classes, permission_classes, renderer_classes, schema,
|
|
|
|
throttle_classes
|
2015-06-25 23:55:51 +03:00
|
|
|
)
|
2013-06-28 20:17:39 +04:00
|
|
|
from rest_framework.parsers import JSONParser
|
|
|
|
from rest_framework.permissions import IsAuthenticated
|
2012-09-26 16:52:29 +04:00
|
|
|
from rest_framework.renderers import JSONRenderer
|
2015-06-25 23:55:51 +03:00
|
|
|
from rest_framework.response import Response
|
2017-09-14 11:46:34 +03:00
|
|
|
from rest_framework.schemas import AutoSchema
|
2013-06-28 20:17:39 +04:00
|
|
|
from rest_framework.test import APIRequestFactory
|
2012-09-27 00:47:19 +04:00
|
|
|
from rest_framework.throttling import UserRateThrottle
|
2012-09-26 16:52:29 +04:00
|
|
|
from rest_framework.views import APIView
|
2012-09-14 19:07:07 +04:00
|
|
|
|
|
|
|
|
|
|
|
class DecoratorTestCase(TestCase):
|
|
|
|
|
|
|
|
def setUp(self):
|
2013-06-28 20:17:39 +04:00
|
|
|
self.factory = APIRequestFactory()
|
2012-09-14 19:07:07 +04:00
|
|
|
|
2012-09-26 16:52:29 +04:00
|
|
|
def _finalize_response(self, request, response, *args, **kwargs):
|
|
|
|
response.request = request
|
|
|
|
return APIView.finalize_response(self, request, response, *args, **kwargs)
|
|
|
|
|
2013-01-19 19:51:14 +04:00
|
|
|
def test_api_view_incorrect(self):
|
|
|
|
"""
|
|
|
|
If @api_view is not applied correct, we should raise an assertion.
|
|
|
|
"""
|
|
|
|
|
|
|
|
@api_view
|
|
|
|
def view(request):
|
|
|
|
return Response()
|
|
|
|
|
|
|
|
request = self.factory.get('/')
|
|
|
|
self.assertRaises(AssertionError, view, request)
|
|
|
|
|
|
|
|
def test_api_view_incorrect_arguments(self):
|
|
|
|
"""
|
|
|
|
If @api_view is missing arguments, we should raise an assertion.
|
|
|
|
"""
|
|
|
|
|
|
|
|
with self.assertRaises(AssertionError):
|
|
|
|
@api_view('GET')
|
|
|
|
def view(request):
|
|
|
|
return Response()
|
|
|
|
|
2012-09-14 19:07:07 +04:00
|
|
|
def test_calling_method(self):
|
|
|
|
|
|
|
|
@api_view(['GET'])
|
|
|
|
def view(request):
|
|
|
|
return Response({})
|
|
|
|
|
|
|
|
request = self.factory.get('/')
|
|
|
|
response = view(request)
|
2016-11-30 13:24:48 +03:00
|
|
|
assert response.status_code == status.HTTP_200_OK
|
2012-09-14 19:07:07 +04:00
|
|
|
|
|
|
|
request = self.factory.post('/')
|
|
|
|
response = view(request)
|
2016-11-30 13:24:48 +03:00
|
|
|
assert response.status_code == status.HTTP_405_METHOD_NOT_ALLOWED
|
2012-10-05 18:55:45 +04:00
|
|
|
|
|
|
|
def test_calling_put_method(self):
|
|
|
|
|
|
|
|
@api_view(['GET', 'PUT'])
|
|
|
|
def view(request):
|
|
|
|
return Response({})
|
|
|
|
|
|
|
|
request = self.factory.put('/')
|
|
|
|
response = view(request)
|
2016-11-30 13:24:48 +03:00
|
|
|
assert response.status_code == status.HTTP_200_OK
|
2012-10-05 18:55:45 +04:00
|
|
|
|
|
|
|
request = self.factory.post('/')
|
|
|
|
response = view(request)
|
2016-11-30 13:24:48 +03:00
|
|
|
assert response.status_code == status.HTTP_405_METHOD_NOT_ALLOWED
|
2012-09-14 19:07:07 +04:00
|
|
|
|
2012-12-20 09:28:01 +04:00
|
|
|
def test_calling_patch_method(self):
|
2012-12-16 23:49:18 +04:00
|
|
|
|
2012-12-20 09:28:01 +04:00
|
|
|
@api_view(['GET', 'PATCH'])
|
|
|
|
def view(request):
|
|
|
|
return Response({})
|
2012-12-16 23:49:18 +04:00
|
|
|
|
2012-12-20 09:28:01 +04:00
|
|
|
request = self.factory.patch('/')
|
|
|
|
response = view(request)
|
2016-11-30 13:24:48 +03:00
|
|
|
assert response.status_code == status.HTTP_200_OK
|
2012-12-16 23:49:18 +04:00
|
|
|
|
2012-12-20 09:28:01 +04:00
|
|
|
request = self.factory.post('/')
|
|
|
|
response = view(request)
|
2016-11-30 13:24:48 +03:00
|
|
|
assert response.status_code == status.HTTP_405_METHOD_NOT_ALLOWED
|
2012-12-16 23:49:18 +04:00
|
|
|
|
2012-09-14 19:07:07 +04:00
|
|
|
def test_renderer_classes(self):
|
|
|
|
|
|
|
|
@api_view(['GET'])
|
2012-09-26 16:52:29 +04:00
|
|
|
@renderer_classes([JSONRenderer])
|
2012-09-14 19:07:07 +04:00
|
|
|
def view(request):
|
|
|
|
return Response({})
|
|
|
|
|
|
|
|
request = self.factory.get('/')
|
|
|
|
response = view(request)
|
2016-11-30 13:24:48 +03:00
|
|
|
assert isinstance(response.accepted_renderer, JSONRenderer)
|
2012-09-14 19:07:07 +04:00
|
|
|
|
|
|
|
def test_parser_classes(self):
|
|
|
|
|
|
|
|
@api_view(['GET'])
|
2012-09-26 16:52:29 +04:00
|
|
|
@parser_classes([JSONParser])
|
2012-09-14 19:07:07 +04:00
|
|
|
def view(request):
|
2016-11-30 13:24:48 +03:00
|
|
|
assert len(request.parsers) == 1
|
|
|
|
assert isinstance(request.parsers[0], JSONParser)
|
2012-09-14 19:07:07 +04:00
|
|
|
return Response({})
|
|
|
|
|
|
|
|
request = self.factory.get('/')
|
2012-09-26 16:52:29 +04:00
|
|
|
view(request)
|
2012-09-14 19:07:07 +04:00
|
|
|
|
|
|
|
def test_authentication_classes(self):
|
|
|
|
|
2012-09-26 16:52:29 +04:00
|
|
|
@api_view(['GET'])
|
2012-09-14 19:07:07 +04:00
|
|
|
@authentication_classes([BasicAuthentication])
|
2012-09-26 16:52:29 +04:00
|
|
|
def view(request):
|
2016-11-30 13:24:48 +03:00
|
|
|
assert len(request.authenticators) == 1
|
|
|
|
assert isinstance(request.authenticators[0], BasicAuthentication)
|
2012-09-26 16:52:29 +04:00
|
|
|
return Response({})
|
|
|
|
|
|
|
|
request = self.factory.get('/')
|
|
|
|
view(request)
|
|
|
|
|
|
|
|
def test_permission_classes(self):
|
|
|
|
|
2012-09-14 19:07:07 +04:00
|
|
|
@api_view(['GET'])
|
2012-09-26 16:52:29 +04:00
|
|
|
@permission_classes([IsAuthenticated])
|
2012-09-14 19:07:07 +04:00
|
|
|
def view(request):
|
|
|
|
return Response({})
|
|
|
|
|
|
|
|
request = self.factory.get('/')
|
2012-09-27 00:47:19 +04:00
|
|
|
response = view(request)
|
2016-11-30 13:24:48 +03:00
|
|
|
assert response.status_code == status.HTTP_403_FORBIDDEN
|
2012-09-14 19:07:07 +04:00
|
|
|
|
2012-09-27 00:47:19 +04:00
|
|
|
def test_throttle_classes(self):
|
|
|
|
class OncePerDayUserThrottle(UserRateThrottle):
|
|
|
|
rate = '1/day'
|
2012-09-14 19:07:07 +04:00
|
|
|
|
2012-09-27 00:47:19 +04:00
|
|
|
@api_view(['GET'])
|
|
|
|
@throttle_classes([OncePerDayUserThrottle])
|
|
|
|
def view(request):
|
|
|
|
return Response({})
|
2012-09-14 19:07:07 +04:00
|
|
|
|
2012-09-27 00:47:19 +04:00
|
|
|
request = self.factory.get('/')
|
|
|
|
response = view(request)
|
2016-11-30 13:24:48 +03:00
|
|
|
assert response.status_code == status.HTTP_200_OK
|
2012-09-14 19:07:07 +04:00
|
|
|
|
2012-09-27 00:47:19 +04:00
|
|
|
response = view(request)
|
2016-11-30 13:24:48 +03:00
|
|
|
assert response.status_code == status.HTTP_429_TOO_MANY_REQUESTS
|
2017-09-14 11:46:34 +03:00
|
|
|
|
|
|
|
def test_schema(self):
|
|
|
|
"""
|
|
|
|
Checks CustomSchema class is set on view
|
|
|
|
"""
|
|
|
|
class CustomSchema(AutoSchema):
|
|
|
|
pass
|
|
|
|
|
|
|
|
@api_view(['GET'])
|
|
|
|
@schema(CustomSchema())
|
|
|
|
def view(request):
|
|
|
|
return Response({})
|
|
|
|
|
|
|
|
assert isinstance(view.cls.schema, CustomSchema)
|
2018-01-25 11:40:49 +03:00
|
|
|
|
|
|
|
|
|
|
|
class ActionDecoratorTestCase(TestCase):
|
|
|
|
|
|
|
|
def test_defaults(self):
|
|
|
|
@action(detail=True)
|
|
|
|
def test_action(request):
|
|
|
|
pass
|
|
|
|
|
|
|
|
assert test_action.bind_to_methods == ['get']
|
|
|
|
assert test_action.detail is True
|
|
|
|
assert test_action.url_path == 'test_action'
|
|
|
|
assert test_action.url_name == 'test-action'
|
|
|
|
|
|
|
|
def test_detail_required(self):
|
|
|
|
with pytest.raises(AssertionError) as excinfo:
|
|
|
|
@action()
|
|
|
|
def test_action(request):
|
|
|
|
pass
|
|
|
|
|
|
|
|
assert str(excinfo.value) == "@action() missing required argument: 'detail'"
|
|
|
|
|
|
|
|
def test_detail_route_deprecation(self):
|
|
|
|
with pytest.warns(PendingDeprecationWarning) as record:
|
|
|
|
@detail_route()
|
|
|
|
def view(request):
|
|
|
|
pass
|
|
|
|
|
|
|
|
assert len(record) == 1
|
|
|
|
assert str(record[0].message) == (
|
|
|
|
"`detail_route` is pending deprecation and will be removed in "
|
|
|
|
"3.10 in favor of `action`, which accepts a `detail` bool. Use "
|
|
|
|
"`@action(detail=True)` instead."
|
|
|
|
)
|
|
|
|
|
|
|
|
def test_list_route_deprecation(self):
|
|
|
|
with pytest.warns(PendingDeprecationWarning) as record:
|
|
|
|
@list_route()
|
|
|
|
def view(request):
|
|
|
|
pass
|
|
|
|
|
|
|
|
assert len(record) == 1
|
|
|
|
assert str(record[0].message) == (
|
|
|
|
"`list_route` is pending deprecation and will be removed in "
|
|
|
|
"3.10 in favor of `action`, which accepts a `detail` bool. Use "
|
|
|
|
"`@action(detail=False)` instead."
|
|
|
|
)
|