tests.py 22 KB

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