testcases.py 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  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
  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. def test_list_objects(self):
  102. response = self.client.get(self._get_url('list'))
  103. self.assertHttpStatus(response, 200)
  104. def test_get_object(self):
  105. instance = self.model.objects.first()
  106. response = self.client.get(instance.get_absolute_url())
  107. self.assertHttpStatus(response, 200)
  108. def test_create_object(self):
  109. initial_count = self.model.objects.count()
  110. request = {
  111. 'path': self._get_url('add'),
  112. 'data': post_data(self.form_data),
  113. 'follow': True,
  114. }
  115. # Attempt to make the request without required permissions
  116. with disable_warnings('django.request'):
  117. self.assertHttpStatus(self.client.post(**request), 403)
  118. # Assign the required permission and submit again
  119. self.add_permissions('{}.add_{}'.format(self.model._meta.app_label, self.model._meta.model_name))
  120. response = self.client.post(**request)
  121. self.assertHttpStatus(response, 200)
  122. self.assertEqual(initial_count + 1, self.model.objects.count())
  123. instance = self.model.objects.order_by('-pk').first()
  124. self.assertDictEqual(model_to_dict(instance), self.form_data)
  125. def test_edit_object(self):
  126. instance = self.model.objects.first()
  127. request = {
  128. 'path': self._get_url('edit', instance),
  129. 'data': post_data(self.form_data),
  130. 'follow': True,
  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('{}.change_{}'.format(self.model._meta.app_label, self.model._meta.model_name))
  137. response = self.client.post(**request)
  138. self.assertHttpStatus(response, 200)
  139. instance = self.model.objects.get(pk=instance.pk)
  140. self.assertDictEqual(model_to_dict(instance), self.form_data)
  141. def test_delete_object(self):
  142. instance = self.model.objects.first()
  143. request = {
  144. 'path': self._get_url('delete', instance),
  145. 'data': {'confirm': True},
  146. 'follow': True,
  147. }
  148. # Attempt to make the request without required permissions
  149. with disable_warnings('django.request'):
  150. self.assertHttpStatus(self.client.post(**request), 403)
  151. # Assign the required permission and submit again
  152. self.add_permissions('{}.delete_{}'.format(self.model._meta.app_label, self.model._meta.model_name))
  153. response = self.client.post(**request)
  154. self.assertHttpStatus(response, 200)
  155. with self.assertRaises(ObjectDoesNotExist):
  156. self.model.objects.get(pk=instance.pk)
  157. def test_import_objects(self):
  158. initial_count = self.model.objects.count()
  159. request = {
  160. 'path': self._get_url('import'),
  161. 'data': {
  162. 'csv': '\n'.join(self.csv_data)
  163. }
  164. }
  165. # Attempt to make the request without required permissions
  166. with disable_warnings('django.request'):
  167. self.assertHttpStatus(self.client.post(**request), 403)
  168. # Assign the required permission and submit again
  169. self.add_permissions('{}.add_{}'.format(self.model._meta.app_label, self.model._meta.model_name))
  170. response = self.client.post(**request)
  171. self.assertHttpStatus(response, 200)
  172. self.assertEqual(self.model.objects.count(), initial_count + len(self.csv_data) - 1)