tests.py 61 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224
  1. from __future__ import absolute_import, unicode_literals
  2. import datetime
  3. import re
  4. from datetime import date
  5. from decimal import Decimal
  6. from django import forms
  7. from django.db import models
  8. from django.forms.models import (_get_foreign_key, inlineformset_factory,
  9. modelformset_factory)
  10. from django.test import TestCase, skipUnlessDBFeature
  11. from django.utils import six
  12. from .models import (Author, BetterAuthor, Book, BookWithCustomPK,
  13. BookWithOptionalAltEditor, AlternateBook, AuthorMeeting, CustomPrimaryKey,
  14. Place, Owner, Location, OwnerProfile, Restaurant, Product, Price,
  15. MexicanRestaurant, ClassyMexicanRestaurant, Repository, Revision,
  16. Person, Membership, Team, Player, Poet, Poem, Post)
  17. class DeletionTests(TestCase):
  18. def test_deletion(self):
  19. PoetFormSet = modelformset_factory(Poet, can_delete=True)
  20. poet = Poet.objects.create(name='test')
  21. data = {
  22. 'form-TOTAL_FORMS': '1',
  23. 'form-INITIAL_FORMS': '1',
  24. 'form-MAX_NUM_FORMS': '0',
  25. 'form-0-id': str(poet.pk),
  26. 'form-0-name': 'test',
  27. 'form-0-DELETE': 'on',
  28. }
  29. formset = PoetFormSet(data, queryset=Poet.objects.all())
  30. formset.save()
  31. self.assertTrue(formset.is_valid())
  32. self.assertEqual(Poet.objects.count(), 0)
  33. def test_add_form_deletion_when_invalid(self):
  34. """
  35. Make sure that an add form that is filled out, but marked for deletion
  36. doesn't cause validation errors.
  37. """
  38. PoetFormSet = modelformset_factory(Poet, can_delete=True)
  39. poet = Poet.objects.create(name='test')
  40. # One existing untouched and two new unvalid forms
  41. data = {
  42. 'form-TOTAL_FORMS': '3',
  43. 'form-INITIAL_FORMS': '1',
  44. 'form-MAX_NUM_FORMS': '0',
  45. 'form-0-id': six.text_type(poet.id),
  46. 'form-0-name': 'test',
  47. 'form-1-id': '',
  48. 'form-1-name': 'x' * 1000, # Too long
  49. 'form-1-id': six.text_type(poet.id), # Violate unique constraint
  50. 'form-1-name': 'test2',
  51. }
  52. formset = PoetFormSet(data, queryset=Poet.objects.all())
  53. # Make sure this form doesn't pass validation.
  54. self.assertEqual(formset.is_valid(), False)
  55. self.assertEqual(Poet.objects.count(), 1)
  56. # Then make sure that it *does* pass validation and delete the object,
  57. # even though the data in new forms aren't actually valid.
  58. data['form-0-DELETE'] = 'on'
  59. data['form-1-DELETE'] = 'on'
  60. data['form-2-DELETE'] = 'on'
  61. formset = PoetFormSet(data, queryset=Poet.objects.all())
  62. self.assertEqual(formset.is_valid(), True)
  63. formset.save()
  64. self.assertEqual(Poet.objects.count(), 0)
  65. def test_change_form_deletion_when_invalid(self):
  66. """
  67. Make sure that a change form that is filled out, but marked for deletion
  68. doesn't cause validation errors.
  69. """
  70. PoetFormSet = modelformset_factory(Poet, can_delete=True)
  71. poet = Poet.objects.create(name='test')
  72. data = {
  73. 'form-TOTAL_FORMS': '1',
  74. 'form-INITIAL_FORMS': '1',
  75. 'form-MAX_NUM_FORMS': '0',
  76. 'form-0-id': six.text_type(poet.id),
  77. 'form-0-name': 'x' * 1000,
  78. }
  79. formset = PoetFormSet(data, queryset=Poet.objects.all())
  80. # Make sure this form doesn't pass validation.
  81. self.assertEqual(formset.is_valid(), False)
  82. self.assertEqual(Poet.objects.count(), 1)
  83. # Then make sure that it *does* pass validation and delete the object,
  84. # even though the data isn't actually valid.
  85. data['form-0-DELETE'] = 'on'
  86. formset = PoetFormSet(data, queryset=Poet.objects.all())
  87. self.assertEqual(formset.is_valid(), True)
  88. formset.save()
  89. self.assertEqual(Poet.objects.count(), 0)
  90. class ModelFormsetTest(TestCase):
  91. def test_simple_save(self):
  92. qs = Author.objects.all()
  93. AuthorFormSet = modelformset_factory(Author, extra=3)
  94. formset = AuthorFormSet(queryset=qs)
  95. self.assertEqual(len(formset.forms), 3)
  96. self.assertHTMLEqual(formset.forms[0].as_p(),
  97. '<p><label for="id_form-0-name">Name:</label> <input id="id_form-0-name" type="text" name="form-0-name" maxlength="100" /><input type="hidden" name="form-0-id" id="id_form-0-id" /></p>')
  98. self.assertHTMLEqual(formset.forms[1].as_p(),
  99. '<p><label for="id_form-1-name">Name:</label> <input id="id_form-1-name" type="text" name="form-1-name" maxlength="100" /><input type="hidden" name="form-1-id" id="id_form-1-id" /></p>')
  100. self.assertHTMLEqual(formset.forms[2].as_p(),
  101. '<p><label for="id_form-2-name">Name:</label> <input id="id_form-2-name" type="text" name="form-2-name" maxlength="100" /><input type="hidden" name="form-2-id" id="id_form-2-id" /></p>')
  102. data = {
  103. 'form-TOTAL_FORMS': '3', # the number of forms rendered
  104. 'form-INITIAL_FORMS': '0', # the number of forms with initial data
  105. 'form-MAX_NUM_FORMS': '', # the max number of forms
  106. 'form-0-name': 'Charles Baudelaire',
  107. 'form-1-name': 'Arthur Rimbaud',
  108. 'form-2-name': '',
  109. }
  110. formset = AuthorFormSet(data=data, queryset=qs)
  111. self.assertTrue(formset.is_valid())
  112. saved = formset.save()
  113. self.assertEqual(len(saved), 2)
  114. author1, author2 = saved
  115. self.assertEqual(author1, Author.objects.get(name='Charles Baudelaire'))
  116. self.assertEqual(author2, Author.objects.get(name='Arthur Rimbaud'))
  117. authors = list(Author.objects.order_by('name'))
  118. self.assertEqual(authors, [author2, author1])
  119. # Gah! We forgot Paul Verlaine. Let's create a formset to edit the
  120. # existing authors with an extra form to add him. We *could* pass in a
  121. # queryset to restrict the Author objects we edit, but in this case
  122. # we'll use it to display them in alphabetical order by name.
  123. qs = Author.objects.order_by('name')
  124. AuthorFormSet = modelformset_factory(Author, extra=1, can_delete=False)
  125. formset = AuthorFormSet(queryset=qs)
  126. self.assertEqual(len(formset.forms), 3)
  127. self.assertHTMLEqual(formset.forms[0].as_p(),
  128. '<p><label for="id_form-0-name">Name:</label> <input id="id_form-0-name" type="text" name="form-0-name" value="Arthur Rimbaud" maxlength="100" /><input type="hidden" name="form-0-id" value="%d" id="id_form-0-id" /></p>' % author2.id)
  129. self.assertHTMLEqual(formset.forms[1].as_p(),
  130. '<p><label for="id_form-1-name">Name:</label> <input id="id_form-1-name" type="text" name="form-1-name" value="Charles Baudelaire" maxlength="100" /><input type="hidden" name="form-1-id" value="%d" id="id_form-1-id" /></p>' % author1.id)
  131. self.assertHTMLEqual(formset.forms[2].as_p(),
  132. '<p><label for="id_form-2-name">Name:</label> <input id="id_form-2-name" type="text" name="form-2-name" maxlength="100" /><input type="hidden" name="form-2-id" id="id_form-2-id" /></p>')
  133. data = {
  134. 'form-TOTAL_FORMS': '3', # the number of forms rendered
  135. 'form-INITIAL_FORMS': '2', # the number of forms with initial data
  136. 'form-MAX_NUM_FORMS': '', # the max number of forms
  137. 'form-0-id': str(author2.id),
  138. 'form-0-name': 'Arthur Rimbaud',
  139. 'form-1-id': str(author1.id),
  140. 'form-1-name': 'Charles Baudelaire',
  141. 'form-2-name': 'Paul Verlaine',
  142. }
  143. formset = AuthorFormSet(data=data, queryset=qs)
  144. self.assertTrue(formset.is_valid())
  145. # Only changed or new objects are returned from formset.save()
  146. saved = formset.save()
  147. self.assertEqual(len(saved), 1)
  148. author3 = saved[0]
  149. self.assertEqual(author3, Author.objects.get(name='Paul Verlaine'))
  150. authors = list(Author.objects.order_by('name'))
  151. self.assertEqual(authors, [author2, author1, author3])
  152. # This probably shouldn't happen, but it will. If an add form was
  153. # marked for deletion, make sure we don't save that form.
  154. qs = Author.objects.order_by('name')
  155. AuthorFormSet = modelformset_factory(Author, extra=1, can_delete=True)
  156. formset = AuthorFormSet(queryset=qs)
  157. self.assertEqual(len(formset.forms), 4)
  158. self.assertHTMLEqual(formset.forms[0].as_p(),
  159. '<p><label for="id_form-0-name">Name:</label> <input id="id_form-0-name" type="text" name="form-0-name" value="Arthur Rimbaud" maxlength="100" /></p>\n'
  160. '<p><label for="id_form-0-DELETE">Delete:</label> <input type="checkbox" name="form-0-DELETE" id="id_form-0-DELETE" /><input type="hidden" name="form-0-id" value="%d" id="id_form-0-id" /></p>' % author2.id)
  161. self.assertHTMLEqual(formset.forms[1].as_p(),
  162. '<p><label for="id_form-1-name">Name:</label> <input id="id_form-1-name" type="text" name="form-1-name" value="Charles Baudelaire" maxlength="100" /></p>\n'
  163. '<p><label for="id_form-1-DELETE">Delete:</label> <input type="checkbox" name="form-1-DELETE" id="id_form-1-DELETE" /><input type="hidden" name="form-1-id" value="%d" id="id_form-1-id" /></p>' % author1.id)
  164. self.assertHTMLEqual(formset.forms[2].as_p(),
  165. '<p><label for="id_form-2-name">Name:</label> <input id="id_form-2-name" type="text" name="form-2-name" value="Paul Verlaine" maxlength="100" /></p>\n'
  166. '<p><label for="id_form-2-DELETE">Delete:</label> <input type="checkbox" name="form-2-DELETE" id="id_form-2-DELETE" /><input type="hidden" name="form-2-id" value="%d" id="id_form-2-id" /></p>' % author3.id)
  167. self.assertHTMLEqual(formset.forms[3].as_p(),
  168. '<p><label for="id_form-3-name">Name:</label> <input id="id_form-3-name" type="text" name="form-3-name" maxlength="100" /></p>\n'
  169. '<p><label for="id_form-3-DELETE">Delete:</label> <input type="checkbox" name="form-3-DELETE" id="id_form-3-DELETE" /><input type="hidden" name="form-3-id" id="id_form-3-id" /></p>')
  170. data = {
  171. 'form-TOTAL_FORMS': '4', # the number of forms rendered
  172. 'form-INITIAL_FORMS': '3', # the number of forms with initial data
  173. 'form-MAX_NUM_FORMS': '', # the max number of forms
  174. 'form-0-id': str(author2.id),
  175. 'form-0-name': 'Arthur Rimbaud',
  176. 'form-1-id': str(author1.id),
  177. 'form-1-name': 'Charles Baudelaire',
  178. 'form-2-id': str(author3.id),
  179. 'form-2-name': 'Paul Verlaine',
  180. 'form-3-name': 'Walt Whitman',
  181. 'form-3-DELETE': 'on',
  182. }
  183. formset = AuthorFormSet(data=data, queryset=qs)
  184. self.assertTrue(formset.is_valid())
  185. # No objects were changed or saved so nothing will come back.
  186. self.assertEqual(formset.save(), [])
  187. authors = list(Author.objects.order_by('name'))
  188. self.assertEqual(authors, [author2, author1, author3])
  189. # Let's edit a record to ensure save only returns that one record.
  190. data = {
  191. 'form-TOTAL_FORMS': '4', # the number of forms rendered
  192. 'form-INITIAL_FORMS': '3', # the number of forms with initial data
  193. 'form-MAX_NUM_FORMS': '', # the max number of forms
  194. 'form-0-id': str(author2.id),
  195. 'form-0-name': 'Walt Whitman',
  196. 'form-1-id': str(author1.id),
  197. 'form-1-name': 'Charles Baudelaire',
  198. 'form-2-id': str(author3.id),
  199. 'form-2-name': 'Paul Verlaine',
  200. 'form-3-name': '',
  201. 'form-3-DELETE': '',
  202. }
  203. formset = AuthorFormSet(data=data, queryset=qs)
  204. self.assertTrue(formset.is_valid())
  205. # One record has changed.
  206. saved = formset.save()
  207. self.assertEqual(len(saved), 1)
  208. self.assertEqual(saved[0], Author.objects.get(name='Walt Whitman'))
  209. def test_commit_false(self):
  210. # Test the behavior of commit=False and save_m2m
  211. author1 = Author.objects.create(name='Charles Baudelaire')
  212. author2 = Author.objects.create(name='Paul Verlaine')
  213. author3 = Author.objects.create(name='Walt Whitman')
  214. meeting = AuthorMeeting.objects.create(created=date.today())
  215. meeting.authors = Author.objects.all()
  216. # create an Author instance to add to the meeting.
  217. author4 = Author.objects.create(name='John Steinbeck')
  218. AuthorMeetingFormSet = modelformset_factory(AuthorMeeting, extra=1, can_delete=True)
  219. data = {
  220. 'form-TOTAL_FORMS': '2', # the number of forms rendered
  221. 'form-INITIAL_FORMS': '1', # the number of forms with initial data
  222. 'form-MAX_NUM_FORMS': '', # the max number of forms
  223. 'form-0-id': str(meeting.id),
  224. 'form-0-name': '2nd Tuesday of the Week Meeting',
  225. 'form-0-authors': [author2.id, author1.id, author3.id, author4.id],
  226. 'form-1-name': '',
  227. 'form-1-authors': '',
  228. 'form-1-DELETE': '',
  229. }
  230. formset = AuthorMeetingFormSet(data=data, queryset=AuthorMeeting.objects.all())
  231. self.assertTrue(formset.is_valid())
  232. instances = formset.save(commit=False)
  233. for instance in instances:
  234. instance.created = date.today()
  235. instance.save()
  236. formset.save_m2m()
  237. self.assertQuerysetEqual(instances[0].authors.all(), [
  238. '<Author: Charles Baudelaire>',
  239. '<Author: John Steinbeck>',
  240. '<Author: Paul Verlaine>',
  241. '<Author: Walt Whitman>',
  242. ])
  243. def test_max_num(self):
  244. # Test the behavior of max_num with model formsets. It should allow
  245. # all existing related objects/inlines for a given object to be
  246. # displayed, but not allow the creation of new inlines beyond max_num.
  247. author1 = Author.objects.create(name='Charles Baudelaire')
  248. author2 = Author.objects.create(name='Paul Verlaine')
  249. author3 = Author.objects.create(name='Walt Whitman')
  250. qs = Author.objects.order_by('name')
  251. AuthorFormSet = modelformset_factory(Author, max_num=None, extra=3)
  252. formset = AuthorFormSet(queryset=qs)
  253. self.assertEqual(len(formset.forms), 6)
  254. self.assertEqual(len(formset.extra_forms), 3)
  255. AuthorFormSet = modelformset_factory(Author, max_num=4, extra=3)
  256. formset = AuthorFormSet(queryset=qs)
  257. self.assertEqual(len(formset.forms), 4)
  258. self.assertEqual(len(formset.extra_forms), 1)
  259. AuthorFormSet = modelformset_factory(Author, max_num=0, extra=3)
  260. formset = AuthorFormSet(queryset=qs)
  261. self.assertEqual(len(formset.forms), 3)
  262. self.assertEqual(len(formset.extra_forms), 0)
  263. AuthorFormSet = modelformset_factory(Author, max_num=None)
  264. formset = AuthorFormSet(queryset=qs)
  265. self.assertQuerysetEqual(formset.get_queryset(), [
  266. '<Author: Charles Baudelaire>',
  267. '<Author: Paul Verlaine>',
  268. '<Author: Walt Whitman>',
  269. ])
  270. AuthorFormSet = modelformset_factory(Author, max_num=0)
  271. formset = AuthorFormSet(queryset=qs)
  272. self.assertQuerysetEqual(formset.get_queryset(), [
  273. '<Author: Charles Baudelaire>',
  274. '<Author: Paul Verlaine>',
  275. '<Author: Walt Whitman>',
  276. ])
  277. AuthorFormSet = modelformset_factory(Author, max_num=4)
  278. formset = AuthorFormSet(queryset=qs)
  279. self.assertQuerysetEqual(formset.get_queryset(), [
  280. '<Author: Charles Baudelaire>',
  281. '<Author: Paul Verlaine>',
  282. '<Author: Walt Whitman>',
  283. ])
  284. def test_custom_save_method(self):
  285. class PoetForm(forms.ModelForm):
  286. def save(self, commit=True):
  287. # change the name to "Vladimir Mayakovsky" just to be a jerk.
  288. author = super(PoetForm, self).save(commit=False)
  289. author.name = "Vladimir Mayakovsky"
  290. if commit:
  291. author.save()
  292. return author
  293. PoetFormSet = modelformset_factory(Poet, form=PoetForm)
  294. data = {
  295. 'form-TOTAL_FORMS': '3', # the number of forms rendered
  296. 'form-INITIAL_FORMS': '0', # the number of forms with initial data
  297. 'form-MAX_NUM_FORMS': '', # the max number of forms
  298. 'form-0-name': 'Walt Whitman',
  299. 'form-1-name': 'Charles Baudelaire',
  300. 'form-2-name': '',
  301. }
  302. qs = Poet.objects.all()
  303. formset = PoetFormSet(data=data, queryset=qs)
  304. self.assertTrue(formset.is_valid())
  305. poets = formset.save()
  306. self.assertEqual(len(poets), 2)
  307. poet1, poet2 = poets
  308. self.assertEqual(poet1.name, 'Vladimir Mayakovsky')
  309. self.assertEqual(poet2.name, 'Vladimir Mayakovsky')
  310. def test_custom_form(self):
  311. """ Test that model_formset respects fields and exclude parameters of
  312. custom form
  313. """
  314. class PostForm1(forms.ModelForm):
  315. class Meta:
  316. model = Post
  317. fields = ('title', 'posted')
  318. class PostForm2(forms.ModelForm):
  319. class Meta:
  320. model = Post
  321. exclude = ('subtitle',)
  322. PostFormSet = modelformset_factory(Post, form=PostForm1)
  323. formset = PostFormSet()
  324. self.assertFalse("subtitle" in formset.forms[0].fields)
  325. PostFormSet = modelformset_factory(Post, form=PostForm2)
  326. formset = PostFormSet()
  327. self.assertFalse("subtitle" in formset.forms[0].fields)
  328. def test_model_inheritance(self):
  329. BetterAuthorFormSet = modelformset_factory(BetterAuthor)
  330. formset = BetterAuthorFormSet()
  331. self.assertEqual(len(formset.forms), 1)
  332. self.assertHTMLEqual(formset.forms[0].as_p(),
  333. '<p><label for="id_form-0-name">Name:</label> <input id="id_form-0-name" type="text" name="form-0-name" maxlength="100" /></p>\n'
  334. '<p><label for="id_form-0-write_speed">Write speed:</label> <input type="text" name="form-0-write_speed" id="id_form-0-write_speed" /><input type="hidden" name="form-0-author_ptr" id="id_form-0-author_ptr" /></p>')
  335. data = {
  336. 'form-TOTAL_FORMS': '1', # the number of forms rendered
  337. 'form-INITIAL_FORMS': '0', # the number of forms with initial data
  338. 'form-MAX_NUM_FORMS': '', # the max number of forms
  339. 'form-0-author_ptr': '',
  340. 'form-0-name': 'Ernest Hemingway',
  341. 'form-0-write_speed': '10',
  342. }
  343. formset = BetterAuthorFormSet(data)
  344. self.assertTrue(formset.is_valid())
  345. saved = formset.save()
  346. self.assertEqual(len(saved), 1)
  347. author1, = saved
  348. self.assertEqual(author1, BetterAuthor.objects.get(name='Ernest Hemingway'))
  349. hemingway_id = BetterAuthor.objects.get(name="Ernest Hemingway").pk
  350. formset = BetterAuthorFormSet()
  351. self.assertEqual(len(formset.forms), 2)
  352. self.assertHTMLEqual(formset.forms[0].as_p(),
  353. '<p><label for="id_form-0-name">Name:</label> <input id="id_form-0-name" type="text" name="form-0-name" value="Ernest Hemingway" maxlength="100" /></p>\n'
  354. '<p><label for="id_form-0-write_speed">Write speed:</label> <input type="text" name="form-0-write_speed" value="10" id="id_form-0-write_speed" /><input type="hidden" name="form-0-author_ptr" value="%d" id="id_form-0-author_ptr" /></p>' % hemingway_id)
  355. self.assertHTMLEqual(formset.forms[1].as_p(),
  356. '<p><label for="id_form-1-name">Name:</label> <input id="id_form-1-name" type="text" name="form-1-name" maxlength="100" /></p>\n'
  357. '<p><label for="id_form-1-write_speed">Write speed:</label> <input type="text" name="form-1-write_speed" id="id_form-1-write_speed" /><input type="hidden" name="form-1-author_ptr" id="id_form-1-author_ptr" /></p>')
  358. data = {
  359. 'form-TOTAL_FORMS': '2', # the number of forms rendered
  360. 'form-INITIAL_FORMS': '1', # the number of forms with initial data
  361. 'form-MAX_NUM_FORMS': '', # the max number of forms
  362. 'form-0-author_ptr': hemingway_id,
  363. 'form-0-name': 'Ernest Hemingway',
  364. 'form-0-write_speed': '10',
  365. 'form-1-author_ptr': '',
  366. 'form-1-name': '',
  367. 'form-1-write_speed': '',
  368. }
  369. formset = BetterAuthorFormSet(data)
  370. self.assertTrue(formset.is_valid())
  371. self.assertEqual(formset.save(), [])
  372. def test_inline_formsets(self):
  373. # We can also create a formset that is tied to a parent model. This is
  374. # how the admin system's edit inline functionality works.
  375. AuthorBooksFormSet = inlineformset_factory(Author, Book, can_delete=False, extra=3)
  376. author = Author.objects.create(name='Charles Baudelaire')
  377. formset = AuthorBooksFormSet(instance=author)
  378. self.assertEqual(len(formset.forms), 3)
  379. self.assertHTMLEqual(formset.forms[0].as_p(),
  380. '<p><label for="id_book_set-0-title">Title:</label> <input id="id_book_set-0-title" type="text" name="book_set-0-title" maxlength="100" /><input type="hidden" name="book_set-0-author" value="%d" id="id_book_set-0-author" /><input type="hidden" name="book_set-0-id" id="id_book_set-0-id" /></p>' % author.id)
  381. self.assertHTMLEqual(formset.forms[1].as_p(),
  382. '<p><label for="id_book_set-1-title">Title:</label> <input id="id_book_set-1-title" type="text" name="book_set-1-title" maxlength="100" /><input type="hidden" name="book_set-1-author" value="%d" id="id_book_set-1-author" /><input type="hidden" name="book_set-1-id" id="id_book_set-1-id" /></p>' % author.id)
  383. self.assertHTMLEqual(formset.forms[2].as_p(),
  384. '<p><label for="id_book_set-2-title">Title:</label> <input id="id_book_set-2-title" type="text" name="book_set-2-title" maxlength="100" /><input type="hidden" name="book_set-2-author" value="%d" id="id_book_set-2-author" /><input type="hidden" name="book_set-2-id" id="id_book_set-2-id" /></p>' % author.id)
  385. data = {
  386. 'book_set-TOTAL_FORMS': '3', # the number of forms rendered
  387. 'book_set-INITIAL_FORMS': '0', # the number of forms with initial data
  388. 'book_set-MAX_NUM_FORMS': '', # the max number of forms
  389. 'book_set-0-title': 'Les Fleurs du Mal',
  390. 'book_set-1-title': '',
  391. 'book_set-2-title': '',
  392. }
  393. formset = AuthorBooksFormSet(data, instance=author)
  394. self.assertTrue(formset.is_valid())
  395. saved = formset.save()
  396. self.assertEqual(len(saved), 1)
  397. book1, = saved
  398. self.assertEqual(book1, Book.objects.get(title='Les Fleurs du Mal'))
  399. self.assertQuerysetEqual(author.book_set.all(), ['<Book: Les Fleurs du Mal>'])
  400. # Now that we've added a book to Charles Baudelaire, let's try adding
  401. # another one. This time though, an edit form will be available for
  402. # every existing book.
  403. AuthorBooksFormSet = inlineformset_factory(Author, Book, can_delete=False, extra=2)
  404. author = Author.objects.get(name='Charles Baudelaire')
  405. formset = AuthorBooksFormSet(instance=author)
  406. self.assertEqual(len(formset.forms), 3)
  407. self.assertHTMLEqual(formset.forms[0].as_p(),
  408. '<p><label for="id_book_set-0-title">Title:</label> <input id="id_book_set-0-title" type="text" name="book_set-0-title" value="Les Fleurs du Mal" maxlength="100" /><input type="hidden" name="book_set-0-author" value="%d" id="id_book_set-0-author" /><input type="hidden" name="book_set-0-id" value="%d" id="id_book_set-0-id" /></p>' % (author.id, book1.id))
  409. self.assertHTMLEqual(formset.forms[1].as_p(),
  410. '<p><label for="id_book_set-1-title">Title:</label> <input id="id_book_set-1-title" type="text" name="book_set-1-title" maxlength="100" /><input type="hidden" name="book_set-1-author" value="%d" id="id_book_set-1-author" /><input type="hidden" name="book_set-1-id" id="id_book_set-1-id" /></p>' % author.id)
  411. self.assertHTMLEqual(formset.forms[2].as_p(),
  412. '<p><label for="id_book_set-2-title">Title:</label> <input id="id_book_set-2-title" type="text" name="book_set-2-title" maxlength="100" /><input type="hidden" name="book_set-2-author" value="%d" id="id_book_set-2-author" /><input type="hidden" name="book_set-2-id" id="id_book_set-2-id" /></p>' % author.id)
  413. data = {
  414. 'book_set-TOTAL_FORMS': '3', # the number of forms rendered
  415. 'book_set-INITIAL_FORMS': '1', # the number of forms with initial data
  416. 'book_set-MAX_NUM_FORMS': '', # the max number of forms
  417. 'book_set-0-id': str(book1.id),
  418. 'book_set-0-title': 'Les Fleurs du Mal',
  419. 'book_set-1-title': 'Les Paradis Artificiels',
  420. 'book_set-2-title': '',
  421. }
  422. formset = AuthorBooksFormSet(data, instance=author)
  423. self.assertTrue(formset.is_valid())
  424. saved = formset.save()
  425. self.assertEqual(len(saved), 1)
  426. book2, = saved
  427. self.assertEqual(book2, Book.objects.get(title='Les Paradis Artificiels'))
  428. # As you can see, 'Les Paradis Artificiels' is now a book belonging to
  429. # Charles Baudelaire.
  430. self.assertQuerysetEqual(author.book_set.order_by('title'), [
  431. '<Book: Les Fleurs du Mal>',
  432. '<Book: Les Paradis Artificiels>',
  433. ])
  434. def test_inline_formsets_save_as_new(self):
  435. # The save_as_new parameter lets you re-associate the data to a new
  436. # instance. This is used in the admin for save_as functionality.
  437. AuthorBooksFormSet = inlineformset_factory(Author, Book, can_delete=False, extra=2)
  438. author = Author.objects.create(name='Charles Baudelaire')
  439. data = {
  440. 'book_set-TOTAL_FORMS': '3', # the number of forms rendered
  441. 'book_set-INITIAL_FORMS': '2', # the number of forms with initial data
  442. 'book_set-MAX_NUM_FORMS': '', # the max number of forms
  443. 'book_set-0-id': '1',
  444. 'book_set-0-title': 'Les Fleurs du Mal',
  445. 'book_set-1-id': '2',
  446. 'book_set-1-title': 'Les Paradis Artificiels',
  447. 'book_set-2-title': '',
  448. }
  449. formset = AuthorBooksFormSet(data, instance=Author(), save_as_new=True)
  450. self.assertTrue(formset.is_valid())
  451. new_author = Author.objects.create(name='Charles Baudelaire')
  452. formset = AuthorBooksFormSet(data, instance=new_author, save_as_new=True)
  453. saved = formset.save()
  454. self.assertEqual(len(saved), 2)
  455. book1, book2 = saved
  456. self.assertEqual(book1.title, 'Les Fleurs du Mal')
  457. self.assertEqual(book2.title, 'Les Paradis Artificiels')
  458. # Test using a custom prefix on an inline formset.
  459. formset = AuthorBooksFormSet(prefix="test")
  460. self.assertEqual(len(formset.forms), 2)
  461. self.assertHTMLEqual(formset.forms[0].as_p(),
  462. '<p><label for="id_test-0-title">Title:</label> <input id="id_test-0-title" type="text" name="test-0-title" maxlength="100" /><input type="hidden" name="test-0-author" id="id_test-0-author" /><input type="hidden" name="test-0-id" id="id_test-0-id" /></p>')
  463. self.assertHTMLEqual(formset.forms[1].as_p(),
  464. '<p><label for="id_test-1-title">Title:</label> <input id="id_test-1-title" type="text" name="test-1-title" maxlength="100" /><input type="hidden" name="test-1-author" id="id_test-1-author" /><input type="hidden" name="test-1-id" id="id_test-1-id" /></p>')
  465. def test_inline_formsets_with_custom_pk(self):
  466. # Test inline formsets where the inline-edited object has a custom
  467. # primary key that is not the fk to the parent object.
  468. AuthorBooksFormSet2 = inlineformset_factory(Author, BookWithCustomPK, can_delete=False, extra=1)
  469. author = Author.objects.create(pk=1, name='Charles Baudelaire')
  470. formset = AuthorBooksFormSet2(instance=author)
  471. self.assertEqual(len(formset.forms), 1)
  472. self.assertHTMLEqual(formset.forms[0].as_p(),
  473. '<p><label for="id_bookwithcustompk_set-0-my_pk">My pk:</label> <input type="text" name="bookwithcustompk_set-0-my_pk" id="id_bookwithcustompk_set-0-my_pk" /></p>\n'
  474. '<p><label for="id_bookwithcustompk_set-0-title">Title:</label> <input id="id_bookwithcustompk_set-0-title" type="text" name="bookwithcustompk_set-0-title" maxlength="100" /><input type="hidden" name="bookwithcustompk_set-0-author" value="1" id="id_bookwithcustompk_set-0-author" /></p>')
  475. data = {
  476. 'bookwithcustompk_set-TOTAL_FORMS': '1', # the number of forms rendered
  477. 'bookwithcustompk_set-INITIAL_FORMS': '0', # the number of forms with initial data
  478. 'bookwithcustompk_set-MAX_NUM_FORMS': '', # the max number of forms
  479. 'bookwithcustompk_set-0-my_pk': '77777',
  480. 'bookwithcustompk_set-0-title': 'Les Fleurs du Mal',
  481. }
  482. formset = AuthorBooksFormSet2(data, instance=author)
  483. self.assertTrue(formset.is_valid())
  484. saved = formset.save()
  485. self.assertEqual(len(saved), 1)
  486. book1, = saved
  487. self.assertEqual(book1.pk, 77777)
  488. book1 = author.bookwithcustompk_set.get()
  489. self.assertEqual(book1.title, 'Les Fleurs du Mal')
  490. def test_inline_formsets_with_multi_table_inheritance(self):
  491. # Test inline formsets where the inline-edited object uses multi-table
  492. # inheritance, thus has a non AutoField yet auto-created primary key.
  493. AuthorBooksFormSet3 = inlineformset_factory(Author, AlternateBook, can_delete=False, extra=1)
  494. author = Author.objects.create(pk=1, name='Charles Baudelaire')
  495. formset = AuthorBooksFormSet3(instance=author)
  496. self.assertEqual(len(formset.forms), 1)
  497. self.assertHTMLEqual(formset.forms[0].as_p(),
  498. '<p><label for="id_alternatebook_set-0-title">Title:</label> <input id="id_alternatebook_set-0-title" type="text" name="alternatebook_set-0-title" maxlength="100" /></p>\n'
  499. '<p><label for="id_alternatebook_set-0-notes">Notes:</label> <input id="id_alternatebook_set-0-notes" type="text" name="alternatebook_set-0-notes" maxlength="100" /><input type="hidden" name="alternatebook_set-0-author" value="1" id="id_alternatebook_set-0-author" /><input type="hidden" name="alternatebook_set-0-book_ptr" id="id_alternatebook_set-0-book_ptr" /></p>')
  500. data = {
  501. 'alternatebook_set-TOTAL_FORMS': '1', # the number of forms rendered
  502. 'alternatebook_set-INITIAL_FORMS': '0', # the number of forms with initial data
  503. 'alternatebook_set-MAX_NUM_FORMS': '', # the max number of forms
  504. 'alternatebook_set-0-title': 'Flowers of Evil',
  505. 'alternatebook_set-0-notes': 'English translation of Les Fleurs du Mal'
  506. }
  507. formset = AuthorBooksFormSet3(data, instance=author)
  508. self.assertTrue(formset.is_valid())
  509. saved = formset.save()
  510. self.assertEqual(len(saved), 1)
  511. book1, = saved
  512. self.assertEqual(book1.title, 'Flowers of Evil')
  513. self.assertEqual(book1.notes, 'English translation of Les Fleurs du Mal')
  514. @skipUnlessDBFeature('ignores_nulls_in_unique_constraints')
  515. def test_inline_formsets_with_nullable_unique_together(self):
  516. # Test inline formsets where the inline-edited object has a
  517. # unique_together constraint with a nullable member
  518. AuthorBooksFormSet4 = inlineformset_factory(Author, BookWithOptionalAltEditor, can_delete=False, extra=2)
  519. author = Author.objects.create(pk=1, name='Charles Baudelaire')
  520. data = {
  521. 'bookwithoptionalalteditor_set-TOTAL_FORMS': '2', # the number of forms rendered
  522. 'bookwithoptionalalteditor_set-INITIAL_FORMS': '0', # the number of forms with initial data
  523. 'bookwithoptionalalteditor_set-MAX_NUM_FORMS': '', # the max number of forms
  524. 'bookwithoptionalalteditor_set-0-author': '1',
  525. 'bookwithoptionalalteditor_set-0-title': 'Les Fleurs du Mal',
  526. 'bookwithoptionalalteditor_set-1-author': '1',
  527. 'bookwithoptionalalteditor_set-1-title': 'Les Fleurs du Mal',
  528. }
  529. formset = AuthorBooksFormSet4(data, instance=author)
  530. self.assertTrue(formset.is_valid())
  531. saved = formset.save()
  532. self.assertEqual(len(saved), 2)
  533. book1, book2 = saved
  534. self.assertEqual(book1.author_id, 1)
  535. self.assertEqual(book1.title, 'Les Fleurs du Mal')
  536. self.assertEqual(book2.author_id, 1)
  537. self.assertEqual(book2.title, 'Les Fleurs du Mal')
  538. def test_inline_formsets_with_custom_save_method(self):
  539. AuthorBooksFormSet = inlineformset_factory(Author, Book, can_delete=False, extra=2)
  540. author = Author.objects.create(pk=1, name='Charles Baudelaire')
  541. book1 = Book.objects.create(pk=1, author=author, title='Les Paradis Artificiels')
  542. book2 = Book.objects.create(pk=2, author=author, title='Les Fleurs du Mal')
  543. book3 = Book.objects.create(pk=3, author=author, title='Flowers of Evil')
  544. class PoemForm(forms.ModelForm):
  545. def save(self, commit=True):
  546. # change the name to "Brooklyn Bridge" just to be a jerk.
  547. poem = super(PoemForm, self).save(commit=False)
  548. poem.name = "Brooklyn Bridge"
  549. if commit:
  550. poem.save()
  551. return poem
  552. PoemFormSet = inlineformset_factory(Poet, Poem, form=PoemForm)
  553. data = {
  554. 'poem_set-TOTAL_FORMS': '3', # the number of forms rendered
  555. 'poem_set-INITIAL_FORMS': '0', # the number of forms with initial data
  556. 'poem_set-MAX_NUM_FORMS': '', # the max number of forms
  557. 'poem_set-0-name': 'The Cloud in Trousers',
  558. 'poem_set-1-name': 'I',
  559. 'poem_set-2-name': '',
  560. }
  561. poet = Poet.objects.create(name='Vladimir Mayakovsky')
  562. formset = PoemFormSet(data=data, instance=poet)
  563. self.assertTrue(formset.is_valid())
  564. saved = formset.save()
  565. self.assertEqual(len(saved), 2)
  566. poem1, poem2 = saved
  567. self.assertEqual(poem1.name, 'Brooklyn Bridge')
  568. self.assertEqual(poem2.name, 'Brooklyn Bridge')
  569. # We can provide a custom queryset to our InlineFormSet:
  570. custom_qs = Book.objects.order_by('-title')
  571. formset = AuthorBooksFormSet(instance=author, queryset=custom_qs)
  572. self.assertEqual(len(formset.forms), 5)
  573. self.assertHTMLEqual(formset.forms[0].as_p(),
  574. '<p><label for="id_book_set-0-title">Title:</label> <input id="id_book_set-0-title" type="text" name="book_set-0-title" value="Les Paradis Artificiels" maxlength="100" /><input type="hidden" name="book_set-0-author" value="1" id="id_book_set-0-author" /><input type="hidden" name="book_set-0-id" value="1" id="id_book_set-0-id" /></p>')
  575. self.assertHTMLEqual(formset.forms[1].as_p(),
  576. '<p><label for="id_book_set-1-title">Title:</label> <input id="id_book_set-1-title" type="text" name="book_set-1-title" value="Les Fleurs du Mal" maxlength="100" /><input type="hidden" name="book_set-1-author" value="1" id="id_book_set-1-author" /><input type="hidden" name="book_set-1-id" value="2" id="id_book_set-1-id" /></p>')
  577. self.assertHTMLEqual(formset.forms[2].as_p(),
  578. '<p><label for="id_book_set-2-title">Title:</label> <input id="id_book_set-2-title" type="text" name="book_set-2-title" value="Flowers of Evil" maxlength="100" /><input type="hidden" name="book_set-2-author" value="1" id="id_book_set-2-author" /><input type="hidden" name="book_set-2-id" value="3" id="id_book_set-2-id" /></p>')
  579. self.assertHTMLEqual(formset.forms[3].as_p(),
  580. '<p><label for="id_book_set-3-title">Title:</label> <input id="id_book_set-3-title" type="text" name="book_set-3-title" maxlength="100" /><input type="hidden" name="book_set-3-author" value="1" id="id_book_set-3-author" /><input type="hidden" name="book_set-3-id" id="id_book_set-3-id" /></p>')
  581. self.assertHTMLEqual(formset.forms[4].as_p(),
  582. '<p><label for="id_book_set-4-title">Title:</label> <input id="id_book_set-4-title" type="text" name="book_set-4-title" maxlength="100" /><input type="hidden" name="book_set-4-author" value="1" id="id_book_set-4-author" /><input type="hidden" name="book_set-4-id" id="id_book_set-4-id" /></p>')
  583. data = {
  584. 'book_set-TOTAL_FORMS': '5', # the number of forms rendered
  585. 'book_set-INITIAL_FORMS': '3', # the number of forms with initial data
  586. 'book_set-MAX_NUM_FORMS': '', # the max number of forms
  587. 'book_set-0-id': str(book1.id),
  588. 'book_set-0-title': 'Les Paradis Artificiels',
  589. 'book_set-1-id': str(book2.id),
  590. 'book_set-1-title': 'Les Fleurs du Mal',
  591. 'book_set-2-id': str(book3.id),
  592. 'book_set-2-title': 'Flowers of Evil',
  593. 'book_set-3-title': 'Revue des deux mondes',
  594. 'book_set-4-title': '',
  595. }
  596. formset = AuthorBooksFormSet(data, instance=author, queryset=custom_qs)
  597. self.assertTrue(formset.is_valid())
  598. custom_qs = Book.objects.filter(title__startswith='F')
  599. formset = AuthorBooksFormSet(instance=author, queryset=custom_qs)
  600. self.assertHTMLEqual(formset.forms[0].as_p(),
  601. '<p><label for="id_book_set-0-title">Title:</label> <input id="id_book_set-0-title" type="text" name="book_set-0-title" value="Flowers of Evil" maxlength="100" /><input type="hidden" name="book_set-0-author" value="1" id="id_book_set-0-author" /><input type="hidden" name="book_set-0-id" value="3" id="id_book_set-0-id" /></p>')
  602. self.assertHTMLEqual(formset.forms[1].as_p(),
  603. '<p><label for="id_book_set-1-title">Title:</label> <input id="id_book_set-1-title" type="text" name="book_set-1-title" maxlength="100" /><input type="hidden" name="book_set-1-author" value="1" id="id_book_set-1-author" /><input type="hidden" name="book_set-1-id" id="id_book_set-1-id" /></p>')
  604. self.assertHTMLEqual(formset.forms[2].as_p(),
  605. '<p><label for="id_book_set-2-title">Title:</label> <input id="id_book_set-2-title" type="text" name="book_set-2-title" maxlength="100" /><input type="hidden" name="book_set-2-author" value="1" id="id_book_set-2-author" /><input type="hidden" name="book_set-2-id" id="id_book_set-2-id" /></p>')
  606. data = {
  607. 'book_set-TOTAL_FORMS': '3', # the number of forms rendered
  608. 'book_set-INITIAL_FORMS': '1', # the number of forms with initial data
  609. 'book_set-MAX_NUM_FORMS': '', # the max number of forms
  610. 'book_set-0-id': str(book3.id),
  611. 'book_set-0-title': 'Flowers of Evil',
  612. 'book_set-1-title': 'Revue des deux mondes',
  613. 'book_set-2-title': '',
  614. }
  615. formset = AuthorBooksFormSet(data, instance=author, queryset=custom_qs)
  616. self.assertTrue(formset.is_valid())
  617. def test_custom_pk(self):
  618. # We need to ensure that it is displayed
  619. CustomPrimaryKeyFormSet = modelformset_factory(CustomPrimaryKey)
  620. formset = CustomPrimaryKeyFormSet()
  621. self.assertEqual(len(formset.forms), 1)
  622. self.assertHTMLEqual(formset.forms[0].as_p(),
  623. '<p><label for="id_form-0-my_pk">My pk:</label> <input id="id_form-0-my_pk" type="text" name="form-0-my_pk" maxlength="10" /></p>\n'
  624. '<p><label for="id_form-0-some_field">Some field:</label> <input id="id_form-0-some_field" type="text" name="form-0-some_field" maxlength="100" /></p>')
  625. # Custom primary keys with ForeignKey, OneToOneField and AutoField ############
  626. place = Place.objects.create(pk=1, name='Giordanos', city='Chicago')
  627. FormSet = inlineformset_factory(Place, Owner, extra=2, can_delete=False)
  628. formset = FormSet(instance=place)
  629. self.assertEqual(len(formset.forms), 2)
  630. self.assertHTMLEqual(formset.forms[0].as_p(),
  631. '<p><label for="id_owner_set-0-name">Name:</label> <input id="id_owner_set-0-name" type="text" name="owner_set-0-name" maxlength="100" /><input type="hidden" name="owner_set-0-place" value="1" id="id_owner_set-0-place" /><input type="hidden" name="owner_set-0-auto_id" id="id_owner_set-0-auto_id" /></p>')
  632. self.assertHTMLEqual(formset.forms[1].as_p(),
  633. '<p><label for="id_owner_set-1-name">Name:</label> <input id="id_owner_set-1-name" type="text" name="owner_set-1-name" maxlength="100" /><input type="hidden" name="owner_set-1-place" value="1" id="id_owner_set-1-place" /><input type="hidden" name="owner_set-1-auto_id" id="id_owner_set-1-auto_id" /></p>')
  634. data = {
  635. 'owner_set-TOTAL_FORMS': '2',
  636. 'owner_set-INITIAL_FORMS': '0',
  637. 'owner_set-MAX_NUM_FORMS': '',
  638. 'owner_set-0-auto_id': '',
  639. 'owner_set-0-name': 'Joe Perry',
  640. 'owner_set-1-auto_id': '',
  641. 'owner_set-1-name': '',
  642. }
  643. formset = FormSet(data, instance=place)
  644. self.assertTrue(formset.is_valid())
  645. saved = formset.save()
  646. self.assertEqual(len(saved), 1)
  647. owner1, = saved
  648. self.assertEqual(owner1.name, 'Joe Perry')
  649. self.assertEqual(owner1.place.name, 'Giordanos')
  650. formset = FormSet(instance=place)
  651. self.assertEqual(len(formset.forms), 3)
  652. self.assertHTMLEqual(formset.forms[0].as_p(),
  653. '<p><label for="id_owner_set-0-name">Name:</label> <input id="id_owner_set-0-name" type="text" name="owner_set-0-name" value="Joe Perry" maxlength="100" /><input type="hidden" name="owner_set-0-place" value="1" id="id_owner_set-0-place" /><input type="hidden" name="owner_set-0-auto_id" value="%d" id="id_owner_set-0-auto_id" /></p>'
  654. % owner1.auto_id)
  655. self.assertHTMLEqual(formset.forms[1].as_p(),
  656. '<p><label for="id_owner_set-1-name">Name:</label> <input id="id_owner_set-1-name" type="text" name="owner_set-1-name" maxlength="100" /><input type="hidden" name="owner_set-1-place" value="1" id="id_owner_set-1-place" /><input type="hidden" name="owner_set-1-auto_id" id="id_owner_set-1-auto_id" /></p>')
  657. self.assertHTMLEqual(formset.forms[2].as_p(),
  658. '<p><label for="id_owner_set-2-name">Name:</label> <input id="id_owner_set-2-name" type="text" name="owner_set-2-name" maxlength="100" /><input type="hidden" name="owner_set-2-place" value="1" id="id_owner_set-2-place" /><input type="hidden" name="owner_set-2-auto_id" id="id_owner_set-2-auto_id" /></p>')
  659. data = {
  660. 'owner_set-TOTAL_FORMS': '3',
  661. 'owner_set-INITIAL_FORMS': '1',
  662. 'owner_set-MAX_NUM_FORMS': '',
  663. 'owner_set-0-auto_id': six.text_type(owner1.auto_id),
  664. 'owner_set-0-name': 'Joe Perry',
  665. 'owner_set-1-auto_id': '',
  666. 'owner_set-1-name': 'Jack Berry',
  667. 'owner_set-2-auto_id': '',
  668. 'owner_set-2-name': '',
  669. }
  670. formset = FormSet(data, instance=place)
  671. self.assertTrue(formset.is_valid())
  672. saved = formset.save()
  673. self.assertEqual(len(saved), 1)
  674. owner2, = saved
  675. self.assertEqual(owner2.name, 'Jack Berry')
  676. self.assertEqual(owner2.place.name, 'Giordanos')
  677. # Ensure a custom primary key that is a ForeignKey or OneToOneField get rendered for the user to choose.
  678. FormSet = modelformset_factory(OwnerProfile)
  679. formset = FormSet()
  680. self.assertHTMLEqual(formset.forms[0].as_p(),
  681. '<p><label for="id_form-0-owner">Owner:</label> <select name="form-0-owner" id="id_form-0-owner">\n'
  682. '<option value="" selected="selected">---------</option>\n'
  683. '<option value="%d">Joe Perry at Giordanos</option>\n'
  684. '<option value="%d">Jack Berry at Giordanos</option>\n'
  685. '</select></p>\n'
  686. '<p><label for="id_form-0-age">Age:</label> <input type="text" name="form-0-age" id="id_form-0-age" /></p>'
  687. % (owner1.auto_id, owner2.auto_id))
  688. owner1 = Owner.objects.get(name='Joe Perry')
  689. FormSet = inlineformset_factory(Owner, OwnerProfile, max_num=1, can_delete=False)
  690. self.assertEqual(FormSet.max_num, 1)
  691. formset = FormSet(instance=owner1)
  692. self.assertEqual(len(formset.forms), 1)
  693. self.assertHTMLEqual(formset.forms[0].as_p(),
  694. '<p><label for="id_ownerprofile-0-age">Age:</label> <input type="text" name="ownerprofile-0-age" id="id_ownerprofile-0-age" /><input type="hidden" name="ownerprofile-0-owner" value="%d" id="id_ownerprofile-0-owner" /></p>'
  695. % owner1.auto_id)
  696. data = {
  697. 'ownerprofile-TOTAL_FORMS': '1',
  698. 'ownerprofile-INITIAL_FORMS': '0',
  699. 'ownerprofile-MAX_NUM_FORMS': '1',
  700. 'ownerprofile-0-owner': '',
  701. 'ownerprofile-0-age': '54',
  702. }
  703. formset = FormSet(data, instance=owner1)
  704. self.assertTrue(formset.is_valid())
  705. saved = formset.save()
  706. self.assertEqual(len(saved), 1)
  707. profile1, = saved
  708. self.assertEqual(profile1.owner, owner1)
  709. self.assertEqual(profile1.age, 54)
  710. formset = FormSet(instance=owner1)
  711. self.assertEqual(len(formset.forms), 1)
  712. self.assertHTMLEqual(formset.forms[0].as_p(),
  713. '<p><label for="id_ownerprofile-0-age">Age:</label> <input type="text" name="ownerprofile-0-age" value="54" id="id_ownerprofile-0-age" /><input type="hidden" name="ownerprofile-0-owner" value="%d" id="id_ownerprofile-0-owner" /></p>'
  714. % owner1.auto_id)
  715. data = {
  716. 'ownerprofile-TOTAL_FORMS': '1',
  717. 'ownerprofile-INITIAL_FORMS': '1',
  718. 'ownerprofile-MAX_NUM_FORMS': '1',
  719. 'ownerprofile-0-owner': six.text_type(owner1.auto_id),
  720. 'ownerprofile-0-age': '55',
  721. }
  722. formset = FormSet(data, instance=owner1)
  723. self.assertTrue(formset.is_valid())
  724. saved = formset.save()
  725. self.assertEqual(len(saved), 1)
  726. profile1, = saved
  727. self.assertEqual(profile1.owner, owner1)
  728. self.assertEqual(profile1.age, 55)
  729. def test_unique_true_enforces_max_num_one(self):
  730. # ForeignKey with unique=True should enforce max_num=1
  731. place = Place.objects.create(pk=1, name='Giordanos', city='Chicago')
  732. FormSet = inlineformset_factory(Place, Location, can_delete=False)
  733. self.assertEqual(FormSet.max_num, 1)
  734. formset = FormSet(instance=place)
  735. self.assertEqual(len(formset.forms), 1)
  736. self.assertHTMLEqual(formset.forms[0].as_p(),
  737. '<p><label for="id_location_set-0-lat">Lat:</label> <input id="id_location_set-0-lat" type="text" name="location_set-0-lat" maxlength="100" /></p>\n'
  738. '<p><label for="id_location_set-0-lon">Lon:</label> <input id="id_location_set-0-lon" type="text" name="location_set-0-lon" maxlength="100" /><input type="hidden" name="location_set-0-place" value="1" id="id_location_set-0-place" /><input type="hidden" name="location_set-0-id" id="id_location_set-0-id" /></p>')
  739. def test_foreign_keys_in_parents(self):
  740. self.assertEqual(type(_get_foreign_key(Restaurant, Owner)), models.ForeignKey)
  741. self.assertEqual(type(_get_foreign_key(MexicanRestaurant, Owner)), models.ForeignKey)
  742. def test_unique_validation(self):
  743. FormSet = modelformset_factory(Product, extra=1)
  744. data = {
  745. 'form-TOTAL_FORMS': '1',
  746. 'form-INITIAL_FORMS': '0',
  747. 'form-MAX_NUM_FORMS': '',
  748. 'form-0-slug': 'car-red',
  749. }
  750. formset = FormSet(data)
  751. self.assertTrue(formset.is_valid())
  752. saved = formset.save()
  753. self.assertEqual(len(saved), 1)
  754. product1, = saved
  755. self.assertEqual(product1.slug, 'car-red')
  756. data = {
  757. 'form-TOTAL_FORMS': '1',
  758. 'form-INITIAL_FORMS': '0',
  759. 'form-MAX_NUM_FORMS': '',
  760. 'form-0-slug': 'car-red',
  761. }
  762. formset = FormSet(data)
  763. self.assertFalse(formset.is_valid())
  764. self.assertEqual(formset.errors, [{'slug': ['Product with this Slug already exists.']}])
  765. def test_unique_together_validation(self):
  766. FormSet = modelformset_factory(Price, extra=1)
  767. data = {
  768. 'form-TOTAL_FORMS': '1',
  769. 'form-INITIAL_FORMS': '0',
  770. 'form-MAX_NUM_FORMS': '',
  771. 'form-0-price': '12.00',
  772. 'form-0-quantity': '1',
  773. }
  774. formset = FormSet(data)
  775. self.assertTrue(formset.is_valid())
  776. saved = formset.save()
  777. self.assertEqual(len(saved), 1)
  778. price1, = saved
  779. self.assertEqual(price1.price, Decimal('12.00'))
  780. self.assertEqual(price1.quantity, 1)
  781. data = {
  782. 'form-TOTAL_FORMS': '1',
  783. 'form-INITIAL_FORMS': '0',
  784. 'form-MAX_NUM_FORMS': '',
  785. 'form-0-price': '12.00',
  786. 'form-0-quantity': '1',
  787. }
  788. formset = FormSet(data)
  789. self.assertFalse(formset.is_valid())
  790. self.assertEqual(formset.errors, [{'__all__': ['Price with this Price and Quantity already exists.']}])
  791. def test_unique_together_with_inlineformset_factory(self):
  792. # Also see bug #8882.
  793. repository = Repository.objects.create(name='Test Repo')
  794. FormSet = inlineformset_factory(Repository, Revision, extra=1)
  795. data = {
  796. 'revision_set-TOTAL_FORMS': '1',
  797. 'revision_set-INITIAL_FORMS': '0',
  798. 'revision_set-MAX_NUM_FORMS': '',
  799. 'revision_set-0-repository': repository.pk,
  800. 'revision_set-0-revision': '146239817507f148d448db38840db7c3cbf47c76',
  801. 'revision_set-0-DELETE': '',
  802. }
  803. formset = FormSet(data, instance=repository)
  804. self.assertTrue(formset.is_valid())
  805. saved = formset.save()
  806. self.assertEqual(len(saved), 1)
  807. revision1, = saved
  808. self.assertEqual(revision1.repository, repository)
  809. self.assertEqual(revision1.revision, '146239817507f148d448db38840db7c3cbf47c76')
  810. # attempt to save the same revision against against the same repo.
  811. data = {
  812. 'revision_set-TOTAL_FORMS': '1',
  813. 'revision_set-INITIAL_FORMS': '0',
  814. 'revision_set-MAX_NUM_FORMS': '',
  815. 'revision_set-0-repository': repository.pk,
  816. 'revision_set-0-revision': '146239817507f148d448db38840db7c3cbf47c76',
  817. 'revision_set-0-DELETE': '',
  818. }
  819. formset = FormSet(data, instance=repository)
  820. self.assertFalse(formset.is_valid())
  821. self.assertEqual(formset.errors, [{'__all__': ['Revision with this Repository and Revision already exists.']}])
  822. # unique_together with inlineformset_factory with overridden form fields
  823. # Also see #9494
  824. FormSet = inlineformset_factory(Repository, Revision, fields=('revision',), extra=1)
  825. data = {
  826. 'revision_set-TOTAL_FORMS': '1',
  827. 'revision_set-INITIAL_FORMS': '0',
  828. 'revision_set-MAX_NUM_FORMS': '',
  829. 'revision_set-0-repository': repository.pk,
  830. 'revision_set-0-revision': '146239817507f148d448db38840db7c3cbf47c76',
  831. 'revision_set-0-DELETE': '',
  832. }
  833. formset = FormSet(data, instance=repository)
  834. self.assertFalse(formset.is_valid())
  835. def test_callable_defaults(self):
  836. # Use of callable defaults (see bug #7975).
  837. person = Person.objects.create(name='Ringo')
  838. FormSet = inlineformset_factory(Person, Membership, can_delete=False, extra=1)
  839. formset = FormSet(instance=person)
  840. # Django will render a hidden field for model fields that have a callable
  841. # default. This is required to ensure the value is tested for change correctly
  842. # when determine what extra forms have changed to save.
  843. self.assertEqual(len(formset.forms), 1) # this formset only has one form
  844. form = formset.forms[0]
  845. now = form.fields['date_joined'].initial()
  846. result = form.as_p()
  847. result = re.sub(r'\d{4}-\d{2}-\d{2} \d{2}:\d{2}:\d{2}(?:\.\d+)?', '__DATETIME__', result)
  848. self.assertHTMLEqual(result,
  849. '<p><label for="id_membership_set-0-date_joined">Date joined:</label> <input type="text" name="membership_set-0-date_joined" value="__DATETIME__" id="id_membership_set-0-date_joined" /><input type="hidden" name="initial-membership_set-0-date_joined" value="__DATETIME__" id="initial-membership_set-0-id_membership_set-0-date_joined" /></p>\n'
  850. '<p><label for="id_membership_set-0-karma">Karma:</label> <input type="text" name="membership_set-0-karma" id="id_membership_set-0-karma" /><input type="hidden" name="membership_set-0-person" value="%d" id="id_membership_set-0-person" /><input type="hidden" name="membership_set-0-id" id="id_membership_set-0-id" /></p>'
  851. % person.id)
  852. # test for validation with callable defaults. Validations rely on hidden fields
  853. data = {
  854. 'membership_set-TOTAL_FORMS': '1',
  855. 'membership_set-INITIAL_FORMS': '0',
  856. 'membership_set-MAX_NUM_FORMS': '',
  857. 'membership_set-0-date_joined': six.text_type(now.strftime('%Y-%m-%d %H:%M:%S')),
  858. 'initial-membership_set-0-date_joined': six.text_type(now.strftime('%Y-%m-%d %H:%M:%S')),
  859. 'membership_set-0-karma': '',
  860. }
  861. formset = FormSet(data, instance=person)
  862. self.assertTrue(formset.is_valid())
  863. # now test for when the data changes
  864. one_day_later = now + datetime.timedelta(days=1)
  865. filled_data = {
  866. 'membership_set-TOTAL_FORMS': '1',
  867. 'membership_set-INITIAL_FORMS': '0',
  868. 'membership_set-MAX_NUM_FORMS': '',
  869. 'membership_set-0-date_joined': six.text_type(one_day_later.strftime('%Y-%m-%d %H:%M:%S')),
  870. 'initial-membership_set-0-date_joined': six.text_type(now.strftime('%Y-%m-%d %H:%M:%S')),
  871. 'membership_set-0-karma': '',
  872. }
  873. formset = FormSet(filled_data, instance=person)
  874. self.assertFalse(formset.is_valid())
  875. # now test with split datetime fields
  876. class MembershipForm(forms.ModelForm):
  877. date_joined = forms.SplitDateTimeField(initial=now)
  878. class Meta:
  879. model = Membership
  880. def __init__(self, **kwargs):
  881. super(MembershipForm, self).__init__(**kwargs)
  882. self.fields['date_joined'].widget = forms.SplitDateTimeWidget()
  883. FormSet = inlineformset_factory(Person, Membership, form=MembershipForm, can_delete=False, extra=1)
  884. data = {
  885. 'membership_set-TOTAL_FORMS': '1',
  886. 'membership_set-INITIAL_FORMS': '0',
  887. 'membership_set-MAX_NUM_FORMS': '',
  888. 'membership_set-0-date_joined_0': six.text_type(now.strftime('%Y-%m-%d')),
  889. 'membership_set-0-date_joined_1': six.text_type(now.strftime('%H:%M:%S')),
  890. 'initial-membership_set-0-date_joined': six.text_type(now.strftime('%Y-%m-%d %H:%M:%S')),
  891. 'membership_set-0-karma': '',
  892. }
  893. formset = FormSet(data, instance=person)
  894. self.assertTrue(formset.is_valid())
  895. def test_inlineformset_factory_with_null_fk(self):
  896. # inlineformset_factory tests with fk having null=True. see #9462.
  897. # create some data that will exbit the issue
  898. team = Team.objects.create(name="Red Vipers")
  899. Player(name="Timmy").save()
  900. Player(name="Bobby", team=team).save()
  901. PlayerInlineFormSet = inlineformset_factory(Team, Player)
  902. formset = PlayerInlineFormSet()
  903. self.assertQuerysetEqual(formset.get_queryset(), [])
  904. formset = PlayerInlineFormSet(instance=team)
  905. players = formset.get_queryset()
  906. self.assertEqual(len(players), 1)
  907. player1, = players
  908. self.assertEqual(player1.team, team)
  909. self.assertEqual(player1.name, 'Bobby')
  910. def test_model_formset_with_custom_pk(self):
  911. # a formset for a Model that has a custom primary key that still needs to be
  912. # added to the formset automatically
  913. FormSet = modelformset_factory(ClassyMexicanRestaurant, fields=["tacos_are_yummy"])
  914. self.assertEqual(sorted(FormSet().forms[0].fields.keys()), ['restaurant', 'tacos_are_yummy'])
  915. def test_prevent_duplicates_from_with_the_same_formset(self):
  916. FormSet = modelformset_factory(Product, extra=2)
  917. data = {
  918. 'form-TOTAL_FORMS': 2,
  919. 'form-INITIAL_FORMS': 0,
  920. 'form-MAX_NUM_FORMS': '',
  921. 'form-0-slug': 'red_car',
  922. 'form-1-slug': 'red_car',
  923. }
  924. formset = FormSet(data)
  925. self.assertFalse(formset.is_valid())
  926. self.assertEqual(formset._non_form_errors,
  927. ['Please correct the duplicate data for slug.'])
  928. FormSet = modelformset_factory(Price, extra=2)
  929. data = {
  930. 'form-TOTAL_FORMS': 2,
  931. 'form-INITIAL_FORMS': 0,
  932. 'form-MAX_NUM_FORMS': '',
  933. 'form-0-price': '25',
  934. 'form-0-quantity': '7',
  935. 'form-1-price': '25',
  936. 'form-1-quantity': '7',
  937. }
  938. formset = FormSet(data)
  939. self.assertFalse(formset.is_valid())
  940. self.assertEqual(formset._non_form_errors,
  941. ['Please correct the duplicate data for price and quantity, which must be unique.'])
  942. # Only the price field is specified, this should skip any unique checks since
  943. # the unique_together is not fulfilled. This will fail with a KeyError if broken.
  944. FormSet = modelformset_factory(Price, fields=("price",), extra=2)
  945. data = {
  946. 'form-TOTAL_FORMS': '2',
  947. 'form-INITIAL_FORMS': '0',
  948. 'form-MAX_NUM_FORMS': '',
  949. 'form-0-price': '24',
  950. 'form-1-price': '24',
  951. }
  952. formset = FormSet(data)
  953. self.assertTrue(formset.is_valid())
  954. FormSet = inlineformset_factory(Author, Book, extra=0)
  955. author = Author.objects.create(pk=1, name='Charles Baudelaire')
  956. book1 = Book.objects.create(pk=1, author=author, title='Les Paradis Artificiels')
  957. book2 = Book.objects.create(pk=2, author=author, title='Les Fleurs du Mal')
  958. book3 = Book.objects.create(pk=3, author=author, title='Flowers of Evil')
  959. book_ids = author.book_set.order_by('id').values_list('id', flat=True)
  960. data = {
  961. 'book_set-TOTAL_FORMS': '2',
  962. 'book_set-INITIAL_FORMS': '2',
  963. 'book_set-MAX_NUM_FORMS': '',
  964. 'book_set-0-title': 'The 2008 Election',
  965. 'book_set-0-author': str(author.id),
  966. 'book_set-0-id': str(book_ids[0]),
  967. 'book_set-1-title': 'The 2008 Election',
  968. 'book_set-1-author': str(author.id),
  969. 'book_set-1-id': str(book_ids[1]),
  970. }
  971. formset = FormSet(data=data, instance=author)
  972. self.assertFalse(formset.is_valid())
  973. self.assertEqual(formset._non_form_errors,
  974. ['Please correct the duplicate data for title.'])
  975. self.assertEqual(formset.errors,
  976. [{}, {'__all__': ['Please correct the duplicate values below.']}])
  977. FormSet = modelformset_factory(Post, extra=2)
  978. data = {
  979. 'form-TOTAL_FORMS': '2',
  980. 'form-INITIAL_FORMS': '0',
  981. 'form-MAX_NUM_FORMS': '',
  982. 'form-0-title': 'blah',
  983. 'form-0-slug': 'Morning',
  984. 'form-0-subtitle': 'foo',
  985. 'form-0-posted': '2009-01-01',
  986. 'form-1-title': 'blah',
  987. 'form-1-slug': 'Morning in Prague',
  988. 'form-1-subtitle': 'rawr',
  989. 'form-1-posted': '2009-01-01'
  990. }
  991. formset = FormSet(data)
  992. self.assertFalse(formset.is_valid())
  993. self.assertEqual(formset._non_form_errors,
  994. ['Please correct the duplicate data for title which must be unique for the date in posted.'])
  995. self.assertEqual(formset.errors,
  996. [{}, {'__all__': ['Please correct the duplicate values below.']}])
  997. data = {
  998. 'form-TOTAL_FORMS': '2',
  999. 'form-INITIAL_FORMS': '0',
  1000. 'form-MAX_NUM_FORMS': '',
  1001. 'form-0-title': 'foo',
  1002. 'form-0-slug': 'Morning in Prague',
  1003. 'form-0-subtitle': 'foo',
  1004. 'form-0-posted': '2009-01-01',
  1005. 'form-1-title': 'blah',
  1006. 'form-1-slug': 'Morning in Prague',
  1007. 'form-1-subtitle': 'rawr',
  1008. 'form-1-posted': '2009-08-02'
  1009. }
  1010. formset = FormSet(data)
  1011. self.assertFalse(formset.is_valid())
  1012. self.assertEqual(formset._non_form_errors,
  1013. ['Please correct the duplicate data for slug which must be unique for the year in posted.'])
  1014. data = {
  1015. 'form-TOTAL_FORMS': '2',
  1016. 'form-INITIAL_FORMS': '0',
  1017. 'form-MAX_NUM_FORMS': '',
  1018. 'form-0-title': 'foo',
  1019. 'form-0-slug': 'Morning in Prague',
  1020. 'form-0-subtitle': 'rawr',
  1021. 'form-0-posted': '2008-08-01',
  1022. 'form-1-title': 'blah',
  1023. 'form-1-slug': 'Prague',
  1024. 'form-1-subtitle': 'rawr',
  1025. 'form-1-posted': '2009-08-02'
  1026. }
  1027. formset = FormSet(data)
  1028. self.assertFalse(formset.is_valid())
  1029. self.assertEqual(formset._non_form_errors,
  1030. ['Please correct the duplicate data for subtitle which must be unique for the month in posted.'])
  1031. class TestModelFormsetWidgets(TestCase):
  1032. def test_modelformset_factory_widgets(self):
  1033. widgets = {
  1034. 'name': forms.TextInput(attrs={'class': 'poet'})
  1035. }
  1036. PoetFormSet = modelformset_factory(Poet, widgets=widgets)
  1037. form = PoetFormSet.form()
  1038. self.assertHTMLEqual(
  1039. "%s" % form['name'],
  1040. '<input id="id_name" maxlength="100" type="text" class="poet" name="name" />'
  1041. )
  1042. def test_inlineformset_factory_widgets(self):
  1043. widgets = {
  1044. 'title': forms.TextInput(attrs={'class': 'book'})
  1045. }
  1046. BookFormSet = inlineformset_factory(Author, Book, widgets=widgets)
  1047. form = BookFormSet.form()
  1048. self.assertHTMLEqual(
  1049. "%s" % form['title'],
  1050. '<input class="book" id="id_title" maxlength="100" name="title" type="text" />'
  1051. )