[1.6.x] Fixed #20988 -- Added model meta option select_on_save
The option can be used to force pre 1.6 style SELECT on save behaviour.
This is needed in case the database returns zero updated rows even if
there is a matching row in the DB. One such case is PostgreSQL update
trigger that returns NULL.
Reviewed by Tim Graham.
Refs #16649
Backport of e973ee6a98
from master
Conflicts:
django/db/models/options.py
tests/basic/tests.py
This commit is contained in:
parent
cd10e998b6
commit
76e38a2177
|
@ -635,7 +635,9 @@ class Model(six.with_metaclass(ModelBase)):
|
||||||
base_qs = cls._base_manager.using(using)
|
base_qs = cls._base_manager.using(using)
|
||||||
values = [(f, None, (getattr(self, f.attname) if raw else f.pre_save(self, False)))
|
values = [(f, None, (getattr(self, f.attname) if raw else f.pre_save(self, False)))
|
||||||
for f in non_pks]
|
for f in non_pks]
|
||||||
updated = self._do_update(base_qs, using, pk_val, values, update_fields)
|
forced_update = update_fields or force_update
|
||||||
|
updated = self._do_update(base_qs, using, pk_val, values, update_fields,
|
||||||
|
forced_update)
|
||||||
if force_update and not updated:
|
if force_update and not updated:
|
||||||
raise DatabaseError("Forced update did not affect any rows.")
|
raise DatabaseError("Forced update did not affect any rows.")
|
||||||
if update_fields and not updated:
|
if update_fields and not updated:
|
||||||
|
@ -659,21 +661,27 @@ class Model(six.with_metaclass(ModelBase)):
|
||||||
setattr(self, meta.pk.attname, result)
|
setattr(self, meta.pk.attname, result)
|
||||||
return updated
|
return updated
|
||||||
|
|
||||||
def _do_update(self, base_qs, using, pk_val, values, update_fields):
|
def _do_update(self, base_qs, using, pk_val, values, update_fields, forced_update):
|
||||||
"""
|
"""
|
||||||
This method will try to update the model. If the model was updated (in
|
This method will try to update the model. If the model was updated (in
|
||||||
the sense that an update query was done and a matching row was found
|
the sense that an update query was done and a matching row was found
|
||||||
from the DB) the method will return True.
|
from the DB) the method will return True.
|
||||||
"""
|
"""
|
||||||
|
filtered = base_qs.filter(pk=pk_val)
|
||||||
if not values:
|
if not values:
|
||||||
# We can end up here when saving a model in inheritance chain where
|
# We can end up here when saving a model in inheritance chain where
|
||||||
# update_fields doesn't target any field in current model. In that
|
# update_fields doesn't target any field in current model. In that
|
||||||
# case we just say the update succeeded. Another case ending up here
|
# case we just say the update succeeded. Another case ending up here
|
||||||
# is a model with just PK - in that case check that the PK still
|
# is a model with just PK - in that case check that the PK still
|
||||||
# exists.
|
# exists.
|
||||||
return update_fields is not None or base_qs.filter(pk=pk_val).exists()
|
return update_fields is not None or filtered.exists()
|
||||||
else:
|
if self._meta.select_on_save and not forced_update:
|
||||||
return base_qs.filter(pk=pk_val)._update(values) > 0
|
if filtered.exists():
|
||||||
|
filtered._update(values)
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
return False
|
||||||
|
return filtered._update(values) > 0
|
||||||
|
|
||||||
def _do_insert(self, manager, using, fields, update_pk, raw):
|
def _do_insert(self, manager, using, fields, update_pk, raw):
|
||||||
"""
|
"""
|
||||||
|
|
|
@ -22,7 +22,7 @@ DEFAULT_NAMES = ('verbose_name', 'verbose_name_plural', 'db_table', 'ordering',
|
||||||
'unique_together', 'permissions', 'get_latest_by',
|
'unique_together', 'permissions', 'get_latest_by',
|
||||||
'order_with_respect_to', 'app_label', 'db_tablespace',
|
'order_with_respect_to', 'app_label', 'db_tablespace',
|
||||||
'abstract', 'managed', 'proxy', 'swappable', 'auto_created',
|
'abstract', 'managed', 'proxy', 'swappable', 'auto_created',
|
||||||
'index_together')
|
'index_together', 'select_on_save')
|
||||||
|
|
||||||
|
|
||||||
@python_2_unicode_compatible
|
@python_2_unicode_compatible
|
||||||
|
@ -36,6 +36,7 @@ class Options(object):
|
||||||
self.ordering = []
|
self.ordering = []
|
||||||
self.unique_together = []
|
self.unique_together = []
|
||||||
self.index_together = []
|
self.index_together = []
|
||||||
|
self.select_on_save = False
|
||||||
self.permissions = []
|
self.permissions = []
|
||||||
self.object_name, self.app_label = None, app_label
|
self.object_name, self.app_label = None, app_label
|
||||||
self.get_latest_by = None
|
self.get_latest_by = None
|
||||||
|
|
|
@ -305,16 +305,23 @@ follows this algorithm:
|
||||||
* If the object's primary key attribute is *not* set or if the ``UPDATE``
|
* If the object's primary key attribute is *not* set or if the ``UPDATE``
|
||||||
didn't update anything, Django executes an ``INSERT``.
|
didn't update anything, Django executes an ``INSERT``.
|
||||||
|
|
||||||
.. versionchanged:: 1.6
|
|
||||||
|
|
||||||
Previously Django used ``SELECT`` - if not found ``INSERT`` else ``UPDATE``
|
|
||||||
algorithm. The old algorithm resulted in one more query in ``UPDATE`` case.
|
|
||||||
|
|
||||||
The one gotcha here is that you should be careful not to specify a primary-key
|
The one gotcha here is that you should be careful not to specify a primary-key
|
||||||
value explicitly when saving new objects, if you cannot guarantee the
|
value explicitly when saving new objects, if you cannot guarantee the
|
||||||
primary-key value is unused. For more on this nuance, see `Explicitly specifying
|
primary-key value is unused. For more on this nuance, see `Explicitly specifying
|
||||||
auto-primary-key values`_ above and `Forcing an INSERT or UPDATE`_ below.
|
auto-primary-key values`_ above and `Forcing an INSERT or UPDATE`_ below.
|
||||||
|
|
||||||
|
.. versionchanged:: 1.6
|
||||||
|
|
||||||
|
Previously Django did a ``SELECT`` when the primary key attribute was set.
|
||||||
|
If the ``SELECT`` found a row, then Django did an ``UPDATE``, otherwise it
|
||||||
|
did an ``INSERT``. The old algorithm results in one more query in the
|
||||||
|
``UPDATE`` case. There are some rare cases where the database doesn't
|
||||||
|
report that a row was updated even if the database contains a row for the
|
||||||
|
object's primary key value. An example is the PostgreSQL ``ON UPDATE``
|
||||||
|
trigger which returns ``NULL``. In such cases it is possible to revert to the
|
||||||
|
old algorithm by setting the :attr:`~django.db.models.Options.select_on_save`
|
||||||
|
option to ``True``.
|
||||||
|
|
||||||
.. _ref-models-force-insert:
|
.. _ref-models-force-insert:
|
||||||
|
|
||||||
Forcing an INSERT or UPDATE
|
Forcing an INSERT or UPDATE
|
||||||
|
|
|
@ -238,6 +238,28 @@ Django quotes column and table names behind the scenes.
|
||||||
If ``proxy = True``, a model which subclasses another model will be treated as
|
If ``proxy = True``, a model which subclasses another model will be treated as
|
||||||
a :ref:`proxy model <proxy-models>`.
|
a :ref:`proxy model <proxy-models>`.
|
||||||
|
|
||||||
|
``select_on_save``
|
||||||
|
------------------
|
||||||
|
|
||||||
|
.. attribute:: Options.select_on_save
|
||||||
|
|
||||||
|
.. versionadded:: 1.6
|
||||||
|
|
||||||
|
Determines if Django will use the pre-1.6
|
||||||
|
:meth:`django.db.models.Model.save()` algorithm. The old algorithm
|
||||||
|
uses ``SELECT`` to determine if there is an existing row to be updated.
|
||||||
|
The new algorith tries an ``UPDATE`` directly. In some rare cases the
|
||||||
|
``UPDATE`` of an existing row isn't visible to Django. An example is the
|
||||||
|
PostgreSQL ``ON UPDATE`` trigger which returns ``NULL``. In such cases the
|
||||||
|
new algorithm will end up doing an ``INSERT`` even when a row exists in
|
||||||
|
the database.
|
||||||
|
|
||||||
|
Usually there is no need to set this attribute. The default is
|
||||||
|
``False``.
|
||||||
|
|
||||||
|
See :meth:`django.db.models.Model.save()` for more about the old and
|
||||||
|
new saving algorithm.
|
||||||
|
|
||||||
``unique_together``
|
``unique_together``
|
||||||
-------------------
|
-------------------
|
||||||
|
|
||||||
|
|
|
@ -138,6 +138,22 @@ A :djadmin:`check` management command was added, enabling you to verify if your
|
||||||
current configuration (currently oriented at settings) is compatible with the
|
current configuration (currently oriented at settings) is compatible with the
|
||||||
current version of Django.
|
current version of Django.
|
||||||
|
|
||||||
|
:meth:`Model.save() <django.db.models.Model.save()>` algorithm changed
|
||||||
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
|
||||||
|
The :meth:`Model.save() <django.db.models.Model.save()>` method now
|
||||||
|
tries to directly ``UPDATE`` the database if the instance has a primary
|
||||||
|
key value. Previously ``SELECT`` was performed to determine if ``UPDATE``
|
||||||
|
or ``INSERT`` were needed. The new algorithm needs only one query for
|
||||||
|
updating an existing row while the old algorithm needed two. See
|
||||||
|
:meth:`Model.save() <django.db.models.Model.save()>` for more details.
|
||||||
|
|
||||||
|
In some rare cases the database doesn't report that a matching row was
|
||||||
|
found when doing an ``UPDATE``. An example is the PostgreSQL ``ON UPDATE``
|
||||||
|
trigger which returns ``NULL``. In such cases it is possible to set
|
||||||
|
:attr:`django.db.models.Options.select_on_save` flag to force saving to
|
||||||
|
use the old algorithm.
|
||||||
|
|
||||||
Minor features
|
Minor features
|
||||||
~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~
|
||||||
|
|
||||||
|
@ -222,10 +238,6 @@ Minor features
|
||||||
* Generic :class:`~django.contrib.gis.db.models.GeometryField` is now editable
|
* Generic :class:`~django.contrib.gis.db.models.GeometryField` is now editable
|
||||||
with the OpenLayers widget in the admin.
|
with the OpenLayers widget in the admin.
|
||||||
|
|
||||||
* The :meth:`Model.save() <django.db.models.Model.save()>` will do
|
|
||||||
``UPDATE`` - if not updated - ``INSERT`` instead of ``SELECT`` - if not
|
|
||||||
found ``INSERT`` else ``UPDATE`` in case the model's primary key is set.
|
|
||||||
|
|
||||||
* The documentation contains a :doc:`deployment checklist
|
* The documentation contains a :doc:`deployment checklist
|
||||||
</howto/deployment/checklist>`.
|
</howto/deployment/checklist>`.
|
||||||
|
|
||||||
|
|
|
@ -19,6 +19,11 @@ class Article(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.headline
|
return self.headline
|
||||||
|
|
||||||
|
class ArticleSelectOnSave(Article):
|
||||||
|
class Meta:
|
||||||
|
proxy = True
|
||||||
|
select_on_save = True
|
||||||
|
|
||||||
@python_2_unicode_compatible
|
@python_2_unicode_compatible
|
||||||
class SelfRef(models.Model):
|
class SelfRef(models.Model):
|
||||||
selfref = models.ForeignKey('self', null=True, blank=True,
|
selfref = models.ForeignKey('self', null=True, blank=True,
|
||||||
|
|
|
@ -5,13 +5,14 @@ import threading
|
||||||
|
|
||||||
from django.core.exceptions import ObjectDoesNotExist, MultipleObjectsReturned
|
from django.core.exceptions import ObjectDoesNotExist, MultipleObjectsReturned
|
||||||
from django.db import connections, DEFAULT_DB_ALIAS
|
from django.db import connections, DEFAULT_DB_ALIAS
|
||||||
|
from django.db import DatabaseError
|
||||||
from django.db.models.fields import Field, FieldDoesNotExist
|
from django.db.models.fields import Field, FieldDoesNotExist
|
||||||
from django.db.models.query import QuerySet, EmptyQuerySet, ValuesListQuerySet
|
from django.db.models.query import QuerySet, EmptyQuerySet, ValuesListQuerySet
|
||||||
from django.test import TestCase, TransactionTestCase, skipIfDBFeature, skipUnlessDBFeature
|
from django.test import TestCase, TransactionTestCase, skipIfDBFeature, skipUnlessDBFeature
|
||||||
from django.utils import six
|
from django.utils import six
|
||||||
from django.utils.translation import ugettext_lazy
|
from django.utils.translation import ugettext_lazy
|
||||||
|
|
||||||
from .models import Article, SelfRef
|
from .models import Article, SelfRef, ArticleSelectOnSave
|
||||||
|
|
||||||
|
|
||||||
class ModelTest(TestCase):
|
class ModelTest(TestCase):
|
||||||
|
@ -712,3 +713,61 @@ class ConcurrentSaveTests(TransactionTestCase):
|
||||||
t.join()
|
t.join()
|
||||||
a.save()
|
a.save()
|
||||||
self.assertEqual(Article.objects.get(pk=a.pk).headline, 'foo')
|
self.assertEqual(Article.objects.get(pk=a.pk).headline, 'foo')
|
||||||
|
|
||||||
|
|
||||||
|
class SelectOnSaveTests(TestCase):
|
||||||
|
def test_select_on_save(self):
|
||||||
|
a1 = Article.objects.create(pub_date=datetime.now())
|
||||||
|
with self.assertNumQueries(1):
|
||||||
|
a1.save()
|
||||||
|
asos = ArticleSelectOnSave.objects.create(pub_date=datetime.now())
|
||||||
|
with self.assertNumQueries(2):
|
||||||
|
asos.save()
|
||||||
|
with self.assertNumQueries(1):
|
||||||
|
asos.save(force_update=True)
|
||||||
|
Article.objects.all().delete()
|
||||||
|
with self.assertRaises(DatabaseError):
|
||||||
|
with self.assertNumQueries(1):
|
||||||
|
asos.save(force_update=True)
|
||||||
|
|
||||||
|
def test_select_on_save_lying_update(self):
|
||||||
|
"""
|
||||||
|
Test that select_on_save works correctly if the database
|
||||||
|
doesn't return correct information about matched rows from
|
||||||
|
UPDATE.
|
||||||
|
"""
|
||||||
|
# Change the manager to not return "row matched" for update().
|
||||||
|
# We are going to change the Article's _base_manager class
|
||||||
|
# dynamically. This is a bit of a hack, but it seems hard to
|
||||||
|
# test this properly otherwise. Article's manager, because
|
||||||
|
# proxy models use their parent model's _base_manager.
|
||||||
|
|
||||||
|
orig_class = Article._base_manager.__class__
|
||||||
|
|
||||||
|
class FakeQuerySet(QuerySet):
|
||||||
|
# Make sure the _update method below is in fact called.
|
||||||
|
called = False
|
||||||
|
|
||||||
|
def _update(self, *args, **kwargs):
|
||||||
|
FakeQuerySet.called = True
|
||||||
|
super(FakeQuerySet, self)._update(*args, **kwargs)
|
||||||
|
return 0
|
||||||
|
|
||||||
|
class FakeManager(orig_class):
|
||||||
|
def get_queryset(self):
|
||||||
|
return FakeQuerySet(self.model)
|
||||||
|
try:
|
||||||
|
Article._base_manager.__class__ = FakeManager
|
||||||
|
asos = ArticleSelectOnSave.objects.create(pub_date=datetime.now())
|
||||||
|
with self.assertNumQueries(2):
|
||||||
|
asos.save()
|
||||||
|
self.assertTrue(FakeQuerySet.called)
|
||||||
|
# This is not wanted behaviour, but this is how Django has always
|
||||||
|
# behaved for databases that do not return correct information
|
||||||
|
# about matched rows for UPDATE.
|
||||||
|
with self.assertRaises(DatabaseError):
|
||||||
|
asos.save(force_update=True)
|
||||||
|
with self.assertRaises(DatabaseError):
|
||||||
|
asos.save(update_fields=['pub_date'])
|
||||||
|
finally:
|
||||||
|
Article._base_manager.__class__ = orig_class
|
||||||
|
|
Loading…
Reference in New Issue