|
@@ -1,3 +1,4 @@
|
|
|
|
|
+from django.conf import settings
|
|
|
from django.contrib.auth.models import User
|
|
from django.contrib.auth.models import User
|
|
|
from django.contrib.contenttypes.models import ContentType
|
|
from django.contrib.contenttypes.models import ContentType
|
|
|
from django.urls import reverse
|
|
from django.urls import reverse
|
|
@@ -21,7 +22,14 @@ __all__ = (
|
|
|
#
|
|
#
|
|
|
|
|
|
|
|
class APITestCase(ModelTestCase):
|
|
class APITestCase(ModelTestCase):
|
|
|
|
|
+ """
|
|
|
|
|
+ Base test case for API requests.
|
|
|
|
|
+
|
|
|
|
|
+ client_class: Test client class
|
|
|
|
|
+ view_namespace: Namespace for API views. If None, the model's app_label will be used.
|
|
|
|
|
+ """
|
|
|
client_class = APIClient
|
|
client_class = APIClient
|
|
|
|
|
+ view_namespace = None
|
|
|
|
|
|
|
|
def setUp(self):
|
|
def setUp(self):
|
|
|
"""
|
|
"""
|
|
@@ -33,12 +41,15 @@ class APITestCase(ModelTestCase):
|
|
|
self.token = Token.objects.create(user=self.user)
|
|
self.token = Token.objects.create(user=self.user)
|
|
|
self.header = {'HTTP_AUTHORIZATION': 'Token {}'.format(self.token.key)}
|
|
self.header = {'HTTP_AUTHORIZATION': 'Token {}'.format(self.token.key)}
|
|
|
|
|
|
|
|
|
|
+ def _get_view_namespace(self):
|
|
|
|
|
+ return f'{self.view_namespace or self.model._meta.app_label}-api'
|
|
|
|
|
+
|
|
|
def _get_detail_url(self, instance):
|
|
def _get_detail_url(self, instance):
|
|
|
- viewname = f'{instance._meta.app_label}-api:{instance._meta.model_name}-detail'
|
|
|
|
|
|
|
+ viewname = f'{self._get_view_namespace()}:{instance._meta.model_name}-detail'
|
|
|
return reverse(viewname, kwargs={'pk': instance.pk})
|
|
return reverse(viewname, kwargs={'pk': instance.pk})
|
|
|
|
|
|
|
|
def _get_list_url(self):
|
|
def _get_list_url(self):
|
|
|
- viewname = f'{self.model._meta.app_label}-api:{self.model._meta.model_name}-list'
|
|
|
|
|
|
|
+ viewname = f'{self._get_view_namespace()}:{self.model._meta.model_name}-list'
|
|
|
return reverse(viewname)
|
|
return reverse(viewname)
|
|
|
|
|
|
|
|
|
|
|
|
@@ -52,8 +63,13 @@ class APIViewTestCases:
|
|
|
GET a single object as an unauthenticated user.
|
|
GET a single object as an unauthenticated user.
|
|
|
"""
|
|
"""
|
|
|
url = self._get_detail_url(self._get_queryset().first())
|
|
url = self._get_detail_url(self._get_queryset().first())
|
|
|
- response = self.client.get(url, **self.header)
|
|
|
|
|
- self.assertHttpStatus(response, status.HTTP_200_OK)
|
|
|
|
|
|
|
+ if (self.model._meta.app_label, self.model._meta.model_name) in settings.EXEMPT_EXCLUDE_MODELS:
|
|
|
|
|
+ # Models listed in EXEMPT_EXCLUDE_MODELS should not be accessible to anonymous users
|
|
|
|
|
+ with disable_warnings('django.request'):
|
|
|
|
|
+ self.assertHttpStatus(self.client.get(url, **self.header), status.HTTP_403_FORBIDDEN)
|
|
|
|
|
+ else:
|
|
|
|
|
+ response = self.client.get(url, **self.header)
|
|
|
|
|
+ self.assertHttpStatus(response, status.HTTP_200_OK)
|
|
|
|
|
|
|
|
@override_settings(EXEMPT_VIEW_PERMISSIONS=[])
|
|
@override_settings(EXEMPT_VIEW_PERMISSIONS=[])
|
|
|
def test_get_object_without_permission(self):
|
|
def test_get_object_without_permission(self):
|
|
@@ -101,10 +117,14 @@ class APIViewTestCases:
|
|
|
GET a list of objects as an unauthenticated user.
|
|
GET a list of objects as an unauthenticated user.
|
|
|
"""
|
|
"""
|
|
|
url = self._get_list_url()
|
|
url = self._get_list_url()
|
|
|
- response = self.client.get(url, **self.header)
|
|
|
|
|
-
|
|
|
|
|
- self.assertHttpStatus(response, status.HTTP_200_OK)
|
|
|
|
|
- self.assertEqual(len(response.data['results']), self._get_queryset().count())
|
|
|
|
|
|
|
+ if (self.model._meta.app_label, self.model._meta.model_name) in settings.EXEMPT_EXCLUDE_MODELS:
|
|
|
|
|
+ # Models listed in EXEMPT_EXCLUDE_MODELS should not be accessible to anonymous users
|
|
|
|
|
+ with disable_warnings('django.request'):
|
|
|
|
|
+ self.assertHttpStatus(self.client.get(url, **self.header), status.HTTP_403_FORBIDDEN)
|
|
|
|
|
+ else:
|
|
|
|
|
+ response = self.client.get(url, **self.header)
|
|
|
|
|
+ self.assertHttpStatus(response, status.HTTP_200_OK)
|
|
|
|
|
+ self.assertEqual(len(response.data['results']), self._get_queryset().count())
|
|
|
|
|
|
|
|
@override_settings(EXEMPT_VIEW_PERMISSIONS=[])
|
|
@override_settings(EXEMPT_VIEW_PERMISSIONS=[])
|
|
|
def test_list_objects_brief(self):
|
|
def test_list_objects_brief(self):
|