mirror of
https://github.com/netbox-community/netbox.git
synced 2025-07-23 17:08:41 -06:00
Refactor APITestCase to provide dynamic queryset specification
This commit is contained in:
parent
a2d957ba0d
commit
2fbe138c71
@ -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,
|
||||||
|
@ -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
|
||||||
|
Loading…
Reference in New Issue
Block a user