Fixed #31926 -- Fixed recreating queryset with FilteredRelation when using a pickled Query.

In a pickled join, the join_fields had the same values, but weren't the
same object (contrary to when not pickling the QuerySet).
This commit is contained in:
David-Wobrock 2020-10-03 20:30:53 +02:00 committed by Mariusz Felisiak
parent 292b3be698
commit c32d8f33d8
2 changed files with 69 additions and 0 deletions

View File

@ -11,6 +11,7 @@ they're the closest concept currently available.
from django.core import exceptions from django.core import exceptions
from django.utils.functional import cached_property from django.utils.functional import cached_property
from django.utils.hashable import make_hashable
from . import BLANK_CHOICE_DASH from . import BLANK_CHOICE_DASH
from .mixins import FieldCacheMixin from .mixins import FieldCacheMixin
@ -115,6 +116,28 @@ class ForeignObjectRel(FieldCacheMixin):
self.related_model._meta.model_name, self.related_model._meta.model_name,
) )
@property
def identity(self):
return (
self.field,
self.model,
self.related_name,
self.related_query_name,
tuple(sorted(make_hashable(self.limit_choices_to))),
self.parent_link,
self.on_delete,
self.symmetrical,
self.multiple,
)
def __eq__(self, other):
if not isinstance(other, self.__class__):
return NotImplemented
return self.identity == other.identity
def __hash__(self):
return hash(self.identity)
def get_choices( def get_choices(
self, include_blank=True, blank_choice=BLANK_CHOICE_DASH, self, include_blank=True, blank_choice=BLANK_CHOICE_DASH,
limit_choices_to=None, ordering=(), limit_choices_to=None, ordering=(),
@ -215,6 +238,10 @@ class ManyToOneRel(ForeignObjectRel):
state.pop('related_model', None) state.pop('related_model', None)
return state return state
@property
def identity(self):
return super().identity + (self.field_name,)
def get_related_field(self): def get_related_field(self):
""" """
Return the Field in the 'to' object to which this relationship is tied. Return the Field in the 'to' object to which this relationship is tied.
@ -279,6 +306,14 @@ class ManyToManyRel(ForeignObjectRel):
self.symmetrical = symmetrical self.symmetrical = symmetrical
self.db_constraint = db_constraint self.db_constraint = db_constraint
@property
def identity(self):
return super().identity + (
self.through,
self.through_fields,
self.db_constraint,
)
def get_related_field(self): def get_related_field(self):
""" """
Return the field in the 'to' object to which this relationship is tied. Return the field in the 'to' object to which this relationship is tied.

View File

@ -219,6 +219,40 @@ class PickleabilityTestCase(TestCase):
with self.assertNumQueries(0): with self.assertNumQueries(0):
self.assert_pickles(groups) self.assert_pickles(groups)
def test_pickle_filteredrelation(self):
group = Group.objects.create(name='group')
event_1 = Event.objects.create(title='Big event', group=group)
event_2 = Event.objects.create(title='Small event', group=group)
Happening.objects.bulk_create([
Happening(event=event_1, number1=5),
Happening(event=event_2, number1=3),
])
groups = Group.objects.annotate(
big_events=models.FilteredRelation(
'event',
condition=models.Q(event__title__startswith='Big'),
),
).annotate(sum_number=models.Sum('big_events__happening__number1'))
groups_query = pickle.loads(pickle.dumps(groups.query))
groups = Group.objects.all()
groups.query = groups_query
self.assertEqual(groups.get().sum_number, 5)
def test_pickle_filteredrelation_m2m(self):
group = Group.objects.create(name='group')
m2mmodel = M2MModel.objects.create()
m2mmodel.groups.add(group)
groups = Group.objects.annotate(
first_m2mmodels=models.FilteredRelation(
'm2mmodel',
condition=models.Q(m2mmodel__pk__lt=10),
),
).annotate(count_groups=models.Count('first_m2mmodels__groups'))
groups_query = pickle.loads(pickle.dumps(groups.query))
groups = Group.objects.all()
groups.query = groups_query
self.assertEqual(groups.get().count_groups, 1)
def test_annotation_with_callable_default(self): def test_annotation_with_callable_default(self):
# Happening.when has a callable default of datetime.datetime.now. # Happening.when has a callable default of datetime.datetime.now.
qs = Happening.objects.annotate(latest_time=models.Max('when')) qs = Happening.objects.annotate(latest_time=models.Max('when'))