tests.py 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475
  1. # -*- coding: utf-8 -*-
  2. from __future__ import unicode_literals
  3. import warnings
  4. from django.contrib import admin
  5. from django.contrib.admin.sites import AdminSite
  6. from django.contrib.auth.models import User
  7. from django.contrib.contenttypes.admin import GenericTabularInline
  8. from django.contrib.contenttypes.forms import generic_inlineformset_factory
  9. from django.forms.formsets import DEFAULT_MAX_NUM
  10. from django.forms.models import ModelForm
  11. from django.test import TestCase, override_settings, RequestFactory
  12. from django.utils.deprecation import RemovedInDjango19Warning
  13. # local test models
  14. from .admin import MediaInline, MediaPermanentInline, site as admin_site
  15. from .models import Episode, Media, EpisodePermanent, Category
  16. @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',),
  17. TEMPLATE_DEBUG=True,
  18. ROOT_URLCONF="generic_inline_admin.urls")
  19. class GenericAdminViewTest(TestCase):
  20. fixtures = ['users.xml']
  21. def setUp(self):
  22. # set TEMPLATE_DEBUG to True to ensure {% include %} will raise
  23. # exceptions since that is how inlines are rendered and #9498 will
  24. # bubble up if it is an issue.
  25. self.client.login(username='super', password='secret')
  26. # Can't load content via a fixture (since the GenericForeignKey
  27. # relies on content type IDs, which will vary depending on what
  28. # other tests have been run), thus we do it here.
  29. e = Episode.objects.create(name='This Week in Django')
  30. self.episode_pk = e.pk
  31. m = Media(content_object=e, url='http://example.com/podcast.mp3')
  32. m.save()
  33. self.mp3_media_pk = m.pk
  34. m = Media(content_object=e, url='http://example.com/logo.png')
  35. m.save()
  36. self.png_media_pk = m.pk
  37. def tearDown(self):
  38. self.client.logout()
  39. def testBasicAddGet(self):
  40. """
  41. A smoke test to ensure GET on the add_view works.
  42. """
  43. response = self.client.get('/generic_inline_admin/admin/generic_inline_admin/episode/add/')
  44. self.assertEqual(response.status_code, 200)
  45. def testBasicEditGet(self):
  46. """
  47. A smoke test to ensure GET on the change_view works.
  48. """
  49. response = self.client.get('/generic_inline_admin/admin/generic_inline_admin/episode/%d/' % self.episode_pk)
  50. self.assertEqual(response.status_code, 200)
  51. def testBasicAddPost(self):
  52. """
  53. A smoke test to ensure POST on add_view works.
  54. """
  55. post_data = {
  56. "name": "This Week in Django",
  57. # inline data
  58. "generic_inline_admin-media-content_type-object_id-TOTAL_FORMS": "1",
  59. "generic_inline_admin-media-content_type-object_id-INITIAL_FORMS": "0",
  60. "generic_inline_admin-media-content_type-object_id-MAX_NUM_FORMS": "0",
  61. }
  62. response = self.client.post('/generic_inline_admin/admin/generic_inline_admin/episode/add/', post_data)
  63. self.assertEqual(response.status_code, 302) # redirect somewhere
  64. def testBasicEditPost(self):
  65. """
  66. A smoke test to ensure POST on edit_view works.
  67. """
  68. post_data = {
  69. "name": "This Week in Django",
  70. # inline data
  71. "generic_inline_admin-media-content_type-object_id-TOTAL_FORMS": "3",
  72. "generic_inline_admin-media-content_type-object_id-INITIAL_FORMS": "2",
  73. "generic_inline_admin-media-content_type-object_id-MAX_NUM_FORMS": "0",
  74. "generic_inline_admin-media-content_type-object_id-0-id": "%d" % self.mp3_media_pk,
  75. "generic_inline_admin-media-content_type-object_id-0-url": "http://example.com/podcast.mp3",
  76. "generic_inline_admin-media-content_type-object_id-1-id": "%d" % self.png_media_pk,
  77. "generic_inline_admin-media-content_type-object_id-1-url": "http://example.com/logo.png",
  78. "generic_inline_admin-media-content_type-object_id-2-id": "",
  79. "generic_inline_admin-media-content_type-object_id-2-url": "",
  80. }
  81. url = '/generic_inline_admin/admin/generic_inline_admin/episode/%d/' % self.episode_pk
  82. response = self.client.post(url, post_data)
  83. self.assertEqual(response.status_code, 302) # redirect somewhere
  84. def testGenericInlineFormset(self):
  85. EpisodeMediaFormSet = generic_inlineformset_factory(Media, can_delete=False, exclude=['description', 'keywords'], extra=3)
  86. e = Episode.objects.get(name='This Week in Django')
  87. # Works with no queryset
  88. formset = EpisodeMediaFormSet(instance=e)
  89. self.assertEqual(len(formset.forms), 5)
  90. self.assertHTMLEqual(formset.forms[0].as_p(), '<p><label for="id_generic_inline_admin-media-content_type-object_id-0-url">Url:</label> <input id="id_generic_inline_admin-media-content_type-object_id-0-url" type="url" name="generic_inline_admin-media-content_type-object_id-0-url" value="http://example.com/podcast.mp3" maxlength="200" /><input type="hidden" name="generic_inline_admin-media-content_type-object_id-0-id" value="%s" id="id_generic_inline_admin-media-content_type-object_id-0-id" /></p>' % self.mp3_media_pk)
  91. self.assertHTMLEqual(formset.forms[1].as_p(), '<p><label for="id_generic_inline_admin-media-content_type-object_id-1-url">Url:</label> <input id="id_generic_inline_admin-media-content_type-object_id-1-url" type="url" name="generic_inline_admin-media-content_type-object_id-1-url" value="http://example.com/logo.png" maxlength="200" /><input type="hidden" name="generic_inline_admin-media-content_type-object_id-1-id" value="%s" id="id_generic_inline_admin-media-content_type-object_id-1-id" /></p>' % self.png_media_pk)
  92. self.assertHTMLEqual(formset.forms[2].as_p(), '<p><label for="id_generic_inline_admin-media-content_type-object_id-2-url">Url:</label> <input id="id_generic_inline_admin-media-content_type-object_id-2-url" type="url" name="generic_inline_admin-media-content_type-object_id-2-url" maxlength="200" /><input type="hidden" name="generic_inline_admin-media-content_type-object_id-2-id" id="id_generic_inline_admin-media-content_type-object_id-2-id" /></p>')
  93. # A queryset can be used to alter display ordering
  94. formset = EpisodeMediaFormSet(instance=e, queryset=Media.objects.order_by('url'))
  95. self.assertEqual(len(formset.forms), 5)
  96. self.assertHTMLEqual(formset.forms[0].as_p(), '<p><label for="id_generic_inline_admin-media-content_type-object_id-0-url">Url:</label> <input id="id_generic_inline_admin-media-content_type-object_id-0-url" type="url" name="generic_inline_admin-media-content_type-object_id-0-url" value="http://example.com/logo.png" maxlength="200" /><input type="hidden" name="generic_inline_admin-media-content_type-object_id-0-id" value="%s" id="id_generic_inline_admin-media-content_type-object_id-0-id" /></p>' % self.png_media_pk)
  97. self.assertHTMLEqual(formset.forms[1].as_p(), '<p><label for="id_generic_inline_admin-media-content_type-object_id-1-url">Url:</label> <input id="id_generic_inline_admin-media-content_type-object_id-1-url" type="url" name="generic_inline_admin-media-content_type-object_id-1-url" value="http://example.com/podcast.mp3" maxlength="200" /><input type="hidden" name="generic_inline_admin-media-content_type-object_id-1-id" value="%s" id="id_generic_inline_admin-media-content_type-object_id-1-id" /></p>' % self.mp3_media_pk)
  98. self.assertHTMLEqual(formset.forms[2].as_p(), '<p><label for="id_generic_inline_admin-media-content_type-object_id-2-url">Url:</label> <input id="id_generic_inline_admin-media-content_type-object_id-2-url" type="url" name="generic_inline_admin-media-content_type-object_id-2-url" maxlength="200" /><input type="hidden" name="generic_inline_admin-media-content_type-object_id-2-id" id="id_generic_inline_admin-media-content_type-object_id-2-id" /></p>')
  99. # Works with a queryset that omits items
  100. formset = EpisodeMediaFormSet(instance=e, queryset=Media.objects.filter(url__endswith=".png"))
  101. self.assertEqual(len(formset.forms), 4)
  102. self.assertHTMLEqual(formset.forms[0].as_p(), '<p><label for="id_generic_inline_admin-media-content_type-object_id-0-url">Url:</label> <input id="id_generic_inline_admin-media-content_type-object_id-0-url" type="url" name="generic_inline_admin-media-content_type-object_id-0-url" value="http://example.com/logo.png" maxlength="200" /><input type="hidden" name="generic_inline_admin-media-content_type-object_id-0-id" value="%s" id="id_generic_inline_admin-media-content_type-object_id-0-id" /></p>' % self.png_media_pk)
  103. self.assertHTMLEqual(formset.forms[1].as_p(), '<p><label for="id_generic_inline_admin-media-content_type-object_id-1-url">Url:</label> <input id="id_generic_inline_admin-media-content_type-object_id-1-url" type="url" name="generic_inline_admin-media-content_type-object_id-1-url" maxlength="200" /><input type="hidden" name="generic_inline_admin-media-content_type-object_id-1-id" id="id_generic_inline_admin-media-content_type-object_id-1-id" /></p>')
  104. def testGenericInlineFormsetFactory(self):
  105. # Regression test for #10522.
  106. inline_formset = generic_inlineformset_factory(Media,
  107. exclude=('url',))
  108. # Regression test for #12340.
  109. e = Episode.objects.get(name='This Week in Django')
  110. formset = inline_formset(instance=e)
  111. self.assertTrue(formset.get_queryset().ordered)
  112. @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',),
  113. ROOT_URLCONF="generic_inline_admin.urls")
  114. class GenericInlineAdminParametersTest(TestCase):
  115. fixtures = ['users.xml']
  116. def setUp(self):
  117. self.client.login(username='super', password='secret')
  118. self.factory = RequestFactory()
  119. def tearDown(self):
  120. self.client.logout()
  121. def _create_object(self, model):
  122. """
  123. Create a model with an attached Media object via GFK. We can't
  124. load content via a fixture (since the GenericForeignKey relies on
  125. content type IDs, which will vary depending on what other tests
  126. have been run), thus we do it here.
  127. """
  128. e = model.objects.create(name='This Week in Django')
  129. Media.objects.create(content_object=e, url='http://example.com/podcast.mp3')
  130. return e
  131. def testNoParam(self):
  132. """
  133. With one initial form, extra (default) at 3, there should be 4 forms.
  134. """
  135. e = self._create_object(Episode)
  136. response = self.client.get('/generic_inline_admin/admin/generic_inline_admin/episode/%s/' % e.pk)
  137. formset = response.context['inline_admin_formsets'][0].formset
  138. self.assertEqual(formset.total_form_count(), 4)
  139. self.assertEqual(formset.initial_form_count(), 1)
  140. def testExtraParam(self):
  141. """
  142. With extra=0, there should be one form.
  143. """
  144. class ExtraInline(GenericTabularInline):
  145. model = Media
  146. extra = 0
  147. modeladmin = admin.ModelAdmin(Episode, admin_site)
  148. modeladmin.inlines = [ExtraInline]
  149. e = self._create_object(Episode)
  150. request = self.factory.get('/generic_inline_admin/admin/generic_inline_admin/episode/%s/' % e.pk)
  151. request.user = User(username='super', is_superuser=True)
  152. response = modeladmin.changeform_view(request, object_id=str(e.pk))
  153. formset = response.context_data['inline_admin_formsets'][0].formset
  154. self.assertEqual(formset.total_form_count(), 1)
  155. self.assertEqual(formset.initial_form_count(), 1)
  156. def testMaxNumParam(self):
  157. """
  158. With extra=5 and max_num=2, there should be only 2 forms.
  159. """
  160. class MaxNumInline(GenericTabularInline):
  161. model = Media
  162. extra = 5
  163. max_num = 2
  164. modeladmin = admin.ModelAdmin(Episode, admin_site)
  165. modeladmin.inlines = [MaxNumInline]
  166. e = self._create_object(Episode)
  167. request = self.factory.get('/generic_inline_admin/admin/generic_inline_admin/episode/%s/' % e.pk)
  168. request.user = User(username='super', is_superuser=True)
  169. response = modeladmin.changeform_view(request, object_id=str(e.pk))
  170. formset = response.context_data['inline_admin_formsets'][0].formset
  171. self.assertEqual(formset.total_form_count(), 2)
  172. self.assertEqual(formset.initial_form_count(), 1)
  173. def testMinNumParam(self):
  174. """
  175. With extra=3 and min_num=2, there should be six forms.
  176. See #22628 - this will change when that's fixed.
  177. """
  178. class MinNumInline(GenericTabularInline):
  179. model = Media
  180. extra = 3
  181. min_num = 2
  182. modeladmin = admin.ModelAdmin(Episode, admin_site)
  183. modeladmin.inlines = [MinNumInline]
  184. e = self._create_object(Episode)
  185. request = self.factory.get('/generic_inline_admin/admin/generic_inline_admin/episode/%s/' % e.pk)
  186. request.user = User(username='super', is_superuser=True)
  187. response = modeladmin.changeform_view(request, object_id=str(e.pk))
  188. formset = response.context_data['inline_admin_formsets'][0].formset
  189. self.assertEqual(formset.total_form_count(), 6)
  190. self.assertEqual(formset.initial_form_count(), 1)
  191. @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',),
  192. ROOT_URLCONF="generic_inline_admin.urls")
  193. class GenericInlineAdminWithUniqueTogetherTest(TestCase):
  194. fixtures = ['users.xml']
  195. def setUp(self):
  196. self.client.login(username='super', password='secret')
  197. def tearDown(self):
  198. self.client.logout()
  199. def testAdd(self):
  200. category_id = Category.objects.create(name='male').pk
  201. post_data = {
  202. "name": "John Doe",
  203. # inline data
  204. "generic_inline_admin-phonenumber-content_type-object_id-TOTAL_FORMS": "1",
  205. "generic_inline_admin-phonenumber-content_type-object_id-INITIAL_FORMS": "0",
  206. "generic_inline_admin-phonenumber-content_type-object_id-MAX_NUM_FORMS": "0",
  207. "generic_inline_admin-phonenumber-content_type-object_id-0-id": "",
  208. "generic_inline_admin-phonenumber-content_type-object_id-0-phone_number": "555-555-5555",
  209. "generic_inline_admin-phonenumber-content_type-object_id-0-category": "%s" % category_id,
  210. }
  211. response = self.client.get('/generic_inline_admin/admin/generic_inline_admin/contact/add/')
  212. self.assertEqual(response.status_code, 200)
  213. response = self.client.post('/generic_inline_admin/admin/generic_inline_admin/contact/add/', post_data)
  214. self.assertEqual(response.status_code, 302) # redirect somewhere
  215. @override_settings(ROOT_URLCONF="generic_inline_admin.urls")
  216. class NoInlineDeletionTest(TestCase):
  217. def test_no_deletion(self):
  218. fake_site = object()
  219. inline = MediaPermanentInline(EpisodePermanent, fake_site)
  220. fake_request = object()
  221. formset = inline.get_formset(fake_request)
  222. self.assertFalse(formset.can_delete)
  223. class MockRequest(object):
  224. pass
  225. class MockSuperUser(object):
  226. def has_perm(self, perm):
  227. return True
  228. request = MockRequest()
  229. request.user = MockSuperUser()
  230. @override_settings(ROOT_URLCONF="generic_inline_admin.urls")
  231. class GenericInlineModelAdminTest(TestCase):
  232. def setUp(self):
  233. self.site = AdminSite()
  234. def test_get_formset_kwargs(self):
  235. media_inline = MediaInline(Media, AdminSite())
  236. # Create a formset with default arguments
  237. formset = media_inline.get_formset(request)
  238. self.assertEqual(formset.max_num, DEFAULT_MAX_NUM)
  239. self.assertEqual(formset.can_order, False)
  240. # Create a formset with custom keyword arguments
  241. formset = media_inline.get_formset(request, max_num=100, can_order=True)
  242. self.assertEqual(formset.max_num, 100)
  243. self.assertEqual(formset.can_order, True)
  244. def test_custom_form_meta_exclude_with_readonly(self):
  245. """
  246. Ensure that the custom ModelForm's `Meta.exclude` is respected when
  247. used in conjunction with `GenericInlineModelAdmin.readonly_fields`
  248. and when no `ModelAdmin.exclude` is defined.
  249. """
  250. class MediaForm(ModelForm):
  251. class Meta:
  252. model = Media
  253. exclude = ['url']
  254. class MediaInline(GenericTabularInline):
  255. readonly_fields = ['description']
  256. form = MediaForm
  257. model = Media
  258. class EpisodeAdmin(admin.ModelAdmin):
  259. inlines = [
  260. MediaInline
  261. ]
  262. ma = EpisodeAdmin(Episode, self.site)
  263. self.assertEqual(
  264. list(list(ma.get_formsets_with_inlines(request))[0][0]().forms[0].fields),
  265. ['keywords', 'id', 'DELETE'])
  266. def test_custom_form_meta_exclude(self):
  267. """
  268. Ensure that the custom ModelForm's `Meta.exclude` is respected by
  269. `GenericInlineModelAdmin.get_formset`, and overridden if
  270. `ModelAdmin.exclude` or `GenericInlineModelAdmin.exclude` are defined.
  271. Refs #15907.
  272. """
  273. # First with `GenericInlineModelAdmin` -----------------
  274. class MediaForm(ModelForm):
  275. class Meta:
  276. model = Media
  277. exclude = ['url']
  278. class MediaInline(GenericTabularInline):
  279. exclude = ['description']
  280. form = MediaForm
  281. model = Media
  282. class EpisodeAdmin(admin.ModelAdmin):
  283. inlines = [
  284. MediaInline
  285. ]
  286. ma = EpisodeAdmin(Episode, self.site)
  287. self.assertEqual(
  288. list(list(ma.get_formsets_with_inlines(request))[0][0]().forms[0].fields),
  289. ['url', 'keywords', 'id', 'DELETE'])
  290. # Then, only with `ModelForm` -----------------
  291. class MediaInline(GenericTabularInline):
  292. form = MediaForm
  293. model = Media
  294. class EpisodeAdmin(admin.ModelAdmin):
  295. inlines = [
  296. MediaInline
  297. ]
  298. ma = EpisodeAdmin(Episode, self.site)
  299. self.assertEqual(
  300. list(list(ma.get_formsets_with_inlines(request))[0][0]().forms[0].fields),
  301. ['description', 'keywords', 'id', 'DELETE'])
  302. def test_get_fieldsets(self):
  303. # Test that get_fieldsets is called when figuring out form fields.
  304. # Refs #18681.
  305. class MediaForm(ModelForm):
  306. class Meta:
  307. model = Media
  308. fields = '__all__'
  309. class MediaInline(GenericTabularInline):
  310. form = MediaForm
  311. model = Media
  312. can_delete = False
  313. def get_fieldsets(self, request, obj=None):
  314. return [(None, {'fields': ['url', 'description']})]
  315. ma = MediaInline(Media, self.site)
  316. form = ma.get_formset(None).form
  317. self.assertEqual(form._meta.fields, ['url', 'description'])
  318. def test_get_formsets_with_inlines(self):
  319. """
  320. get_formsets() triggers a deprecation warning when get_formsets is
  321. overridden.
  322. """
  323. class MediaForm(ModelForm):
  324. class Meta:
  325. model = Media
  326. exclude = ['url']
  327. class MediaInline(GenericTabularInline):
  328. exclude = ['description']
  329. form = MediaForm
  330. model = Media
  331. class EpisodeAdmin(admin.ModelAdmin):
  332. inlines = [
  333. MediaInline
  334. ]
  335. def get_formsets(self, request, obj=None):
  336. return []
  337. with warnings.catch_warnings(record=True) as w:
  338. warnings.simplefilter("always")
  339. ma = EpisodeAdmin(Episode, self.site)
  340. list(ma.get_formsets_with_inlines(request))
  341. # Verify that the deprecation warning was triggered when get_formsets was called
  342. # This verifies that we called that method.
  343. self.assertEqual(len(w), 1)
  344. self.assertTrue(issubclass(w[0].category, RemovedInDjango19Warning))
  345. class EpisodeAdmin(admin.ModelAdmin):
  346. inlines = [
  347. MediaInline
  348. ]
  349. with warnings.catch_warnings(record=True) as w:
  350. warnings.simplefilter("always")
  351. ma = EpisodeAdmin(Episode, self.site)
  352. list(ma.get_formsets_with_inlines(request))
  353. self.assertEqual(len(w), 0)
  354. def test_get_formsets_with_inlines_returns_tuples(self):
  355. """
  356. Ensure that get_formsets_with_inlines() returns the correct tuples.
  357. """
  358. class MediaForm(ModelForm):
  359. class Meta:
  360. model = Media
  361. exclude = ['url']
  362. class MediaInline(GenericTabularInline):
  363. form = MediaForm
  364. model = Media
  365. class AlternateInline(GenericTabularInline):
  366. form = MediaForm
  367. model = Media
  368. class EpisodeAdmin(admin.ModelAdmin):
  369. inlines = [
  370. AlternateInline, MediaInline
  371. ]
  372. ma = EpisodeAdmin(Episode, self.site)
  373. inlines = ma.get_inline_instances(request)
  374. for (formset, inline), other_inline in zip(ma.get_formsets_with_inlines(request), inlines):
  375. self.assertIsInstance(formset, other_inline.get_formset(request).__class__)
  376. class EpisodeAdmin(admin.ModelAdmin):
  377. inlines = [
  378. AlternateInline, MediaInline
  379. ]
  380. def get_formsets(self, request, obj=None):
  381. # Catch the deprecation warning to force the usage of get_formsets
  382. with warnings.catch_warnings(record=True):
  383. warnings.simplefilter("always")
  384. return super(EpisodeAdmin, self).get_formsets(request, obj)
  385. ma = EpisodeAdmin(Episode, self.site)
  386. inlines = ma.get_inline_instances(request)
  387. with warnings.catch_warnings(record=True):
  388. warnings.simplefilter("always")
  389. for (formset, inline), other_inline in zip(ma.get_formsets_with_inlines(request), inlines):
  390. self.assertIsInstance(formset, other_inline.get_formset(request).__class__)