2019-04-19 18:12:04 +02:00
|
|
|
import datetime
|
|
|
|
|
2010-10-09 16:26:48 +00:00
|
|
|
from django.test import TestCase
|
|
|
|
|
2011-10-13 18:04:12 +00:00
|
|
|
from .models import Person
|
2010-10-09 16:26:48 +00:00
|
|
|
|
|
|
|
|
|
|
|
class RecursiveM2MTests(TestCase):
|
2018-11-23 20:59:38 -05:00
|
|
|
@classmethod
|
|
|
|
def setUpTestData(cls):
|
|
|
|
cls.a, cls.b, cls.c, cls.d = [
|
2010-10-09 16:26:48 +00:00
|
|
|
Person.objects.create(name=name)
|
2022-02-03 20:24:19 +01:00
|
|
|
for name in ["Anne", "Bill", "Chuck", "David"]
|
2010-10-09 16:26:48 +00:00
|
|
|
]
|
2018-11-23 20:59:38 -05:00
|
|
|
cls.a.friends.add(cls.b, cls.c)
|
2019-05-31 16:19:11 +02:00
|
|
|
# Add m2m for Anne and Chuck in reverse direction.
|
2018-11-23 20:59:38 -05:00
|
|
|
cls.d.friends.add(cls.a, cls.c)
|
2011-10-13 18:04:12 +00:00
|
|
|
|
2014-02-10 05:21:42 -05:00
|
|
|
def test_recursive_m2m_all(self):
|
2019-05-31 16:19:11 +02:00
|
|
|
for person, friends in (
|
|
|
|
(self.a, [self.b, self.c, self.d]),
|
|
|
|
(self.b, [self.a]),
|
|
|
|
(self.c, [self.a, self.d]),
|
|
|
|
(self.d, [self.a, self.c]),
|
|
|
|
):
|
|
|
|
with self.subTest(person=person):
|
|
|
|
self.assertSequenceEqual(person.friends.all(), friends)
|
2014-02-10 05:21:42 -05:00
|
|
|
|
|
|
|
def test_recursive_m2m_reverse_add(self):
|
2019-05-31 16:19:11 +02:00
|
|
|
# Add m2m for Anne in reverse direction.
|
2014-02-10 05:21:42 -05:00
|
|
|
self.b.friends.add(self.a)
|
2019-05-31 16:19:11 +02:00
|
|
|
self.assertSequenceEqual(self.a.friends.all(), [self.b, self.c, self.d])
|
|
|
|
self.assertSequenceEqual(self.b.friends.all(), [self.a])
|
2014-02-10 05:21:42 -05:00
|
|
|
|
|
|
|
def test_recursive_m2m_remove(self):
|
|
|
|
self.b.friends.remove(self.a)
|
2019-05-31 16:19:11 +02:00
|
|
|
self.assertSequenceEqual(self.a.friends.all(), [self.c, self.d])
|
|
|
|
self.assertSequenceEqual(self.b.friends.all(), [])
|
2010-10-09 16:26:48 +00:00
|
|
|
|
2014-02-10 05:21:42 -05:00
|
|
|
def test_recursive_m2m_clear(self):
|
2019-05-31 16:19:11 +02:00
|
|
|
# Clear m2m for Anne.
|
2014-02-10 05:21:42 -05:00
|
|
|
self.a.friends.clear()
|
2019-05-31 16:19:11 +02:00
|
|
|
self.assertSequenceEqual(self.a.friends.all(), [])
|
|
|
|
# Reverse m2m relationships should be removed.
|
|
|
|
self.assertSequenceEqual(self.c.friends.all(), [self.d])
|
|
|
|
self.assertSequenceEqual(self.d.friends.all(), [self.c])
|
2010-10-09 16:26:48 +00:00
|
|
|
|
2014-02-10 05:21:42 -05:00
|
|
|
def test_recursive_m2m_add_via_related_name(self):
|
2019-05-31 16:19:11 +02:00
|
|
|
# Add m2m with custom related name for Anne in reverse direction.
|
2014-02-10 05:21:42 -05:00
|
|
|
self.d.stalkers.add(self.a)
|
2019-05-31 16:19:11 +02:00
|
|
|
self.assertSequenceEqual(self.a.idols.all(), [self.d])
|
|
|
|
self.assertSequenceEqual(self.a.stalkers.all(), [])
|
2014-02-10 05:21:42 -05:00
|
|
|
|
|
|
|
def test_recursive_m2m_add_in_both_directions(self):
|
2019-05-31 16:19:11 +02:00
|
|
|
# Adding the same relation twice results in a single relation.
|
2014-02-10 05:21:42 -05:00
|
|
|
self.a.idols.add(self.d)
|
|
|
|
self.d.stalkers.add(self.a)
|
2019-05-31 16:19:11 +02:00
|
|
|
self.assertSequenceEqual(self.a.idols.all(), [self.d])
|
2014-02-10 05:21:42 -05:00
|
|
|
|
|
|
|
def test_recursive_m2m_related_to_self(self):
|
|
|
|
self.a.idols.add(self.a)
|
2019-05-31 16:19:11 +02:00
|
|
|
self.assertSequenceEqual(self.a.idols.all(), [self.a])
|
|
|
|
self.assertSequenceEqual(self.a.stalkers.all(), [self.a])
|
2019-04-19 18:12:04 +02:00
|
|
|
|
|
|
|
|
|
|
|
class RecursiveSymmetricalM2MThroughTests(TestCase):
|
|
|
|
@classmethod
|
|
|
|
def setUpTestData(cls):
|
|
|
|
cls.a, cls.b, cls.c, cls.d = [
|
|
|
|
Person.objects.create(name=name)
|
2022-02-03 20:24:19 +01:00
|
|
|
for name in ["Anne", "Bill", "Chuck", "David"]
|
2019-04-19 18:12:04 +02:00
|
|
|
]
|
2022-02-03 20:24:19 +01:00
|
|
|
cls.a.colleagues.add(
|
|
|
|
cls.b,
|
|
|
|
cls.c,
|
|
|
|
through_defaults={
|
|
|
|
"first_meet": datetime.date(2013, 1, 5),
|
|
|
|
},
|
|
|
|
)
|
2019-04-19 18:12:04 +02:00
|
|
|
# Add m2m for Anne and Chuck in reverse direction.
|
2022-02-03 20:24:19 +01:00
|
|
|
cls.d.colleagues.add(
|
|
|
|
cls.a,
|
|
|
|
cls.c,
|
|
|
|
through_defaults={
|
|
|
|
"first_meet": datetime.date(2015, 6, 15),
|
|
|
|
},
|
|
|
|
)
|
2019-04-19 18:12:04 +02:00
|
|
|
|
|
|
|
def test_recursive_m2m_all(self):
|
|
|
|
for person, colleagues in (
|
|
|
|
(self.a, [self.b, self.c, self.d]),
|
|
|
|
(self.b, [self.a]),
|
|
|
|
(self.c, [self.a, self.d]),
|
|
|
|
(self.d, [self.a, self.c]),
|
|
|
|
):
|
|
|
|
with self.subTest(person=person):
|
|
|
|
self.assertSequenceEqual(person.colleagues.all(), colleagues)
|
|
|
|
|
|
|
|
def test_recursive_m2m_reverse_add(self):
|
|
|
|
# Add m2m for Anne in reverse direction.
|
2022-02-03 20:24:19 +01:00
|
|
|
self.b.colleagues.add(
|
|
|
|
self.a,
|
|
|
|
through_defaults={
|
|
|
|
"first_meet": datetime.date(2013, 1, 5),
|
|
|
|
},
|
|
|
|
)
|
2022-04-14 12:12:13 +02:00
|
|
|
self.assertCountEqual(self.a.colleagues.all(), [self.b, self.c, self.d])
|
2019-04-19 18:12:04 +02:00
|
|
|
self.assertSequenceEqual(self.b.colleagues.all(), [self.a])
|
|
|
|
|
|
|
|
def test_recursive_m2m_remove(self):
|
|
|
|
self.b.colleagues.remove(self.a)
|
|
|
|
self.assertSequenceEqual(self.a.colleagues.all(), [self.c, self.d])
|
|
|
|
self.assertSequenceEqual(self.b.colleagues.all(), [])
|
|
|
|
|
|
|
|
def test_recursive_m2m_clear(self):
|
|
|
|
# Clear m2m for Anne.
|
|
|
|
self.a.colleagues.clear()
|
|
|
|
self.assertSequenceEqual(self.a.friends.all(), [])
|
|
|
|
# Reverse m2m relationships is removed.
|
|
|
|
self.assertSequenceEqual(self.c.colleagues.all(), [self.d])
|
|
|
|
self.assertSequenceEqual(self.d.colleagues.all(), [self.c])
|
|
|
|
|
|
|
|
def test_recursive_m2m_set(self):
|
|
|
|
# Set new relationships for Chuck.
|
2022-02-03 20:24:19 +01:00
|
|
|
self.c.colleagues.set(
|
|
|
|
[self.b, self.d],
|
|
|
|
through_defaults={
|
|
|
|
"first_meet": datetime.date(2013, 1, 5),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
self.assertSequenceEqual(self.c.colleagues.order_by("name"), [self.b, self.d])
|
2019-04-19 18:12:04 +02:00
|
|
|
# Reverse m2m relationships is removed.
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertSequenceEqual(self.a.colleagues.order_by("name"), [self.b, self.d])
|