api.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282
  1. from django.contrib.auth.models import User
  2. from django.contrib.contenttypes.models import ContentType
  3. from django.urls import reverse
  4. from django.test import override_settings
  5. from rest_framework import status
  6. from rest_framework.test import APIClient
  7. from users.models import ObjectPermission, Token
  8. from .utils import disable_warnings
  9. from .views import TestCase
  10. __all__ = (
  11. 'APITestCase',
  12. 'APIViewTestCases',
  13. )
  14. #
  15. # REST API Tests
  16. #
  17. class APITestCase(TestCase):
  18. client_class = APIClient
  19. model = None
  20. def setUp(self):
  21. """
  22. Create a superuser and token for API calls.
  23. """
  24. # Create the test user and assign permissions
  25. self.user = User.objects.create_user(username='testuser')
  26. self.add_permissions(*self.user_permissions)
  27. self.token = Token.objects.create(user=self.user)
  28. self.header = {'HTTP_AUTHORIZATION': 'Token {}'.format(self.token.key)}
  29. def _get_detail_url(self, instance):
  30. viewname = f'{instance._meta.app_label}-api:{instance._meta.model_name}-detail'
  31. return reverse(viewname, kwargs={'pk': instance.pk})
  32. def _get_list_url(self):
  33. viewname = f'{self.model._meta.app_label}-api:{self.model._meta.model_name}-list'
  34. return reverse(viewname)
  35. class APIViewTestCases:
  36. class GetObjectViewTestCase(APITestCase):
  37. @override_settings(EXEMPT_VIEW_PERMISSIONS=['*'])
  38. def test_get_object_anonymous(self):
  39. """
  40. GET a single object as an unauthenticated user.
  41. """
  42. url = self._get_detail_url(self.model.objects.first())
  43. response = self.client.get(url, **self.header)
  44. self.assertHttpStatus(response, status.HTTP_200_OK)
  45. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  46. def test_get_object_without_permission(self):
  47. """
  48. GET a single object as an authenticated user without the required permission.
  49. """
  50. url = self._get_detail_url(self.model.objects.first())
  51. # Try GET without permission
  52. with disable_warnings('django.request'):
  53. self.assertHttpStatus(self.client.get(url, **self.header), status.HTTP_403_FORBIDDEN)
  54. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  55. def test_get_object(self):
  56. """
  57. GET a single object as an authenticated user with permission to view the object.
  58. """
  59. self.assertGreaterEqual(self.model.objects.count(), 2,
  60. f"Test requires the creation of at least two {self.model} instances")
  61. instance1, instance2 = self.model.objects.all()[:2]
  62. # Add object-level permission
  63. obj_perm = ObjectPermission(
  64. constraints={'pk': instance1.pk},
  65. actions=['view']
  66. )
  67. obj_perm.save()
  68. obj_perm.users.add(self.user)
  69. obj_perm.object_types.add(ContentType.objects.get_for_model(self.model))
  70. # Try GET to permitted object
  71. url = self._get_detail_url(instance1)
  72. self.assertHttpStatus(self.client.get(url, **self.header), status.HTTP_200_OK)
  73. # Try GET to non-permitted object
  74. url = self._get_detail_url(instance2)
  75. self.assertHttpStatus(self.client.get(url, **self.header), status.HTTP_404_NOT_FOUND)
  76. class ListObjectsViewTestCase(APITestCase):
  77. brief_fields = []
  78. @override_settings(EXEMPT_VIEW_PERMISSIONS=['*'])
  79. def test_list_objects_anonymous(self):
  80. """
  81. GET a list of objects as an unauthenticated user.
  82. """
  83. url = self._get_list_url()
  84. response = self.client.get(url, **self.header)
  85. self.assertEqual(len(response.data['results']), self.model.objects.count())
  86. self.assertHttpStatus(response, status.HTTP_200_OK)
  87. @override_settings(EXEMPT_VIEW_PERMISSIONS=['*'])
  88. def test_list_objects_brief(self):
  89. """
  90. GET a list of objects using the "brief" parameter as an unauthenticated user.
  91. """
  92. url = f'{self._get_list_url()}?brief=1'
  93. response = self.client.get(url, **self.header)
  94. self.assertEqual(len(response.data['results']), self.model.objects.count())
  95. self.assertEqual(sorted(response.data['results'][0]), self.brief_fields)
  96. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  97. def test_list_objects_without_permission(self):
  98. """
  99. GET a list of objects as an authenticated user without the required permission.
  100. """
  101. url = self._get_list_url()
  102. # Try GET without permission
  103. with disable_warnings('django.request'):
  104. self.assertHttpStatus(self.client.get(url, **self.header), status.HTTP_403_FORBIDDEN)
  105. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  106. def test_list_objects(self):
  107. """
  108. GET a list of objects as an authenticated user with permission to view the objects.
  109. """
  110. self.assertGreaterEqual(self.model.objects.count(), 3,
  111. f"Test requires the creation of at least three {self.model} instances")
  112. instance1, instance2 = self.model.objects.all()[:2]
  113. # Add object-level permission
  114. obj_perm = ObjectPermission(
  115. constraints={'pk__in': [instance1.pk, instance2.pk]},
  116. actions=['view']
  117. )
  118. obj_perm.save()
  119. obj_perm.users.add(self.user)
  120. obj_perm.object_types.add(ContentType.objects.get_for_model(self.model))
  121. # Try GET to permitted objects
  122. response = self.client.get(self._get_list_url(), **self.header)
  123. self.assertHttpStatus(response, status.HTTP_200_OK)
  124. self.assertEqual(len(response.data['results']), 2)
  125. class CreateObjectViewTestCase(APITestCase):
  126. create_data = []
  127. def test_create_object_without_permission(self):
  128. """
  129. POST a single object without permission.
  130. """
  131. url = self._get_list_url()
  132. # Try POST without permission
  133. with disable_warnings('django.request'):
  134. response = self.client.post(url, self.create_data[0], format='json', **self.header)
  135. self.assertHttpStatus(response, status.HTTP_403_FORBIDDEN)
  136. def test_create_object(self):
  137. """
  138. POST a single object with permission.
  139. """
  140. # Add object-level permission
  141. obj_perm = ObjectPermission(
  142. actions=['add']
  143. )
  144. obj_perm.save()
  145. obj_perm.users.add(self.user)
  146. obj_perm.object_types.add(ContentType.objects.get_for_model(self.model))
  147. initial_count = self.model.objects.count()
  148. response = self.client.post(self._get_list_url(), self.create_data[0], format='json', **self.header)
  149. self.assertHttpStatus(response, status.HTTP_201_CREATED)
  150. self.assertEqual(self.model.objects.count(), initial_count + 1)
  151. self.assertInstanceEqual(self.model.objects.get(pk=response.data['id']), self.create_data[0], api=True)
  152. def test_bulk_create_objects(self):
  153. """
  154. POST a set of objects in a single request.
  155. """
  156. # Add object-level permission
  157. obj_perm = ObjectPermission(
  158. actions=['add']
  159. )
  160. obj_perm.save()
  161. obj_perm.users.add(self.user)
  162. obj_perm.object_types.add(ContentType.objects.get_for_model(self.model))
  163. initial_count = self.model.objects.count()
  164. response = self.client.post(self._get_list_url(), self.create_data, format='json', **self.header)
  165. self.assertHttpStatus(response, status.HTTP_201_CREATED)
  166. self.assertEqual(len(response.data), len(self.create_data))
  167. self.assertEqual(self.model.objects.count(), initial_count + len(self.create_data))
  168. for i, obj in enumerate(response.data):
  169. self.assertInstanceEqual(self.model.objects.get(pk=obj['id']), self.create_data[i], api=True)
  170. class UpdateObjectViewTestCase(APITestCase):
  171. update_data = {}
  172. def test_update_object_without_permission(self):
  173. """
  174. PATCH a single object without permission.
  175. """
  176. url = self._get_detail_url(self.model.objects.first())
  177. update_data = self.update_data or getattr(self, 'create_data')[0]
  178. # Try PATCH without permission
  179. with disable_warnings('django.request'):
  180. response = self.client.patch(url, update_data, format='json', **self.header)
  181. self.assertHttpStatus(response, status.HTTP_403_FORBIDDEN)
  182. def test_update_object(self):
  183. """
  184. PATCH a single object identified by its numeric ID.
  185. """
  186. instance = self.model.objects.first()
  187. url = self._get_detail_url(instance)
  188. update_data = self.update_data or getattr(self, 'create_data')[0]
  189. # Add object-level permission
  190. obj_perm = ObjectPermission(
  191. actions=['change']
  192. )
  193. obj_perm.save()
  194. obj_perm.users.add(self.user)
  195. obj_perm.object_types.add(ContentType.objects.get_for_model(self.model))
  196. response = self.client.patch(url, update_data, format='json', **self.header)
  197. self.assertHttpStatus(response, status.HTTP_200_OK)
  198. instance.refresh_from_db()
  199. self.assertInstanceEqual(instance, self.update_data, api=True)
  200. class DeleteObjectViewTestCase(APITestCase):
  201. def test_delete_object_without_permission(self):
  202. """
  203. DELETE a single object without permission.
  204. """
  205. url = self._get_detail_url(self.model.objects.first())
  206. # Try DELETE without permission
  207. with disable_warnings('django.request'):
  208. response = self.client.delete(url, **self.header)
  209. self.assertHttpStatus(response, status.HTTP_403_FORBIDDEN)
  210. def test_delete_object(self):
  211. """
  212. DELETE a single object identified by its numeric ID.
  213. """
  214. instance = self.model.objects.first()
  215. url = self._get_detail_url(instance)
  216. # Add object-level permission
  217. obj_perm = ObjectPermission(
  218. actions=['delete']
  219. )
  220. obj_perm.save()
  221. obj_perm.users.add(self.user)
  222. obj_perm.object_types.add(ContentType.objects.get_for_model(self.model))
  223. response = self.client.delete(url, **self.header)
  224. self.assertHttpStatus(response, status.HTTP_204_NO_CONTENT)
  225. self.assertFalse(self.model.objects.filter(pk=instance.pk).exists())
  226. class APIViewTestCase(
  227. GetObjectViewTestCase,
  228. ListObjectsViewTestCase,
  229. CreateObjectViewTestCase,
  230. UpdateObjectViewTestCase,
  231. DeleteObjectViewTestCase
  232. ):
  233. pass