tests.py 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417
  1. import datetime
  2. from operator import attrgetter
  3. from django import forms
  4. from django.core.exceptions import FieldError
  5. from django.test import TestCase, skipUnlessDBFeature
  6. from django.utils import translation
  7. from .models import (
  8. Article, ArticleIdea, ArticleTag, ArticleTranslation, Country, Friendship,
  9. Group, Membership, NewsArticle, Person,
  10. )
  11. # Note that these tests are testing internal implementation details.
  12. # ForeignObject is not part of public API.
  13. class MultiColumnFKTests(TestCase):
  14. def setUp(self):
  15. # Creating countries
  16. self.usa = Country.objects.create(name="United States of America")
  17. self.soviet_union = Country.objects.create(name="Soviet Union")
  18. Person()
  19. # Creating People
  20. self.bob = Person()
  21. self.bob.name = 'Bob'
  22. self.bob.person_country = self.usa
  23. self.bob.save()
  24. self.jim = Person.objects.create(name='Jim', person_country=self.usa)
  25. self.george = Person.objects.create(name='George', person_country=self.usa)
  26. self.jane = Person.objects.create(name='Jane', person_country=self.soviet_union)
  27. self.mark = Person.objects.create(name='Mark', person_country=self.soviet_union)
  28. self.sam = Person.objects.create(name='Sam', person_country=self.soviet_union)
  29. # Creating Groups
  30. self.kgb = Group.objects.create(name='KGB', group_country=self.soviet_union)
  31. self.cia = Group.objects.create(name='CIA', group_country=self.usa)
  32. self.republican = Group.objects.create(name='Republican', group_country=self.usa)
  33. self.democrat = Group.objects.create(name='Democrat', group_country=self.usa)
  34. def test_get_succeeds_on_multicolumn_match(self):
  35. # Membership objects have access to their related Person if both
  36. # country_ids match between them
  37. membership = Membership.objects.create(
  38. membership_country_id=self.usa.id, person_id=self.bob.id, group_id=self.cia.id)
  39. person = membership.person
  40. self.assertEqual((person.id, person.name), (self.bob.id, "Bob"))
  41. def test_get_fails_on_multicolumn_mismatch(self):
  42. # Membership objects returns DoesNotExist error when the there is no
  43. # Person with the same id and country_id
  44. membership = Membership.objects.create(
  45. membership_country_id=self.usa.id, person_id=self.jane.id, group_id=self.cia.id)
  46. self.assertRaises(Person.DoesNotExist, getattr, membership, 'person')
  47. def test_reverse_query_returns_correct_result(self):
  48. # Creating a valid membership because it has the same country has the person
  49. Membership.objects.create(
  50. membership_country_id=self.usa.id, person_id=self.bob.id, group_id=self.cia.id)
  51. # Creating an invalid membership because it has a different country has the person
  52. Membership.objects.create(
  53. membership_country_id=self.soviet_union.id, person_id=self.bob.id,
  54. group_id=self.republican.id)
  55. self.assertQuerysetEqual(
  56. self.bob.membership_set.all(), [
  57. self.cia.id
  58. ],
  59. attrgetter("group_id")
  60. )
  61. def test_query_filters_correctly(self):
  62. # Creating a to valid memberships
  63. Membership.objects.create(
  64. membership_country_id=self.usa.id, person_id=self.bob.id, group_id=self.cia.id)
  65. Membership.objects.create(
  66. membership_country_id=self.usa.id, person_id=self.jim.id,
  67. group_id=self.cia.id)
  68. # Creating an invalid membership
  69. Membership.objects.create(membership_country_id=self.soviet_union.id,
  70. person_id=self.george.id, group_id=self.cia.id)
  71. self.assertQuerysetEqual(
  72. Membership.objects.filter(person__name__contains='o'), [
  73. self.bob.id
  74. ],
  75. attrgetter("person_id")
  76. )
  77. def test_reverse_query_filters_correctly(self):
  78. timemark = datetime.datetime.utcnow()
  79. timedelta = datetime.timedelta(days=1)
  80. # Creating a to valid memberships
  81. Membership.objects.create(
  82. membership_country_id=self.usa.id, person_id=self.bob.id,
  83. group_id=self.cia.id, date_joined=timemark - timedelta)
  84. Membership.objects.create(
  85. membership_country_id=self.usa.id, person_id=self.jim.id,
  86. group_id=self.cia.id, date_joined=timemark + timedelta)
  87. # Creating an invalid membership
  88. Membership.objects.create(
  89. membership_country_id=self.soviet_union.id, person_id=self.george.id,
  90. group_id=self.cia.id, date_joined=timemark + timedelta)
  91. self.assertQuerysetEqual(
  92. Person.objects.filter(membership__date_joined__gte=timemark), [
  93. 'Jim'
  94. ],
  95. attrgetter('name')
  96. )
  97. def test_forward_in_lookup_filters_correctly(self):
  98. Membership.objects.create(membership_country_id=self.usa.id, person_id=self.bob.id,
  99. group_id=self.cia.id)
  100. Membership.objects.create(membership_country_id=self.usa.id, person_id=self.jim.id,
  101. group_id=self.cia.id)
  102. # Creating an invalid membership
  103. Membership.objects.create(
  104. membership_country_id=self.soviet_union.id, person_id=self.george.id,
  105. group_id=self.cia.id)
  106. self.assertQuerysetEqual(
  107. Membership.objects.filter(person__in=[self.george, self.jim]), [
  108. self.jim.id,
  109. ],
  110. attrgetter('person_id')
  111. )
  112. self.assertQuerysetEqual(
  113. Membership.objects.filter(person__in=Person.objects.filter(name='Jim')), [
  114. self.jim.id,
  115. ],
  116. attrgetter('person_id')
  117. )
  118. def test_double_nested_query(self):
  119. m1 = Membership.objects.create(membership_country_id=self.usa.id, person_id=self.bob.id,
  120. group_id=self.cia.id)
  121. m2 = Membership.objects.create(membership_country_id=self.usa.id, person_id=self.jim.id,
  122. group_id=self.cia.id)
  123. Friendship.objects.create(from_friend_country_id=self.usa.id, from_friend_id=self.bob.id,
  124. to_friend_country_id=self.usa.id, to_friend_id=self.jim.id)
  125. self.assertQuerysetEqual(Membership.objects.filter(
  126. person__in=Person.objects.filter(
  127. from_friend__in=Friendship.objects.filter(
  128. to_friend__in=Person.objects.all()))),
  129. [m1], lambda x: x)
  130. self.assertQuerysetEqual(Membership.objects.exclude(
  131. person__in=Person.objects.filter(
  132. from_friend__in=Friendship.objects.filter(
  133. to_friend__in=Person.objects.all()))),
  134. [m2], lambda x: x)
  135. def test_select_related_foreignkey_forward_works(self):
  136. Membership.objects.create(membership_country=self.usa, person=self.bob, group=self.cia)
  137. Membership.objects.create(membership_country=self.usa, person=self.jim, group=self.democrat)
  138. with self.assertNumQueries(1):
  139. people = [m.person for m in Membership.objects.select_related('person').order_by('pk')]
  140. normal_people = [m.person for m in Membership.objects.all().order_by('pk')]
  141. self.assertEqual(people, normal_people)
  142. def test_prefetch_foreignkey_forward_works(self):
  143. Membership.objects.create(membership_country=self.usa, person=self.bob, group=self.cia)
  144. Membership.objects.create(membership_country=self.usa, person=self.jim, group=self.democrat)
  145. with self.assertNumQueries(2):
  146. people = [
  147. m.person for m in Membership.objects.prefetch_related('person').order_by('pk')]
  148. normal_people = [m.person for m in Membership.objects.order_by('pk')]
  149. self.assertEqual(people, normal_people)
  150. def test_prefetch_foreignkey_reverse_works(self):
  151. Membership.objects.create(membership_country=self.usa, person=self.bob, group=self.cia)
  152. Membership.objects.create(membership_country=self.usa, person=self.jim, group=self.democrat)
  153. with self.assertNumQueries(2):
  154. membership_sets = [
  155. list(p.membership_set.all())
  156. for p in Person.objects.prefetch_related('membership_set').order_by('pk')]
  157. normal_membership_sets = [list(p.membership_set.all())
  158. for p in Person.objects.order_by('pk')]
  159. self.assertEqual(membership_sets, normal_membership_sets)
  160. def test_m2m_through_forward_returns_valid_members(self):
  161. # We start out by making sure that the Group 'CIA' has no members.
  162. self.assertQuerysetEqual(
  163. self.cia.members.all(),
  164. []
  165. )
  166. Membership.objects.create(membership_country=self.usa, person=self.bob, group=self.cia)
  167. Membership.objects.create(membership_country=self.usa, person=self.jim, group=self.cia)
  168. # Let's check to make sure that it worked. Bob and Jim should be members of the CIA.
  169. self.assertQuerysetEqual(
  170. self.cia.members.all(), [
  171. 'Bob',
  172. 'Jim'
  173. ], attrgetter("name")
  174. )
  175. def test_m2m_through_reverse_returns_valid_members(self):
  176. # We start out by making sure that Bob is in no groups.
  177. self.assertQuerysetEqual(
  178. self.bob.groups.all(),
  179. []
  180. )
  181. Membership.objects.create(membership_country=self.usa, person=self.bob, group=self.cia)
  182. Membership.objects.create(membership_country=self.usa, person=self.bob,
  183. group=self.republican)
  184. # Bob should be in the CIA and a Republican
  185. self.assertQuerysetEqual(
  186. self.bob.groups.all(), [
  187. 'CIA',
  188. 'Republican'
  189. ], attrgetter("name")
  190. )
  191. def test_m2m_through_forward_ignores_invalid_members(self):
  192. # We start out by making sure that the Group 'CIA' has no members.
  193. self.assertQuerysetEqual(
  194. self.cia.members.all(),
  195. []
  196. )
  197. # Something adds jane to group CIA but Jane is in Soviet Union which isn't CIA's country
  198. Membership.objects.create(membership_country=self.usa, person=self.jane, group=self.cia)
  199. # There should still be no members in CIA
  200. self.assertQuerysetEqual(
  201. self.cia.members.all(),
  202. []
  203. )
  204. def test_m2m_through_reverse_ignores_invalid_members(self):
  205. # We start out by making sure that Jane has no groups.
  206. self.assertQuerysetEqual(
  207. self.jane.groups.all(),
  208. []
  209. )
  210. # Something adds jane to group CIA but Jane is in Soviet Union which isn't CIA's country
  211. Membership.objects.create(membership_country=self.usa, person=self.jane, group=self.cia)
  212. # Jane should still not be in any groups
  213. self.assertQuerysetEqual(
  214. self.jane.groups.all(),
  215. []
  216. )
  217. def test_m2m_through_on_self_works(self):
  218. self.assertQuerysetEqual(
  219. self.jane.friends.all(),
  220. []
  221. )
  222. Friendship.objects.create(
  223. from_friend_country=self.jane.person_country, from_friend=self.jane,
  224. to_friend_country=self.george.person_country, to_friend=self.george)
  225. self.assertQuerysetEqual(
  226. self.jane.friends.all(),
  227. ['George'], attrgetter("name")
  228. )
  229. def test_m2m_through_on_self_ignores_mismatch_columns(self):
  230. self.assertQuerysetEqual(self.jane.friends.all(), [])
  231. # Note that we use ids instead of instances. This is because instances on ForeignObject
  232. # properties will set all related field off of the given instance
  233. Friendship.objects.create(
  234. from_friend_id=self.jane.id, to_friend_id=self.george.id,
  235. to_friend_country_id=self.jane.person_country_id,
  236. from_friend_country_id=self.george.person_country_id)
  237. self.assertQuerysetEqual(self.jane.friends.all(), [])
  238. def test_prefetch_related_m2m_foward_works(self):
  239. Membership.objects.create(membership_country=self.usa, person=self.bob, group=self.cia)
  240. Membership.objects.create(membership_country=self.usa, person=self.jim, group=self.democrat)
  241. with self.assertNumQueries(2):
  242. members_lists = [list(g.members.all())
  243. for g in Group.objects.prefetch_related('members')]
  244. normal_members_lists = [list(g.members.all()) for g in Group.objects.all()]
  245. self.assertEqual(members_lists, normal_members_lists)
  246. def test_prefetch_related_m2m_reverse_works(self):
  247. Membership.objects.create(membership_country=self.usa, person=self.bob, group=self.cia)
  248. Membership.objects.create(membership_country=self.usa, person=self.jim, group=self.democrat)
  249. with self.assertNumQueries(2):
  250. groups_lists = [list(p.groups.all()) for p in Person.objects.prefetch_related('groups')]
  251. normal_groups_lists = [list(p.groups.all()) for p in Person.objects.all()]
  252. self.assertEqual(groups_lists, normal_groups_lists)
  253. @translation.override('fi')
  254. def test_translations(self):
  255. a1 = Article.objects.create(pub_date=datetime.date.today())
  256. at1_fi = ArticleTranslation(article=a1, lang='fi', title='Otsikko', body='Diipadaapa')
  257. at1_fi.save()
  258. at2_en = ArticleTranslation(article=a1, lang='en', title='Title', body='Lalalalala')
  259. at2_en.save()
  260. self.assertEqual(Article.objects.get(pk=a1.pk).active_translation, at1_fi)
  261. with self.assertNumQueries(1):
  262. fetched = Article.objects.select_related('active_translation').get(
  263. active_translation__title='Otsikko')
  264. self.assertEqual(fetched.active_translation.title, 'Otsikko')
  265. a2 = Article.objects.create(pub_date=datetime.date.today())
  266. at2_fi = ArticleTranslation(article=a2, lang='fi', title='Atsikko', body='Diipadaapa',
  267. abstract='dipad')
  268. at2_fi.save()
  269. a3 = Article.objects.create(pub_date=datetime.date.today())
  270. at3_en = ArticleTranslation(article=a3, lang='en', title='A title', body='lalalalala',
  271. abstract='lala')
  272. at3_en.save()
  273. # Test model initialization with active_translation field.
  274. a3 = Article(id=a3.id, pub_date=a3.pub_date, active_translation=at3_en)
  275. a3.save()
  276. self.assertEqual(
  277. list(Article.objects.filter(active_translation__abstract=None)),
  278. [a1, a3])
  279. self.assertEqual(
  280. list(Article.objects.filter(active_translation__abstract=None,
  281. active_translation__pk__isnull=False)),
  282. [a1])
  283. with translation.override('en'):
  284. self.assertEqual(
  285. list(Article.objects.filter(active_translation__abstract=None)),
  286. [a1, a2])
  287. def test_foreign_key_raises_informative_does_not_exist(self):
  288. referrer = ArticleTranslation()
  289. with self.assertRaisesMessage(Article.DoesNotExist, 'ArticleTranslation has no article'):
  290. referrer.article
  291. def test_foreign_key_related_query_name(self):
  292. a1 = Article.objects.create(pub_date=datetime.date.today())
  293. ArticleTag.objects.create(article=a1, name="foo")
  294. self.assertEqual(Article.objects.filter(tag__name="foo").count(), 1)
  295. self.assertEqual(Article.objects.filter(tag__name="bar").count(), 0)
  296. with self.assertRaises(FieldError):
  297. Article.objects.filter(tags__name="foo")
  298. def test_many_to_many_related_query_name(self):
  299. a1 = Article.objects.create(pub_date=datetime.date.today())
  300. i1 = ArticleIdea.objects.create(name="idea1")
  301. a1.ideas.add(i1)
  302. self.assertEqual(Article.objects.filter(idea_things__name="idea1").count(), 1)
  303. self.assertEqual(Article.objects.filter(idea_things__name="idea2").count(), 0)
  304. with self.assertRaises(FieldError):
  305. Article.objects.filter(ideas__name="idea1")
  306. @translation.override('fi')
  307. def test_inheritance(self):
  308. na = NewsArticle.objects.create(pub_date=datetime.date.today())
  309. ArticleTranslation.objects.create(
  310. article=na, lang="fi", title="foo", body="bar")
  311. self.assertQuerysetEqual(
  312. NewsArticle.objects.select_related('active_translation'),
  313. [na], lambda x: x
  314. )
  315. with self.assertNumQueries(1):
  316. self.assertEqual(
  317. NewsArticle.objects.select_related(
  318. 'active_translation')[0].active_translation.title,
  319. "foo")
  320. @skipUnlessDBFeature('has_bulk_insert')
  321. def test_batch_create_foreign_object(self):
  322. """ See: https://code.djangoproject.com/ticket/21566 """
  323. objs = [Person(name="abcd_%s" % i, person_country=self.usa) for i in range(0, 5)]
  324. Person.objects.bulk_create(objs, 10)
  325. class FormsTests(TestCase):
  326. # ForeignObjects should not have any form fields, currently the user needs
  327. # to manually deal with the foreignobject relation.
  328. class ArticleForm(forms.ModelForm):
  329. class Meta:
  330. model = Article
  331. fields = '__all__'
  332. def test_foreign_object_form(self):
  333. # A very crude test checking that the non-concrete fields do not get form fields.
  334. form = FormsTests.ArticleForm()
  335. self.assertIn('id_pub_date', form.as_table())
  336. self.assertNotIn('active_translation', form.as_table())
  337. form = FormsTests.ArticleForm(data={'pub_date': str(datetime.date.today())})
  338. self.assertTrue(form.is_valid())
  339. a = form.save()
  340. self.assertEqual(a.pub_date, datetime.date.today())
  341. form = FormsTests.ArticleForm(instance=a, data={'pub_date': '2013-01-01'})
  342. a2 = form.save()
  343. self.assertEqual(a.pk, a2.pk)
  344. self.assertEqual(a2.pub_date, datetime.date(2013, 1, 1))