summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--AUTHORS1
-rw-r--r--django/forms/models.py5
-rw-r--r--tests/model_formsets/tests.py31
3 files changed, 34 insertions, 3 deletions
diff --git a/AUTHORS b/AUTHORS
index bf5bf03f12..486a50f0e2 100644
--- a/AUTHORS
+++ b/AUTHORS
@@ -916,6 +916,7 @@ answer newbie questions, and generally made Django that much better:
Rob Hudson <https://rob.cogit8.org/>
Rob Nguyen <tienrobertnguyenn@gmail.com>
Robin Munn <http://www.geekforgod.com/>
+ Rodrigo Bastos Vieira <rodrigo.vieira@gmail.com>
Rodrigo Pinheiro Marques de Araújo <fenrrir@gmail.com>
Roel Delos Reyes <https://roelzkie.dev>
Rohith P R <https://rohithpr.com>
diff --git a/django/forms/models.py b/django/forms/models.py
index 104369c0b0..a53f119995 100644
--- a/django/forms/models.py
+++ b/django/forms/models.py
@@ -776,8 +776,9 @@ class BaseModelFormSet(BaseFormSet, AltersData):
# If the queryset isn't already ordered we need to add an
# artificial ordering here to make sure that all formsets
# constructed from this queryset have the same form order.
- if not qs.ordered:
- qs = qs.order_by(self.model._meta.pk.name)
+ if not qs.totally_ordered:
+ current_ordering = qs.query.order_by or qs.model._meta.ordering or []
+ qs = qs.order_by(*current_ordering, "pk")
# Removed queryset limiting here. As per discussion re: #13023
# on django-dev, max_num should not prevent existing
diff --git a/tests/model_formsets/tests.py b/tests/model_formsets/tests.py
index 40f206277b..9436642573 100644
--- a/tests/model_formsets/tests.py
+++ b/tests/model_formsets/tests.py
@@ -1782,7 +1782,36 @@ class ModelFormsetTest(TestCase):
formset = ProductFormSet()
queryset = formset.get_queryset()
- self.assertEqual(queryset.query.order_by, ("id",))
+ self.assertEqual(queryset.query.order_by, ("pk",))
+
+ def test_get_queryset_appends_pk_to_explicit_queryset_ordering(self):
+ # Author.name is non-unique, so order_by("name") is not totally
+ # ordered.
+ AuthorFormSet = modelformset_factory(Author, fields="__all__")
+ formset = AuthorFormSet(queryset=Author.objects.order_by("name"))
+
+ queryset = formset.get_queryset()
+ self.assertEqual(queryset.query.order_by, ("name", "pk"))
+
+ def test_get_queryset_appends_pk_to_meta_ordering(self):
+ # Author has Meta.ordering = ("name",) which is not deterministic
+ # by itself.
+ AuthorFormSet = modelformset_factory(Author, fields="__all__")
+ formset = AuthorFormSet()
+
+ queryset = formset.get_queryset()
+ self.assertEqual(queryset.query.order_by, ("name", "pk"))
+
+ def test_get_queryset_unchanged_when_already_totally_ordered(self):
+ # Ordering by pk is already totally ordered; pk must not be
+ # appended again.
+ AuthorFormSet = modelformset_factory(Author, fields="__all__")
+ formset = AuthorFormSet(queryset=Author.objects.order_by("pk"))
+
+ queryset = formset.get_queryset()
+ # Must be exactly ("pk",), not ("pk", "pk").
+ self.assertEqual(queryset.query.order_by, ("pk",))
+ self.assertIs(queryset.totally_ordered, True)
def test_prevent_duplicates_from_with_the_same_formset(self):
FormSet = modelformset_factory(Product, fields="__all__", extra=2)