test_edit.py 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428
  1. from django import forms
  2. from django.core.exceptions import ImproperlyConfigured
  3. from django.test import SimpleTestCase, TestCase, override_settings
  4. from django.test.client import RequestFactory
  5. from django.urls import reverse
  6. from django.views.generic.base import View
  7. from django.views.generic.edit import CreateView, FormMixin, ModelFormMixin
  8. from . import views
  9. from .forms import AuthorForm
  10. from .models import Artist, Author
  11. class FormMixinTests(SimpleTestCase):
  12. request_factory = RequestFactory()
  13. def test_initial_data(self):
  14. """ Test instance independence of initial data dict (see #16138) """
  15. initial_1 = FormMixin().get_initial()
  16. initial_1['foo'] = 'bar'
  17. initial_2 = FormMixin().get_initial()
  18. self.assertNotEqual(initial_1, initial_2)
  19. def test_get_prefix(self):
  20. """ Test prefix can be set (see #18872) """
  21. test_string = 'test'
  22. get_request = self.request_factory.get('/')
  23. class TestFormMixin(FormMixin):
  24. request = get_request
  25. default_kwargs = TestFormMixin().get_form_kwargs()
  26. self.assertIsNone(default_kwargs.get('prefix'))
  27. set_mixin = TestFormMixin()
  28. set_mixin.prefix = test_string
  29. set_kwargs = set_mixin.get_form_kwargs()
  30. self.assertEqual(test_string, set_kwargs.get('prefix'))
  31. def test_get_form(self):
  32. class TestFormMixin(FormMixin):
  33. request = self.request_factory.get('/')
  34. self.assertIsInstance(
  35. TestFormMixin().get_form(forms.Form), forms.Form,
  36. 'get_form() should use provided form class.'
  37. )
  38. class FormClassTestFormMixin(TestFormMixin):
  39. form_class = forms.Form
  40. self.assertIsInstance(
  41. FormClassTestFormMixin().get_form(), forms.Form,
  42. 'get_form() should fallback to get_form_class() if none is provided.'
  43. )
  44. def test_get_context_data(self):
  45. class FormContext(FormMixin):
  46. request = self.request_factory.get('/')
  47. form_class = forms.Form
  48. self.assertIsInstance(FormContext().get_context_data()['form'], forms.Form)
  49. @override_settings(ROOT_URLCONF='generic_views.urls')
  50. class BasicFormTests(TestCase):
  51. def test_post_data(self):
  52. res = self.client.post('/contact/', {'name': "Me", 'message': "Hello"})
  53. self.assertRedirects(res, '/list/authors/')
  54. def test_late_form_validation(self):
  55. """
  56. A form can be marked invalid in the form_valid() method (#25548).
  57. """
  58. res = self.client.post('/late-validation/', {'name': "Me", 'message': "Hello"})
  59. self.assertFalse(res.context['form'].is_valid())
  60. class ModelFormMixinTests(SimpleTestCase):
  61. def test_get_form(self):
  62. form_class = views.AuthorGetQuerySetFormView().get_form_class()
  63. self.assertEqual(form_class._meta.model, Author)
  64. def test_get_form_checks_for_object(self):
  65. mixin = ModelFormMixin()
  66. mixin.request = RequestFactory().get('/')
  67. self.assertEqual({'initial': {}, 'prefix': None},
  68. mixin.get_form_kwargs())
  69. @override_settings(ROOT_URLCONF='generic_views.urls')
  70. class CreateViewTests(TestCase):
  71. def test_create(self):
  72. res = self.client.get('/edit/authors/create/')
  73. self.assertEqual(res.status_code, 200)
  74. self.assertIsInstance(res.context['form'], forms.ModelForm)
  75. self.assertIsInstance(res.context['view'], View)
  76. self.assertNotIn('object', res.context)
  77. self.assertNotIn('author', res.context)
  78. self.assertTemplateUsed(res, 'generic_views/author_form.html')
  79. res = self.client.post('/edit/authors/create/', {'name': 'Randall Munroe', 'slug': 'randall-munroe'})
  80. self.assertEqual(res.status_code, 302)
  81. self.assertRedirects(res, '/list/authors/')
  82. self.assertQuerysetEqual(Author.objects.values_list('name', flat=True), ['Randall Munroe'])
  83. def test_create_invalid(self):
  84. res = self.client.post('/edit/authors/create/', {'name': 'A' * 101, 'slug': 'randall-munroe'})
  85. self.assertEqual(res.status_code, 200)
  86. self.assertTemplateUsed(res, 'generic_views/author_form.html')
  87. self.assertEqual(len(res.context['form'].errors), 1)
  88. self.assertEqual(Author.objects.count(), 0)
  89. def test_create_with_object_url(self):
  90. res = self.client.post('/edit/artists/create/', {'name': 'Rene Magritte'})
  91. self.assertEqual(res.status_code, 302)
  92. artist = Artist.objects.get(name='Rene Magritte')
  93. self.assertRedirects(res, '/detail/artist/%d/' % artist.pk)
  94. self.assertQuerysetEqual(Artist.objects.all(), [artist])
  95. def test_create_with_redirect(self):
  96. res = self.client.post('/edit/authors/create/redirect/', {'name': 'Randall Munroe', 'slug': 'randall-munroe'})
  97. self.assertEqual(res.status_code, 302)
  98. self.assertRedirects(res, '/edit/authors/create/')
  99. self.assertQuerysetEqual(Author.objects.values_list('name', flat=True), ['Randall Munroe'])
  100. def test_create_with_interpolated_redirect(self):
  101. res = self.client.post(
  102. '/edit/authors/create/interpolate_redirect/',
  103. {'name': 'Randall Munroe', 'slug': 'randall-munroe'}
  104. )
  105. self.assertQuerysetEqual(Author.objects.values_list('name', flat=True), ['Randall Munroe'])
  106. self.assertEqual(res.status_code, 302)
  107. pk = Author.objects.first().pk
  108. self.assertRedirects(res, '/edit/author/%d/update/' % pk)
  109. # Also test with escaped chars in URL
  110. res = self.client.post(
  111. '/edit/authors/create/interpolate_redirect_nonascii/',
  112. {'name': 'John Doe', 'slug': 'john-doe'}
  113. )
  114. self.assertEqual(res.status_code, 302)
  115. pk = Author.objects.get(name='John Doe').pk
  116. self.assertRedirects(res, '/%C3%A9dit/author/{}/update/'.format(pk))
  117. def test_create_with_special_properties(self):
  118. res = self.client.get('/edit/authors/create/special/')
  119. self.assertEqual(res.status_code, 200)
  120. self.assertIsInstance(res.context['form'], views.AuthorForm)
  121. self.assertNotIn('object', res.context)
  122. self.assertNotIn('author', res.context)
  123. self.assertTemplateUsed(res, 'generic_views/form.html')
  124. res = self.client.post('/edit/authors/create/special/', {'name': 'Randall Munroe', 'slug': 'randall-munroe'})
  125. self.assertEqual(res.status_code, 302)
  126. obj = Author.objects.get(slug='randall-munroe')
  127. self.assertRedirects(res, reverse('author_detail', kwargs={'pk': obj.pk}))
  128. self.assertQuerysetEqual(Author.objects.all(), [obj])
  129. def test_create_without_redirect(self):
  130. msg = (
  131. 'No URL to redirect to. Either provide a url or define a '
  132. 'get_absolute_url method on the Model.'
  133. )
  134. with self.assertRaisesMessage(ImproperlyConfigured, msg):
  135. self.client.post('/edit/authors/create/naive/', {'name': 'Randall Munroe', 'slug': 'randall-munroe'})
  136. def test_create_restricted(self):
  137. res = self.client.post(
  138. '/edit/authors/create/restricted/',
  139. {'name': 'Randall Munroe', 'slug': 'randall-munroe'}
  140. )
  141. self.assertEqual(res.status_code, 302)
  142. self.assertRedirects(res, '/accounts/login/?next=/edit/authors/create/restricted/')
  143. def test_create_view_with_restricted_fields(self):
  144. class MyCreateView(CreateView):
  145. model = Author
  146. fields = ['name']
  147. self.assertEqual(list(MyCreateView().get_form_class().base_fields), ['name'])
  148. def test_create_view_all_fields(self):
  149. class MyCreateView(CreateView):
  150. model = Author
  151. fields = '__all__'
  152. self.assertEqual(list(MyCreateView().get_form_class().base_fields), ['name', 'slug'])
  153. def test_create_view_without_explicit_fields(self):
  154. class MyCreateView(CreateView):
  155. model = Author
  156. message = (
  157. "Using ModelFormMixin (base class of MyCreateView) without the "
  158. "'fields' attribute is prohibited."
  159. )
  160. with self.assertRaisesMessage(ImproperlyConfigured, message):
  161. MyCreateView().get_form_class()
  162. def test_define_both_fields_and_form_class(self):
  163. class MyCreateView(CreateView):
  164. model = Author
  165. form_class = AuthorForm
  166. fields = ['name']
  167. message = "Specifying both 'fields' and 'form_class' is not permitted."
  168. with self.assertRaisesMessage(ImproperlyConfigured, message):
  169. MyCreateView().get_form_class()
  170. @override_settings(ROOT_URLCONF='generic_views.urls')
  171. class UpdateViewTests(TestCase):
  172. @classmethod
  173. def setUpTestData(cls):
  174. cls.author = Author.objects.create(
  175. pk=1, # Required for OneAuthorUpdate.
  176. name='Randall Munroe',
  177. slug='randall-munroe',
  178. )
  179. def test_update_post(self):
  180. res = self.client.get('/edit/author/%d/update/' % self.author.pk)
  181. self.assertEqual(res.status_code, 200)
  182. self.assertIsInstance(res.context['form'], forms.ModelForm)
  183. self.assertEqual(res.context['object'], self.author)
  184. self.assertEqual(res.context['author'], self.author)
  185. self.assertTemplateUsed(res, 'generic_views/author_form.html')
  186. self.assertEqual(res.context['view'].get_form_called_count, 1)
  187. # Modification with both POST and PUT (browser compatible)
  188. res = self.client.post(
  189. '/edit/author/%d/update/' % self.author.pk,
  190. {'name': 'Randall Munroe (xkcd)', 'slug': 'randall-munroe'}
  191. )
  192. self.assertEqual(res.status_code, 302)
  193. self.assertRedirects(res, '/list/authors/')
  194. self.assertQuerysetEqual(Author.objects.values_list('name', flat=True), ['Randall Munroe (xkcd)'])
  195. def test_update_invalid(self):
  196. res = self.client.post(
  197. '/edit/author/%d/update/' % self.author.pk,
  198. {'name': 'A' * 101, 'slug': 'randall-munroe'}
  199. )
  200. self.assertEqual(res.status_code, 200)
  201. self.assertTemplateUsed(res, 'generic_views/author_form.html')
  202. self.assertEqual(len(res.context['form'].errors), 1)
  203. self.assertQuerysetEqual(Author.objects.all(), [self.author])
  204. self.assertEqual(res.context['view'].get_form_called_count, 1)
  205. def test_update_with_object_url(self):
  206. a = Artist.objects.create(name='Rene Magritte')
  207. res = self.client.post('/edit/artists/%d/update/' % a.pk, {'name': 'Rene Magritte'})
  208. self.assertEqual(res.status_code, 302)
  209. self.assertRedirects(res, '/detail/artist/%d/' % a.pk)
  210. self.assertQuerysetEqual(Artist.objects.all(), [a])
  211. def test_update_with_redirect(self):
  212. res = self.client.post(
  213. '/edit/author/%d/update/redirect/' % self.author.pk,
  214. {'name': 'Randall Munroe (author of xkcd)', 'slug': 'randall-munroe'}
  215. )
  216. self.assertEqual(res.status_code, 302)
  217. self.assertRedirects(res, '/edit/authors/create/')
  218. self.assertQuerysetEqual(Author.objects.values_list('name', flat=True), ['Randall Munroe (author of xkcd)'])
  219. def test_update_with_interpolated_redirect(self):
  220. res = self.client.post(
  221. '/edit/author/%d/update/interpolate_redirect/' % self.author.pk,
  222. {'name': 'Randall Munroe (author of xkcd)', 'slug': 'randall-munroe'}
  223. )
  224. self.assertQuerysetEqual(Author.objects.values_list('name', flat=True), ['Randall Munroe (author of xkcd)'])
  225. self.assertEqual(res.status_code, 302)
  226. pk = Author.objects.first().pk
  227. self.assertRedirects(res, '/edit/author/%d/update/' % pk)
  228. # Also test with escaped chars in URL
  229. res = self.client.post(
  230. '/edit/author/%d/update/interpolate_redirect_nonascii/' % self.author.pk,
  231. {'name': 'John Doe', 'slug': 'john-doe'}
  232. )
  233. self.assertEqual(res.status_code, 302)
  234. pk = Author.objects.get(name='John Doe').pk
  235. self.assertRedirects(res, '/%C3%A9dit/author/{}/update/'.format(pk))
  236. def test_update_with_special_properties(self):
  237. res = self.client.get('/edit/author/%d/update/special/' % self.author.pk)
  238. self.assertEqual(res.status_code, 200)
  239. self.assertIsInstance(res.context['form'], views.AuthorForm)
  240. self.assertEqual(res.context['object'], self.author)
  241. self.assertEqual(res.context['thingy'], self.author)
  242. self.assertNotIn('author', res.context)
  243. self.assertTemplateUsed(res, 'generic_views/form.html')
  244. res = self.client.post(
  245. '/edit/author/%d/update/special/' % self.author.pk,
  246. {'name': 'Randall Munroe (author of xkcd)', 'slug': 'randall-munroe'}
  247. )
  248. self.assertEqual(res.status_code, 302)
  249. self.assertRedirects(res, '/detail/author/%d/' % self.author.pk)
  250. self.assertQuerysetEqual(Author.objects.values_list('name', flat=True), ['Randall Munroe (author of xkcd)'])
  251. def test_update_without_redirect(self):
  252. msg = (
  253. 'No URL to redirect to. Either provide a url or define a '
  254. 'get_absolute_url method on the Model.'
  255. )
  256. with self.assertRaisesMessage(ImproperlyConfigured, msg):
  257. self.client.post(
  258. '/edit/author/%d/update/naive/' % self.author.pk,
  259. {'name': 'Randall Munroe (author of xkcd)', 'slug': 'randall-munroe'}
  260. )
  261. def test_update_get_object(self):
  262. res = self.client.get('/edit/author/update/')
  263. self.assertEqual(res.status_code, 200)
  264. self.assertIsInstance(res.context['form'], forms.ModelForm)
  265. self.assertIsInstance(res.context['view'], View)
  266. self.assertEqual(res.context['object'], self.author)
  267. self.assertEqual(res.context['author'], self.author)
  268. self.assertTemplateUsed(res, 'generic_views/author_form.html')
  269. # Modification with both POST and PUT (browser compatible)
  270. res = self.client.post('/edit/author/update/', {'name': 'Randall Munroe (xkcd)', 'slug': 'randall-munroe'})
  271. self.assertEqual(res.status_code, 302)
  272. self.assertRedirects(res, '/list/authors/')
  273. self.assertQuerysetEqual(Author.objects.values_list('name', flat=True), ['Randall Munroe (xkcd)'])
  274. @override_settings(ROOT_URLCONF='generic_views.urls')
  275. class DeleteViewTests(TestCase):
  276. @classmethod
  277. def setUpTestData(cls):
  278. cls.author = Author.objects.create(
  279. name='Randall Munroe',
  280. slug='randall-munroe',
  281. )
  282. def test_delete_by_post(self):
  283. res = self.client.get('/edit/author/%d/delete/' % self.author.pk)
  284. self.assertEqual(res.status_code, 200)
  285. self.assertEqual(res.context['object'], self.author)
  286. self.assertEqual(res.context['author'], self.author)
  287. self.assertTemplateUsed(res, 'generic_views/author_confirm_delete.html')
  288. # Deletion with POST
  289. res = self.client.post('/edit/author/%d/delete/' % self.author.pk)
  290. self.assertEqual(res.status_code, 302)
  291. self.assertRedirects(res, '/list/authors/')
  292. self.assertQuerysetEqual(Author.objects.all(), [])
  293. def test_delete_by_delete(self):
  294. # Deletion with browser compatible DELETE method
  295. res = self.client.delete('/edit/author/%d/delete/' % self.author.pk)
  296. self.assertEqual(res.status_code, 302)
  297. self.assertRedirects(res, '/list/authors/')
  298. self.assertQuerysetEqual(Author.objects.all(), [])
  299. def test_delete_with_redirect(self):
  300. res = self.client.post('/edit/author/%d/delete/redirect/' % self.author.pk)
  301. self.assertEqual(res.status_code, 302)
  302. self.assertRedirects(res, '/edit/authors/create/')
  303. self.assertQuerysetEqual(Author.objects.all(), [])
  304. def test_delete_with_interpolated_redirect(self):
  305. res = self.client.post('/edit/author/%d/delete/interpolate_redirect/' % self.author.pk)
  306. self.assertEqual(res.status_code, 302)
  307. self.assertRedirects(res, '/edit/authors/create/?deleted=%d' % self.author.pk)
  308. self.assertQuerysetEqual(Author.objects.all(), [])
  309. # Also test with escaped chars in URL
  310. a = Author.objects.create(**{'name': 'Randall Munroe', 'slug': 'randall-munroe'})
  311. res = self.client.post('/edit/author/{}/delete/interpolate_redirect_nonascii/'.format(a.pk))
  312. self.assertEqual(res.status_code, 302)
  313. self.assertRedirects(res, '/%C3%A9dit/authors/create/?deleted={}'.format(a.pk))
  314. def test_delete_with_special_properties(self):
  315. res = self.client.get('/edit/author/%d/delete/special/' % self.author.pk)
  316. self.assertEqual(res.status_code, 200)
  317. self.assertEqual(res.context['object'], self.author)
  318. self.assertEqual(res.context['thingy'], self.author)
  319. self.assertNotIn('author', res.context)
  320. self.assertTemplateUsed(res, 'generic_views/confirm_delete.html')
  321. res = self.client.post('/edit/author/%d/delete/special/' % self.author.pk)
  322. self.assertEqual(res.status_code, 302)
  323. self.assertRedirects(res, '/list/authors/')
  324. self.assertQuerysetEqual(Author.objects.all(), [])
  325. def test_delete_without_redirect(self):
  326. msg = 'No URL to redirect to. Provide a success_url.'
  327. with self.assertRaisesMessage(ImproperlyConfigured, msg):
  328. self.client.post('/edit/author/%d/delete/naive/' % self.author.pk)
  329. def test_delete_with_form_as_post(self):
  330. res = self.client.get('/edit/author/%d/delete/form/' % self.author.pk)
  331. self.assertEqual(res.status_code, 200)
  332. self.assertEqual(res.context['object'], self.author)
  333. self.assertEqual(res.context['author'], self.author)
  334. self.assertTemplateUsed(res, 'generic_views/author_confirm_delete.html')
  335. res = self.client.post(
  336. '/edit/author/%d/delete/form/' % self.author.pk, data={'confirm': True}
  337. )
  338. self.assertEqual(res.status_code, 302)
  339. self.assertRedirects(res, '/list/authors/')
  340. self.assertSequenceEqual(Author.objects.all(), [])
  341. def test_delete_with_form_as_post_with_validation_error(self):
  342. res = self.client.get('/edit/author/%d/delete/form/' % self.author.pk)
  343. self.assertEqual(res.status_code, 200)
  344. self.assertEqual(res.context['object'], self.author)
  345. self.assertEqual(res.context['author'], self.author)
  346. self.assertTemplateUsed(res, 'generic_views/author_confirm_delete.html')
  347. res = self.client.post('/edit/author/%d/delete/form/' % self.author.pk)
  348. self.assertEqual(res.status_code, 200)
  349. self.assertEqual(len(res.context_data['form'].errors), 2)
  350. self.assertEqual(
  351. res.context_data['form'].errors['__all__'],
  352. ['You must confirm the delete.'],
  353. )
  354. self.assertEqual(
  355. res.context_data['form'].errors['confirm'],
  356. ['This field is required.'],
  357. )