Opened 27 hours ago

Last modified 15 hours ago

#36074 closed Bug

Updating a model instance with a composite primary key through .save() results in unnecessary re-assignment of primary key fields — at Version 1

Reported by: Simon Charette Owned by: Simon Charette
Component: Database layer (models, ORM) Version: dev
Severity: Release blocker Keywords:
Cc: Triage Stage: Ready for checkin
Has patch: yes Needs documentation: no
Needs tests: no Patch needs improvement: no
Easy pickings: no UI/UX: no

Description (last modified by Simon Charette)

Looking at the code that gets involved and the generated SQL It appears that every save that results in an UPDATE for a model with a composite primary key also includes all the primary key field in the updated fields.

For example, for a model of the form

class User(models.Model):
    tenant = models.ForeignKey(Tenant)
    id = models.UUIDField(default=uuid.uuid4)
    username = models.EmailField()

Then doing

user = User.objects.create(tenant=tenant, id=1, email="foo@bar.org")
user.email = "something@example.com"
user.save()
# Will result in UPDATE user SET tenant_id = %s, id = %s, email = %s WHERE tenant_id = %s AND id = %s
#                                ^^^^^^^^^^^^^^^^^^^^^^^  <- tenant_id and id should not be marked for UPDATE

Will result in the following SQL

INSERT INTO user(tenant_id, id, email) VALUES (%s, %s, %s)
UPDATE user SET tenant_id = %s, id = %s, email = %s WHERE tenant_id = %s AND id = %s

But the UPDATE statement should not set tenant_id and id as they are part of the primary key lookup. The UPDATE should be

UPDATE user SET email = %s WHERE tenant_id = %s AND id = %s

which is due to a non-audited usage of .primary_key in `save_tables

  • django/db/models/base.py

    diff --git a/django/db/models/base.py b/django/db/models/base.py
    index a7a26b405c..6d66080c20 100644
    a b def _save_table(  
    10911091        for a single table.
    10921092        """
    10931093        meta = cls._meta
     1094        pk_fields = meta.pk_fields
    10941095        non_pks_non_generated = [
    10951096            f
    10961097            for f in meta.local_concrete_fields
    1097             if not f.primary_key and not f.generated
     1098            if f not in pk_fields and not f.generated
    10981099        ]
    10991100
    11001101        if update_fields:

and makes me fear there might be others lying around that went unnoticed. If you grep for .primary_key you'll notice that most of the checks against this field need to be adjusted to use field in opts.pk_fields instead which makes me wonder if we took the right decision by not setting this flag to True when included as a member of CompositePrimaryKey.

Change History (1)

comment:1 by Simon Charette, 26 hours ago

Description: modified (diff)
Note: See TracTickets for help on using tickets.
Back to Top