testcases.py 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  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. # Built-in CSV export
  113. if hasattr(self.model, 'csv_headers'):
  114. response = self.client.get('{}?export'.format(self._get_url('list')))
  115. self.assertHttpStatus(response, 200)
  116. self.assertEqual(response.get('Content-Type'), 'text/csv')
  117. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  118. def test_get_object(self):
  119. instance = self.model.objects.first()
  120. # Attempt to make the request without required permissions
  121. with disable_warnings('django.request'):
  122. self.assertHttpStatus(self.client.get(instance.get_absolute_url()), 403)
  123. # Assign the required permission and submit again
  124. self.add_permissions(
  125. '{}.view_{}'.format(self.model._meta.app_label, self.model._meta.model_name)
  126. )
  127. response = self.client.get(instance.get_absolute_url())
  128. self.assertHttpStatus(response, 200)
  129. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  130. def test_create_object(self):
  131. initial_count = self.model.objects.count()
  132. request = {
  133. 'path': self._get_url('add'),
  134. 'data': post_data(self.form_data),
  135. 'follow': False, # Do not follow 302 redirects
  136. }
  137. # Attempt to make the request without required permissions
  138. with disable_warnings('django.request'):
  139. self.assertHttpStatus(self.client.post(**request), 403)
  140. # Assign the required permission and submit again
  141. self.add_permissions(
  142. '{}.add_{}'.format(self.model._meta.app_label, self.model._meta.model_name)
  143. )
  144. response = self.client.post(**request)
  145. self.assertHttpStatus(response, 302)
  146. self.assertEqual(initial_count + 1, self.model.objects.count())
  147. instance = self.model.objects.order_by('-pk').first()
  148. self.assertDictEqual(model_to_dict(instance), self.form_data)
  149. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  150. def test_edit_object(self):
  151. instance = self.model.objects.first()
  152. request = {
  153. 'path': self._get_url('edit', instance),
  154. 'data': post_data(self.form_data),
  155. 'follow': False, # Do not follow 302 redirects
  156. }
  157. # Attempt to make the request without required permissions
  158. with disable_warnings('django.request'):
  159. self.assertHttpStatus(self.client.post(**request), 403)
  160. # Assign the required permission and submit again
  161. self.add_permissions(
  162. '{}.change_{}'.format(self.model._meta.app_label, self.model._meta.model_name)
  163. )
  164. response = self.client.post(**request)
  165. self.assertHttpStatus(response, 302)
  166. instance = self.model.objects.get(pk=instance.pk)
  167. self.assertDictEqual(model_to_dict(instance), self.form_data)
  168. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  169. def test_delete_object(self):
  170. instance = self.model.objects.first()
  171. request = {
  172. 'path': self._get_url('delete', instance),
  173. 'data': {'confirm': True},
  174. 'follow': False, # Do not follow 302 redirects
  175. }
  176. # Attempt to make the request without required permissions
  177. with disable_warnings('django.request'):
  178. self.assertHttpStatus(self.client.post(**request), 403)
  179. # Assign the required permission and submit again
  180. self.add_permissions(
  181. '{}.delete_{}'.format(self.model._meta.app_label, self.model._meta.model_name)
  182. )
  183. response = self.client.post(**request)
  184. self.assertHttpStatus(response, 302)
  185. with self.assertRaises(ObjectDoesNotExist):
  186. self.model.objects.get(pk=instance.pk)
  187. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  188. def test_import_objects(self):
  189. initial_count = self.model.objects.count()
  190. request = {
  191. 'path': self._get_url('import'),
  192. 'data': {
  193. 'csv': '\n'.join(self.csv_data)
  194. }
  195. }
  196. # Attempt to make the request without required permissions
  197. with disable_warnings('django.request'):
  198. self.assertHttpStatus(self.client.post(**request), 403)
  199. # Assign the required permission and submit again
  200. self.add_permissions(
  201. '{}.view_{}'.format(self.model._meta.app_label, self.model._meta.model_name),
  202. '{}.add_{}'.format(self.model._meta.app_label, self.model._meta.model_name)
  203. )
  204. response = self.client.post(**request)
  205. self.assertHttpStatus(response, 200)
  206. self.assertEqual(self.model.objects.count(), initial_count + len(self.csv_data) - 1)