test_edit.py 20 KB

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