Refactor APITestCase to provide dynamic queryset specification

This commit is contained in:
Jeremy Stretch 2020-07-07 13:13:19 -04:00
parent a2d957ba0d
commit 2fbe138c71
2 changed files with 34 additions and 34 deletions

View File

@ -7,7 +7,7 @@ from rest_framework.test import APIClient
from users.models import ObjectPermission, Token from users.models import ObjectPermission, Token
from .utils import disable_warnings from .utils import disable_warnings
from .views import TestCase from .views import ModelTestCase
__all__ = ( __all__ = (
@ -20,9 +20,8 @@ __all__ = (
# REST API Tests # REST API Tests
# #
class APITestCase(TestCase): class APITestCase(ModelTestCase):
client_class = APIClient client_class = APIClient
model = None
def setUp(self): def setUp(self):
""" """
@ -52,7 +51,7 @@ class APIViewTestCases:
""" """
GET a single object as an unauthenticated user. GET a single object as an unauthenticated user.
""" """
url = self._get_detail_url(self.model.objects.unrestricted().first()) url = self._get_detail_url(self._get_queryset().first())
response = self.client.get(url, **self.header) response = self.client.get(url, **self.header)
self.assertHttpStatus(response, status.HTTP_200_OK) self.assertHttpStatus(response, status.HTTP_200_OK)
@ -61,7 +60,7 @@ class APIViewTestCases:
""" """
GET a single object as an authenticated user without the required permission. GET a single object as an authenticated user without the required permission.
""" """
url = self._get_detail_url(self.model.objects.unrestricted().first()) url = self._get_detail_url(self._get_queryset().first())
# Try GET without permission # Try GET without permission
with disable_warnings('django.request'): with disable_warnings('django.request'):
@ -72,9 +71,9 @@ class APIViewTestCases:
""" """
GET a single object as an authenticated user with permission to view the object. GET a single object as an authenticated user with permission to view the object.
""" """
self.assertGreaterEqual(self.model.objects.unrestricted().count(), 2, self.assertGreaterEqual(self._get_queryset().count(), 2,
f"Test requires the creation of at least two {self.model} instances") f"Test requires the creation of at least two {self.model} instances")
instance1, instance2 = self.model.objects.unrestricted()[:2] instance1, instance2 = self._get_queryset()[:2]
# Add object-level permission # Add object-level permission
obj_perm = ObjectPermission( obj_perm = ObjectPermission(
@ -104,7 +103,7 @@ class APIViewTestCases:
url = self._get_list_url() url = self._get_list_url()
response = self.client.get(url, **self.header) response = self.client.get(url, **self.header)
self.assertEqual(len(response.data['results']), self.model.objects.unrestricted().count()) self.assertEqual(len(response.data['results']), self._get_queryset().count())
self.assertHttpStatus(response, status.HTTP_200_OK) self.assertHttpStatus(response, status.HTTP_200_OK)
@override_settings(EXEMPT_VIEW_PERMISSIONS=['*']) @override_settings(EXEMPT_VIEW_PERMISSIONS=['*'])
@ -115,7 +114,7 @@ class APIViewTestCases:
url = f'{self._get_list_url()}?brief=1' url = f'{self._get_list_url()}?brief=1'
response = self.client.get(url, **self.header) response = self.client.get(url, **self.header)
self.assertEqual(len(response.data['results']), self.model.objects.unrestricted().count()) self.assertEqual(len(response.data['results']), self._get_queryset().count())
self.assertEqual(sorted(response.data['results'][0]), self.brief_fields) self.assertEqual(sorted(response.data['results'][0]), self.brief_fields)
@override_settings(EXEMPT_VIEW_PERMISSIONS=[]) @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
@ -134,9 +133,9 @@ class APIViewTestCases:
""" """
GET a list of objects as an authenticated user with permission to view the objects. GET a list of objects as an authenticated user with permission to view the objects.
""" """
self.assertGreaterEqual(self.model.objects.unrestricted().count(), 3, self.assertGreaterEqual(self._get_queryset().count(), 3,
f"Test requires the creation of at least three {self.model} instances") f"Test requires the creation of at least three {self.model} instances")
instance1, instance2 = self.model.objects.unrestricted()[:2] instance1, instance2 = self._get_queryset()[:2]
# Add object-level permission # Add object-level permission
obj_perm = ObjectPermission( obj_perm = ObjectPermission(
@ -178,12 +177,12 @@ class APIViewTestCases:
obj_perm.users.add(self.user) obj_perm.users.add(self.user)
obj_perm.object_types.add(ContentType.objects.get_for_model(self.model)) obj_perm.object_types.add(ContentType.objects.get_for_model(self.model))
initial_count = self.model.objects.unrestricted().count() initial_count = self._get_queryset().count()
response = self.client.post(self._get_list_url(), self.create_data[0], format='json', **self.header) response = self.client.post(self._get_list_url(), self.create_data[0], format='json', **self.header)
self.assertHttpStatus(response, status.HTTP_201_CREATED) self.assertHttpStatus(response, status.HTTP_201_CREATED)
self.assertEqual(self.model.objects.unrestricted().count(), initial_count + 1) self.assertEqual(self._get_queryset().count(), initial_count + 1)
self.assertInstanceEqual( self.assertInstanceEqual(
self.model.objects.unrestricted().get(pk=response.data['id']), self._get_queryset().get(pk=response.data['id']),
self.create_data[0], self.create_data[0],
api=True api=True
) )
@ -200,14 +199,14 @@ class APIViewTestCases:
obj_perm.users.add(self.user) obj_perm.users.add(self.user)
obj_perm.object_types.add(ContentType.objects.get_for_model(self.model)) obj_perm.object_types.add(ContentType.objects.get_for_model(self.model))
initial_count = self.model.objects.unrestricted().count() initial_count = self._get_queryset().count()
response = self.client.post(self._get_list_url(), self.create_data, format='json', **self.header) response = self.client.post(self._get_list_url(), self.create_data, format='json', **self.header)
self.assertHttpStatus(response, status.HTTP_201_CREATED) self.assertHttpStatus(response, status.HTTP_201_CREATED)
self.assertEqual(len(response.data), len(self.create_data)) self.assertEqual(len(response.data), len(self.create_data))
self.assertEqual(self.model.objects.unrestricted().count(), initial_count + len(self.create_data)) self.assertEqual(self._get_queryset().count(), initial_count + len(self.create_data))
for i, obj in enumerate(response.data): for i, obj in enumerate(response.data):
self.assertInstanceEqual( self.assertInstanceEqual(
self.model.objects.unrestricted().get(pk=obj['id']), self._get_queryset().get(pk=obj['id']),
self.create_data[i], self.create_data[i],
api=True api=True
) )
@ -219,7 +218,7 @@ class APIViewTestCases:
""" """
PATCH a single object without permission. PATCH a single object without permission.
""" """
url = self._get_detail_url(self.model.objects.unrestricted().first()) url = self._get_detail_url(self._get_queryset().first())
update_data = self.update_data or getattr(self, 'create_data')[0] update_data = self.update_data or getattr(self, 'create_data')[0]
# Try PATCH without permission # Try PATCH without permission
@ -231,7 +230,7 @@ class APIViewTestCases:
""" """
PATCH a single object identified by its numeric ID. PATCH a single object identified by its numeric ID.
""" """
instance = self.model.objects.unrestricted().first() instance = self._get_queryset().first()
url = self._get_detail_url(instance) url = self._get_detail_url(instance)
update_data = self.update_data or getattr(self, 'create_data')[0] update_data = self.update_data or getattr(self, 'create_data')[0]
@ -254,7 +253,7 @@ class APIViewTestCases:
""" """
DELETE a single object without permission. DELETE a single object without permission.
""" """
url = self._get_detail_url(self.model.objects.unrestricted().first()) url = self._get_detail_url(self._get_queryset().first())
# Try DELETE without permission # Try DELETE without permission
with disable_warnings('django.request'): with disable_warnings('django.request'):
@ -265,7 +264,7 @@ class APIViewTestCases:
""" """
DELETE a single object identified by its numeric ID. DELETE a single object identified by its numeric ID.
""" """
instance = self.model.objects.unrestricted().first() instance = self._get_queryset().first()
url = self._get_detail_url(instance) url = self._get_detail_url(instance)
# Add object-level permission # Add object-level permission
@ -278,7 +277,7 @@ class APIViewTestCases:
response = self.client.delete(url, **self.header) response = self.client.delete(url, **self.header)
self.assertHttpStatus(response, status.HTTP_204_NO_CONTENT) self.assertHttpStatus(response, status.HTTP_204_NO_CONTENT)
self.assertFalse(self.model.objects.unrestricted().filter(pk=instance.pk).exists()) self.assertFalse(self._get_queryset().filter(pk=instance.pk).exists())
class APIViewTestCase( class APIViewTestCase(
GetObjectViewTestCase, GetObjectViewTestCase,

View File

@ -18,6 +18,7 @@ from .utils import disable_warnings, post_data
__all__ = ( __all__ = (
'TestCase', 'TestCase',
'ModelTestCase',
'ModelViewTestCase', 'ModelViewTestCase',
'ViewTestCases', 'ViewTestCases',
) )
@ -149,22 +150,12 @@ class TestCase(_TestCase):
return tags return tags
# class ModelTestCase(TestCase):
# UI Tests
#
class ModelViewTestCase(TestCase):
""" """
Base TestCase for model views. Subclass to test individual views. Parent class for TestCases which deal with models.
""" """
model = None model = None
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
if self.model is None:
raise Exception("Test case requires model to be defined")
def _get_queryset(self): def _get_queryset(self):
""" """
Return a base queryset suitable for use in test methods. Call unrestricted() if RestrictedQuerySet is in use. Return a base queryset suitable for use in test methods. Call unrestricted() if RestrictedQuerySet is in use.
@ -173,6 +164,16 @@ class ModelViewTestCase(TestCase):
return self.model.objects.unrestricted() return self.model.objects.unrestricted()
return self.model.objects.all() return self.model.objects.all()
#
# UI Tests
#
class ModelViewTestCase(ModelTestCase):
"""
Base TestCase for model views. Subclass to test individual views.
"""
def _get_base_url(self): def _get_base_url(self):
""" """
Return the base format for a URL for the test's model. Override this to test for a model which belongs Return the base format for a URL for the test's model. Override this to test for a model which belongs