testcases.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325
  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. # Data to be sent when creating/editing individual objects
  72. form_data = {}
  73. # CSV lines used for bulk import of new objects
  74. csv_data = ()
  75. # Form data to be used when editing multiple objects at once
  76. bulk_edit_data = {}
  77. maxDiff = None
  78. def __init__(self, *args, **kwargs):
  79. super().__init__(*args, **kwargs)
  80. if self.model is None:
  81. raise Exception("Test case requires model to be defined")
  82. def _get_url(self, action, instance=None):
  83. """
  84. Return the URL name for a specific action. An instance must be specified for
  85. get/edit/delete views.
  86. """
  87. url_format = '{}:{}_{{}}'.format(
  88. self.model._meta.app_label,
  89. self.model._meta.model_name
  90. )
  91. if action in ('list', 'add', 'import', 'bulk_edit', 'bulk_delete'):
  92. return reverse(url_format.format(action))
  93. elif action in ('get', 'edit', 'delete'):
  94. if instance is None:
  95. raise Exception("Resolving {} URL requires specifying an instance".format(action))
  96. # Attempt to resolve using slug first
  97. if hasattr(self.model, 'slug'):
  98. try:
  99. return reverse(url_format.format(action), kwargs={'slug': instance.slug})
  100. except NoReverseMatch:
  101. pass
  102. return reverse(url_format.format(action), kwargs={'pk': instance.pk})
  103. else:
  104. raise Exception("Invalid action for URL resolution: {}".format(action))
  105. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  106. def test_list_objects(self):
  107. # Attempt to make the request without required permissions
  108. with disable_warnings('django.request'):
  109. self.assertHttpStatus(self.client.get(self._get_url('list')), 403)
  110. # Assign the required permission and submit again
  111. self.add_permissions(
  112. '{}.view_{}'.format(self.model._meta.app_label, self.model._meta.model_name)
  113. )
  114. response = self.client.get(self._get_url('list'))
  115. self.assertHttpStatus(response, 200)
  116. # Built-in CSV export
  117. if hasattr(self.model, 'csv_headers'):
  118. response = self.client.get('{}?export'.format(self._get_url('list')))
  119. self.assertHttpStatus(response, 200)
  120. self.assertEqual(response.get('Content-Type'), 'text/csv')
  121. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  122. def test_get_object(self):
  123. instance = self.model.objects.first()
  124. # Attempt to make the request without required permissions
  125. with disable_warnings('django.request'):
  126. self.assertHttpStatus(self.client.get(instance.get_absolute_url()), 403)
  127. # Assign the required permission and submit again
  128. self.add_permissions(
  129. '{}.view_{}'.format(self.model._meta.app_label, self.model._meta.model_name)
  130. )
  131. response = self.client.get(instance.get_absolute_url())
  132. self.assertHttpStatus(response, 200)
  133. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  134. def test_create_object(self):
  135. initial_count = self.model.objects.count()
  136. request = {
  137. 'path': self._get_url('add'),
  138. 'data': post_data(self.form_data),
  139. 'follow': False, # Do not follow 302 redirects
  140. }
  141. # Attempt to make the request without required permissions
  142. with disable_warnings('django.request'):
  143. self.assertHttpStatus(self.client.post(**request), 403)
  144. # Assign the required permission and submit again
  145. self.add_permissions(
  146. '{}.add_{}'.format(self.model._meta.app_label, self.model._meta.model_name)
  147. )
  148. response = self.client.post(**request)
  149. self.assertHttpStatus(response, 302)
  150. self.assertEqual(initial_count + 1, self.model.objects.count())
  151. instance = self.model.objects.order_by('-pk').first()
  152. self.assertDictEqual(model_to_dict(instance), self.form_data)
  153. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  154. def test_edit_object(self):
  155. instance = self.model.objects.first()
  156. request = {
  157. 'path': self._get_url('edit', instance),
  158. 'data': post_data(self.form_data),
  159. 'follow': False, # Do not follow 302 redirects
  160. }
  161. # Attempt to make the request without required permissions
  162. with disable_warnings('django.request'):
  163. self.assertHttpStatus(self.client.post(**request), 403)
  164. # Assign the required permission and submit again
  165. self.add_permissions(
  166. '{}.change_{}'.format(self.model._meta.app_label, self.model._meta.model_name)
  167. )
  168. response = self.client.post(**request)
  169. self.assertHttpStatus(response, 302)
  170. instance = self.model.objects.get(pk=instance.pk)
  171. self.assertDictEqual(model_to_dict(instance), self.form_data)
  172. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  173. def test_delete_object(self):
  174. instance = self.model.objects.first()
  175. request = {
  176. 'path': self._get_url('delete', instance),
  177. 'data': {'confirm': True},
  178. 'follow': False, # Do not follow 302 redirects
  179. }
  180. # Attempt to make the request without required permissions
  181. with disable_warnings('django.request'):
  182. self.assertHttpStatus(self.client.post(**request), 403)
  183. # Assign the required permission and submit again
  184. self.add_permissions(
  185. '{}.delete_{}'.format(self.model._meta.app_label, self.model._meta.model_name)
  186. )
  187. response = self.client.post(**request)
  188. self.assertHttpStatus(response, 302)
  189. with self.assertRaises(ObjectDoesNotExist):
  190. self.model.objects.get(pk=instance.pk)
  191. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  192. def test_import_objects(self):
  193. initial_count = self.model.objects.count()
  194. request = {
  195. 'path': self._get_url('import'),
  196. 'data': {
  197. 'csv': '\n'.join(self.csv_data)
  198. }
  199. }
  200. # Attempt to make the request without required permissions
  201. with disable_warnings('django.request'):
  202. self.assertHttpStatus(self.client.post(**request), 403)
  203. # Assign the required permission and submit again
  204. self.add_permissions(
  205. '{}.view_{}'.format(self.model._meta.app_label, self.model._meta.model_name),
  206. '{}.add_{}'.format(self.model._meta.app_label, self.model._meta.model_name)
  207. )
  208. response = self.client.post(**request)
  209. self.assertHttpStatus(response, 200)
  210. self.assertEqual(self.model.objects.count(), initial_count + len(self.csv_data) - 1)
  211. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  212. def test_bulk_edit_objects(self):
  213. pk_list = self.model.objects.values_list('pk', flat=True)
  214. request = {
  215. 'path': self._get_url('bulk_edit'),
  216. 'data': {
  217. 'pk': pk_list,
  218. '_apply': True, # Form button
  219. },
  220. 'follow': False, # Do not follow 302 redirects
  221. }
  222. # Append the form data to the request
  223. request['data'].update(post_data(self.bulk_edit_data))
  224. # Attempt to make the request without required permissions
  225. with disable_warnings('django.request'):
  226. self.assertHttpStatus(self.client.post(**request), 403)
  227. # Assign the required permission and submit again
  228. self.add_permissions(
  229. '{}.change_{}'.format(self.model._meta.app_label, self.model._meta.model_name)
  230. )
  231. response = self.client.post(**request)
  232. self.assertHttpStatus(response, 302)
  233. bulk_edit_fields = self.bulk_edit_data.keys()
  234. for i, instance in enumerate(self.model.objects.filter(pk__in=pk_list)):
  235. self.assertDictEqual(
  236. model_to_dict(instance, fields=bulk_edit_fields),
  237. self.bulk_edit_data,
  238. msg="Instance {} failed to validate after bulk edit: {}".format(i, instance)
  239. )
  240. @override_settings(EXEMPT_VIEW_PERMISSIONS=[])
  241. def test_bulk_delete_objects(self):
  242. pk_list = self.model.objects.values_list('pk', flat=True)
  243. request = {
  244. 'path': self._get_url('bulk_delete'),
  245. 'data': {
  246. 'pk': pk_list,
  247. 'confirm': True,
  248. '_confirm': True, # Form button
  249. },
  250. 'follow': False, # Do not follow 302 redirects
  251. }
  252. # Attempt to make the request without required permissions
  253. with disable_warnings('django.request'):
  254. self.assertHttpStatus(self.client.post(**request), 403)
  255. # Assign the required permission and submit again
  256. self.add_permissions(
  257. '{}.delete_{}'.format(self.model._meta.app_label, self.model._meta.model_name)
  258. )
  259. response = self.client.post(**request)
  260. self.assertHttpStatus(response, 302)
  261. # Check that all objects were deleted
  262. self.assertEqual(self.model.objects.count(), 0)