#12247 closed (fixed)
update on empty queryset in which the update kwargs refer to inherited columns update all rows of base table
| Reported by: | Jacob Smullyan | Owned by: | matiasb | 
|---|---|---|---|
| Component: | Database layer (models, ORM) | Version: | 1.1 | 
| Severity: | Keywords: | ||
| Cc: | Triage Stage: | Accepted | |
| Has patch: | yes | Needs documentation: | no | 
| Needs tests: | no | Patch needs improvement: | no | 
| Easy pickings: | no | UI/UX: | no | 
Description
A queryset that happens to return nothing if evaluated, when updated, may erroneously update all rows if the queryset refers to an inherited model and the update columns are in the base table.  I'm testing with django 1.1.X on rev 11752.  This is a bug with potentially very serious consequences.  (Note: I believe this bug to be distinct, although similar in its effects, to #12142.)
Here is test code to produce the problem.  One of the below tests will fail (test_empty_update_with_inheritance).
#models.py
from django.db import models
# Create your models here.
class A(models.Model):
    x=models.IntegerField(default=10)
class B(models.Model):
    a=models.ForeignKey(A)
    y=models.IntegerField(default=10)
class C(models.Model):
    y=models.IntegerField(default=10)
class D(C):
    a=models.ForeignKey(A)
# tests.py
from django.test import TestCase
from models import A, B, D
class SimpleTest(TestCase):
    def setUp(self):
        self.a1=A.objects.create()
        self.a2=A.objects.create()
        for x in range(20):
            B.objects.create(a=self.a1)
            D.objects.create(a=self.a1)
    def tearDown(self):
        B.objects.all().delete()
        A.objects.all().delete()
        
        
    def test_nonempty_update(self):
        """
        Test that update changes that right number of rows for a nonempty queryset
        """
        num_updated=self.a1.b_set.update(y=100)
        self.failUnlessEqual(num_updated, 20)
        cnt=B.objects.filter(y=100).count()
        self.failUnlessEqual(cnt, 20)
    def test_empty_update(self):
        """
        Test that update changes that right number of rows for an empty queryset
        """
        num_updated=self.a2.b_set.update(y=100)
        self.failUnlessEqual(num_updated, 0)
        cnt=B.objects.filter(y=100).count()
        self.failUnlessEqual(cnt, 0)
    def test_nonempty_update_with_inheritance(self):
        """
        Test that update changes that right number of rows for an empty queryset
        when the update affects only a base table
        """
        num_updated=self.a1.d_set.update(y=100)
        self.failUnlessEqual(num_updated, 20)
        cnt=D.objects.filter(y=100).count()
        self.failUnlessEqual(cnt, 20)
    def test_empty_update_with_inheritance(self):
        """
        Test that update changes that right number of rows for an empty queryset
        when the update affects only a base table
        """
        num_updated=self.a2.d_set.update(y=100)
        self.failUnlessEqual(num_updated, 0)
        cnt=D.objects.filter(y=100).count()
        self.failUnlessEqual(cnt, 0)
      Attachments (2)
Change History (14)
comment:1 by , 16 years ago
| milestone: | → 1.2 | 
|---|---|
| Triage Stage: | Unreviewed → Accepted | 
comment:2 by , 16 years ago
| Owner: | changed from to | 
|---|---|
| Status: | new → assigned | 
comment:3 by , 16 years ago
| Owner: | removed | 
|---|---|
| Status: | assigned → new | 
comment:4 by , 16 years ago
| Owner: | set to | 
|---|---|
| Status: | new → assigned | 
by , 16 years ago
| Attachment: | 12247-1.diff added | 
|---|
comment:5 by , 16 years ago
After long debugging to find the issue, added patch to manage empty related_ids when building related_updates queries.
Thanks to nessita for the previous debugging info and jsmullyan for the provided tests.
comment:6 by , 16 years ago
| Has patch: | set | 
|---|
comment:7 by , 16 years ago
The patch looks pretty good, and seems to fix the problem, but I wonder if the bug isn't another level down. This seems to be triggered when self.related_ids is [], so we're essentially calling query.add_filter(('pk__in', [])). It seems like pk__in=[] should essentially be a no-op, but it sounds like it's actually selecting everything.
by , 16 years ago
| Attachment: | 12247-jkocherhans.diff added | 
|---|
Moved the models and tests from matiasb's patch into the existing update test and changed the formatting to follow pep8.
comment:9 by , 16 years ago
@jkocherhans - add_filter(pkin=[]) *is* a no-op (or rather, it raises EmptyResultSet). The issue here is that the "if self.related_ids" condition means that you *aren't* adding the [] filter, which means that you are left with a default "update everything" filter. Changing the condition to "if self.related_ids is None" means that the [] filter will be added, and thereby shortcut the update into a no-op.
comment:10 by , 16 years ago
| Resolution: | → fixed | 
|---|---|
| Status: | assigned → closed | 
Fix for related updates query construction