diff --git a/django/db/models/query.py b/django/db/models/query.py index de49e1c58c..be8580f2c4 100644 --- a/django/db/models/query.py +++ b/django/db/models/query.py @@ -732,11 +732,8 @@ class QuerySet(AltersData): "update_fields." ) if unique_fields: - # Primary key is allowed in unique_fields. unique_fields = [ - self.model._meta.get_field(name) - for name in unique_fields - if name != "pk" + self.model._meta.get_field(name) for name in unique_fields ] if any(not f.concrete or f.many_to_many for f in unique_fields): raise ValueError( @@ -785,6 +782,12 @@ class QuerySet(AltersData): raise ValueError("Can't bulk create a multi-table inherited model") if not objs: return objs + opts = self.model._meta + if unique_fields: + # Primary key is allowed in unique_fields. + unique_fields = [ + opts.pk.name if name == "pk" else name for name in unique_fields + ] on_conflict = self._check_bulk_create_options( ignore_conflicts, update_conflicts, @@ -792,7 +795,6 @@ class QuerySet(AltersData): unique_fields, ) self._for_write = True - opts = self.model._meta fields = opts.concrete_fields objs = list(objs) self._prepare_for_bulk_create(objs) diff --git a/docs/releases/4.1.4.txt b/docs/releases/4.1.4.txt index c8f97b5278..7cdd2521ea 100644 --- a/docs/releases/4.1.4.txt +++ b/docs/releases/4.1.4.txt @@ -20,3 +20,6 @@ Bugfixes * Fixed a bug in Django 4.1 that caused a crash of ``acreate()``, ``aget_or_create()``, and ``aupdate_or_create()`` asynchronous methods for related managers (:ticket:`34139`). + +* Fixed a bug in Django 4.1 that caused a crash of ``QuerySet.bulk_create()`` + with ``"pk"`` in ``unique_fields`` (:ticket:`34177`). diff --git a/tests/bulk_create/tests.py b/tests/bulk_create/tests.py index bc2900110d..6c490d1235 100644 --- a/tests/bulk_create/tests.py +++ b/tests/bulk_create/tests.py @@ -595,6 +595,39 @@ class BulkCreateTests(TestCase): def test_update_conflicts_two_fields_unique_fields_second(self): self._test_update_conflicts_two_fields(["f2"]) + @skipUnlessDBFeature( + "supports_update_conflicts", "supports_update_conflicts_with_target" + ) + def test_update_conflicts_unique_fields_pk(self): + TwoFields.objects.bulk_create( + [ + TwoFields(f1=1, f2=1, name="a"), + TwoFields(f1=2, f2=2, name="b"), + ] + ) + self.assertEqual(TwoFields.objects.count(), 2) + + obj1 = TwoFields.objects.get(f1=1) + obj2 = TwoFields.objects.get(f1=2) + conflicting_objects = [ + TwoFields(pk=obj1.pk, f1=3, f2=3, name="c"), + TwoFields(pk=obj2.pk, f1=4, f2=4, name="d"), + ] + TwoFields.objects.bulk_create( + conflicting_objects, + update_conflicts=True, + unique_fields=["pk"], + update_fields=["name"], + ) + self.assertEqual(TwoFields.objects.count(), 2) + self.assertCountEqual( + TwoFields.objects.values("f1", "f2", "name"), + [ + {"f1": 1, "f2": 1, "name": "c"}, + {"f1": 2, "f2": 2, "name": "d"}, + ], + ) + @skipUnlessDBFeature( "supports_update_conflicts", "supports_update_conflicts_with_target" )