testcases.py 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249
  1. from django.contrib.auth.models import Permission, User
  2. from django.core.exceptions import ObjectDoesNotExist
  3. from django.test import Client, TestCase as _TestCase, override_settings
  4. from django.urls import reverse, NoReverseMatch
  5. from rest_framework.test import APIClient
  6. from users.models import Token
  7. from .utils import disable_warnings, model_to_dict, post_data
  8. class TestCase(_TestCase):
  9. user_permissions = ()
  10. def setUp(self):
  11. # Create the test user and assign permissions
  12. self.user = User.objects.create_user(username='testuser')
  13. self.add_permissions(*self.user_permissions)
  14. # Initialize the test client
  15. self.client = Client()
  16. self.client.force_login(self.user)
  17. #
  18. # Permissions management
  19. #
  20. def add_permissions(self, *names):
  21. """
  22. Assign a set of permissions to the test user. Accepts permission names in the form <app>.<action>_<model>.
  23. """
  24. for name in names:
  25. app, codename = name.split('.')
  26. perm = Permission.objects.get(content_type__app_label=app, codename=codename)
  27. self.user.user_permissions.add(perm)
  28. def remove_permissions(self, *names):
  29. """
  30. Remove a set of permissions from the test user, if assigned.
  31. """
  32. for name in names:
  33. app, codename = name.split('.')
  34. perm = Permission.objects.get(content_type__app_label=app, codename=codename)
  35. self.user.user_permissions.remove(perm)
  36. #
  37. # Convenience methods
  38. #
  39. def assertHttpStatus(self, response, expected_status):
  40. """
  41. TestCase method. Provide more detail in the event of an unexpected HTTP response.
  42. """
  43. err_message = "Expected HTTP status {}; received {}: {}"
  44. self.assertEqual(response.status_code, expected_status, err_message.format(
  45. expected_status, response.status_code, getattr(response, 'data', 'No data')
  46. ))
  47. class APITestCase(TestCase):
  48. client_class = APIClient
  49. def setUp(self):
  50. """
  51. Create a superuser and token for API calls.
  52. """
  53. self.user = User.objects.create(username='testuser', is_superuser=True)
  54. self.token = Token.objects.create(user=self.user)
  55. self.header = {'HTTP_AUTHORIZATION': 'Token {}'.format(self.token.key)}
  56. class StandardTestCases:
  57. """
  58. We keep any TestCases with test_* methods inside a class to prevent unittest from trying to run them.
  59. """
  60. class Views(TestCase):
  61. """
  62. Stock TestCase suitable for testing all standard View functions:
  63. - List objects
  64. - View single object
  65. - Create new object
  66. - Modify existing object
  67. - Delete existing object
  68. - Import multiple new objects
  69. """
  70. model = None
  71. form_data = {}
  72. csv_data = {}
  73. maxDiff = None
  74. def __init__(self, *args, **kwargs):
  75. super().__init__(*args, **kwargs)
  76. if self.model is None:
  77. raise Exception("Test case requires model to be defined")
  78. def _get_url(self, action, instance=None):
  79. """
  80. Return the URL name for a specific action. An instance must be specified for
  81. get/edit/delete views.
  82. """
  83. url_format = '{}:{}_{{}}'.format(
  84. self.model._meta.app_label,
  85. self.model._meta.model_name
  86. )
  87. if action in ('list', 'add', 'import'):
  88. return reverse(url_format.format(action))
  89. elif action in ('get', 'edit', 'delete'):
  90. if instance is None:
  91. raise Exception("Resolving {} URL requires specifying an instance".format(action))
  92. # Attempt to resolve using slug first
  93. if hasattr(self.model, 'slug'):
  94. try:
  95. return reverse(url_format.format(action), kwargs={'slug': instance.slug})
  96. except NoReverseMatch:
  97. pass
  98. return reverse(url_format.format(action), kwargs={'pk': instance.pk})
  99. else:
  100. raise Exception("Invalid action for URL resolution: {}".format(action))
  101. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  102. def test_list_objects(self):
  103. # Attempt to make the request without required permissions
  104. with disable_warnings('django.request'):
  105. self.assertHttpStatus(self.client.get(self._get_url('list')), 403)
  106. # Assign the required permission and submit again
  107. self.add_permissions(
  108. '{}.view_{}'.format(self.model._meta.app_label, self.model._meta.model_name)
  109. )
  110. response = self.client.get(self._get_url('list'))
  111. self.assertHttpStatus(response, 200)
  112. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  113. def test_get_object(self):
  114. instance = self.model.objects.first()
  115. # Attempt to make the request without required permissions
  116. with disable_warnings('django.request'):
  117. self.assertHttpStatus(self.client.get(instance.get_absolute_url()), 403)
  118. # Assign the required permission and submit again
  119. self.add_permissions(
  120. '{}.view_{}'.format(self.model._meta.app_label, self.model._meta.model_name)
  121. )
  122. response = self.client.get(instance.get_absolute_url())
  123. self.assertHttpStatus(response, 200)
  124. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  125. def test_create_object(self):
  126. initial_count = self.model.objects.count()
  127. request = {
  128. 'path': self._get_url('add'),
  129. 'data': post_data(self.form_data),
  130. 'follow': False, # Do not follow 302 redirects
  131. }
  132. # Attempt to make the request without required permissions
  133. with disable_warnings('django.request'):
  134. self.assertHttpStatus(self.client.post(**request), 403)
  135. # Assign the required permission and submit again
  136. self.add_permissions(
  137. '{}.add_{}'.format(self.model._meta.app_label, self.model._meta.model_name)
  138. )
  139. response = self.client.post(**request)
  140. self.assertHttpStatus(response, 302)
  141. self.assertEqual(initial_count + 1, self.model.objects.count())
  142. instance = self.model.objects.order_by('-pk').first()
  143. self.assertDictEqual(model_to_dict(instance), self.form_data)
  144. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  145. def test_edit_object(self):
  146. instance = self.model.objects.first()
  147. request = {
  148. 'path': self._get_url('edit', instance),
  149. 'data': post_data(self.form_data),
  150. 'follow': False, # Do not follow 302 redirects
  151. }
  152. # Attempt to make the request without required permissions
  153. with disable_warnings('django.request'):
  154. self.assertHttpStatus(self.client.post(**request), 403)
  155. # Assign the required permission and submit again
  156. self.add_permissions(
  157. '{}.change_{}'.format(self.model._meta.app_label, self.model._meta.model_name)
  158. )
  159. response = self.client.post(**request)
  160. self.assertHttpStatus(response, 302)
  161. instance = self.model.objects.get(pk=instance.pk)
  162. self.assertDictEqual(model_to_dict(instance), self.form_data)
  163. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  164. def test_delete_object(self):
  165. instance = self.model.objects.first()
  166. request = {
  167. 'path': self._get_url('delete', instance),
  168. 'data': {'confirm': True},
  169. 'follow': False, # Do not follow 302 redirects
  170. }
  171. # Attempt to make the request without required permissions
  172. with disable_warnings('django.request'):
  173. self.assertHttpStatus(self.client.post(**request), 403)
  174. # Assign the required permission and submit again
  175. self.add_permissions(
  176. '{}.delete_{}'.format(self.model._meta.app_label, self.model._meta.model_name)
  177. )
  178. response = self.client.post(**request)
  179. self.assertHttpStatus(response, 302)
  180. with self.assertRaises(ObjectDoesNotExist):
  181. self.model.objects.get(pk=instance.pk)
  182. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  183. def test_import_objects(self):
  184. initial_count = self.model.objects.count()
  185. request = {
  186. 'path': self._get_url('import'),
  187. 'data': {
  188. 'csv': '\n'.join(self.csv_data)
  189. }
  190. }
  191. # Attempt to make the request without required permissions
  192. with disable_warnings('django.request'):
  193. self.assertHttpStatus(self.client.post(**request), 403)
  194. # Assign the required permission and submit again
  195. self.add_permissions(
  196. '{}.view_{}'.format(self.model._meta.app_label, self.model._meta.model_name),
  197. '{}.add_{}'.format(self.model._meta.app_label, self.model._meta.model_name)
  198. )
  199. response = self.client.post(**request)
  200. self.assertHttpStatus(response, 200)
  201. self.assertEqual(self.model.objects.count(), initial_count + len(self.csv_data) - 1)