diff options
author | François Freitag <mail@franek.fr> | 2018-04-13 18:15:22 -0700 |
---|---|---|
committer | Tim Graham <timograham@gmail.com> | 2018-04-23 12:39:43 -0400 |
commit | 3fca95e1ad5b355f7813b98c97a194a30f2ab47b (patch) | |
tree | e8357bddc0c38b6035f2dbd03d67d45728ab7d82 /tests/model_forms | |
parent | 9ec77f3d6633691c9bbf31d10af2313689170c85 (diff) | |
download | django-3fca95e1ad5b355f7813b98c97a194a30f2ab47b.tar.gz |
Fixed #28312 -- Made ModelChoiceIterator.__len__() more memory-efficient.
Instead of loading all QuerySet results in memory, count the number of
entries. This adds an extra query when list() or tuple() is called on the
choices (because both call __len__() then __iter__()) but uses less
memory since the QuerySet results won't be cached. In most cases, the
choices will only be iterated on, meaning that __len__() won't be called
and only one query will be executed.
Diffstat (limited to 'tests/model_forms')
-rw-r--r-- | tests/model_forms/test_modelchoicefield.py | 30 | ||||
-rw-r--r-- | tests/model_forms/tests.py | 2 |
2 files changed, 20 insertions, 12 deletions
diff --git a/tests/model_forms/test_modelchoicefield.py b/tests/model_forms/test_modelchoicefield.py index 733175823f..975873c7f9 100644 --- a/tests/model_forms/test_modelchoicefield.py +++ b/tests/model_forms/test_modelchoicefield.py @@ -96,6 +96,25 @@ class ModelChoiceFieldTests(TestCase): (self.c3.pk, 'category Third'), ]) + def test_choices_freshness(self): + f = forms.ModelChoiceField(Category.objects.all()) + self.assertEqual(len(f.choices), 4) + self.assertEqual(list(f.choices), [ + ('', '---------'), + (self.c1.pk, 'Entertainment'), + (self.c2.pk, 'A test'), + (self.c3.pk, 'Third'), + ]) + c4 = Category.objects.create(name='Fourth', slug='4th', url='4th') + self.assertEqual(len(f.choices), 5) + self.assertEqual(list(f.choices), [ + ('', '---------'), + (self.c1.pk, 'Entertainment'), + (self.c2.pk, 'A test'), + (self.c3.pk, 'Third'), + (c4.pk, 'Fourth'), + ]) + def test_deepcopies_widget(self): class ModelChoiceForm(forms.Form): category = forms.ModelChoiceField(Category.objects.all()) @@ -257,17 +276,6 @@ class ModelChoiceFieldTests(TestCase): (self.c3.pk, 'Third'), ]) - def test_queryset_result_cache_is_reused(self): - f = forms.ModelChoiceField(Category.objects.all()) - with self.assertNumQueries(1): - # list() calls __len__() and __iter__(); no duplicate queries. - self.assertEqual(list(f.choices), [ - ('', '---------'), - (self.c1.pk, 'Entertainment'), - (self.c2.pk, 'A test'), - (self.c3.pk, 'Third'), - ]) - def test_num_queries(self): """ Widgets that render multiple subwidgets shouldn't make more than one diff --git a/tests/model_forms/tests.py b/tests/model_forms/tests.py index 4276bc68f1..1d6d9efa96 100644 --- a/tests/model_forms/tests.py +++ b/tests/model_forms/tests.py @@ -2345,7 +2345,7 @@ class OtherModelFormTests(TestCase): return ', '.join(c.name for c in obj.colours.all()) field = ColorModelChoiceField(ColourfulItem.objects.prefetch_related('colours')) - with self.assertNumQueries(2): # would be 3 if prefetch is ignored + with self.assertNumQueries(3): # would be 4 if prefetch is ignored self.assertEqual(tuple(field.choices), ( ('', '---------'), (multicolor_item.pk, 'blue, red'), |