test_edit.py 19 KB

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