Kaydet (Commit) 2f3e1fe3 authored tarafından Aymeric Augustin's avatar Aymeric Augustin

Merge pull request #2489 from loic/related_transactions

Fixed transaction handling for a number of operations on related objects.
...@@ -394,9 +394,12 @@ class ReverseGenericRelatedObjectsDescriptor(object): ...@@ -394,9 +394,12 @@ class ReverseGenericRelatedObjectsDescriptor(object):
def __set__(self, instance, value): def __set__(self, instance, value):
manager = self.__get__(instance) manager = self.__get__(instance)
manager.clear()
for obj in value: db = router.db_for_write(manager.model, instance=manager.instance)
manager.add(obj) with transaction.atomic(using=db, savepoint=False):
manager.clear()
for obj in value:
manager.add(obj)
def create_generic_related_manager(superclass): def create_generic_related_manager(superclass):
...@@ -474,12 +477,14 @@ def create_generic_related_manager(superclass): ...@@ -474,12 +477,14 @@ def create_generic_related_manager(superclass):
self.prefetch_cache_name) self.prefetch_cache_name)
def add(self, *objs): def add(self, *objs):
for obj in objs: db = router.db_for_write(self.model, instance=self.instance)
if not isinstance(obj, self.model): with transaction.atomic(using=db, savepoint=False):
raise TypeError("'%s' instance expected" % self.model._meta.object_name) for obj in objs:
setattr(obj, self.content_type_field_name, self.content_type) if not isinstance(obj, self.model):
setattr(obj, self.object_id_field_name, self.pk_val) raise TypeError("'%s' instance expected" % self.model._meta.object_name)
obj.save() setattr(obj, self.content_type_field_name, self.content_type)
setattr(obj, self.object_id_field_name, self.pk_val)
obj.save()
add.alters_data = True add.alters_data = True
def remove(self, *objs, **kwargs): def remove(self, *objs, **kwargs):
...@@ -498,6 +503,8 @@ def create_generic_related_manager(superclass): ...@@ -498,6 +503,8 @@ def create_generic_related_manager(superclass):
db = router.db_for_write(self.model, instance=self.instance) db = router.db_for_write(self.model, instance=self.instance)
queryset = queryset.using(db) queryset = queryset.using(db)
if bulk: if bulk:
# `QuerySet.delete()` creates its own atomic block which
# contains the `pre_delete` and `post_delete` signal handlers.
queryset.delete() queryset.delete()
else: else:
with transaction.atomic(using=db, savepoint=False): with transaction.atomic(using=db, savepoint=False):
......
...@@ -168,7 +168,19 @@ Backwards incompatible changes in 1.8 ...@@ -168,7 +168,19 @@ Backwards incompatible changes in 1.8
deprecation timeline for a given feature, its removal may appear as a deprecation timeline for a given feature, its removal may appear as a
backwards incompatible change. backwards incompatible change.
... * Some operations on related objects such as
:meth:`~django.db.models.fields.related.RelatedManager.add()` or
:ref:`direct assignment<direct-assignment>` ran multiple data modifying
queries without wrapping them in transactions. To reduce the risk of data
corruption, all data modifying methods that affect multiple related objects
(i.e. ``add()``, ``remove()``, ``clear()``, and
:ref:`direct assignment<direct-assignment>`) now perform their data modifying
queries from within a transaction, provided your database supports
transactions.
This has one backwards incompatible side effect, signal handlers triggered
from these methods are now executed within the method's transaction and
any exception in a signal handler will prevent the whole operation.
Miscellaneous Miscellaneous
~~~~~~~~~~~~~ ~~~~~~~~~~~~~
......
from __future__ import unicode_literals from __future__ import unicode_literals
from django.db import transaction
from django.test import TestCase from django.test import TestCase
from django.utils import six from django.utils import six
...@@ -54,7 +55,9 @@ class ManyToManyTests(TestCase): ...@@ -54,7 +55,9 @@ class ManyToManyTests(TestCase):
# Adding an object of the wrong type raises TypeError # Adding an object of the wrong type raises TypeError
with six.assertRaisesRegex(self, TypeError, "'Publication' instance expected, got <Article.*"): with six.assertRaisesRegex(self, TypeError, "'Publication' instance expected, got <Article.*"):
a6.publications.add(a5) with transaction.atomic():
a6.publications.add(a5)
# Add a Publication directly via publications.add by using keyword arguments. # Add a Publication directly via publications.add by using keyword arguments.
a6.publications.create(title='Highlights for Adults') a6.publications.create(title='Highlights for Adults')
self.assertQuerysetEqual(a6.publications.all(), self.assertQuerysetEqual(a6.publications.all(),
......
...@@ -289,39 +289,29 @@ class QueryTestCase(TestCase): ...@@ -289,39 +289,29 @@ class QueryTestCase(TestCase):
mark = Person.objects.using('other').create(name="Mark Pilgrim") mark = Person.objects.using('other').create(name="Mark Pilgrim")
# Set a foreign key set with an object from a different database # Set a foreign key set with an object from a different database
try: with self.assertRaises(ValueError):
marty.book_set = [pro, dive] with transaction.atomic(using='default'):
self.fail("Shouldn't be able to assign across databases") marty.edited = [pro, dive]
except ValueError:
pass
# Add to an m2m with an object from a different database # Add to an m2m with an object from a different database
try: with self.assertRaises(ValueError):
marty.book_set.add(dive) with transaction.atomic(using='default'):
self.fail("Shouldn't be able to assign across databases") marty.book_set.add(dive)
except ValueError:
pass
# Set a m2m with an object from a different database # Set a m2m with an object from a different database
try: with self.assertRaises(ValueError):
marty.book_set = [pro, dive] with transaction.atomic(using='default'):
self.fail("Shouldn't be able to assign across databases") marty.book_set = [pro, dive]
except ValueError:
pass
# Add to a reverse m2m with an object from a different database # Add to a reverse m2m with an object from a different database
try: with self.assertRaises(ValueError):
dive.authors.add(marty) with transaction.atomic(using='other'):
self.fail("Shouldn't be able to assign across databases") dive.authors.add(marty)
except ValueError:
pass
# Set a reverse m2m with an object from a different database # Set a reverse m2m with an object from a different database
try: with self.assertRaises(ValueError):
dive.authors = [mark, marty] with transaction.atomic(using='other'):
self.fail("Shouldn't be able to assign across databases") dive.authors = [mark, marty]
except ValueError:
pass
def test_m2m_deletion(self): def test_m2m_deletion(self):
"Cascaded deletions of m2m relations issue queries on the right database" "Cascaded deletions of m2m relations issue queries on the right database"
...@@ -489,28 +479,18 @@ class QueryTestCase(TestCase): ...@@ -489,28 +479,18 @@ class QueryTestCase(TestCase):
mark = Person.objects.using('other').create(name="Mark Pilgrim") mark = Person.objects.using('other').create(name="Mark Pilgrim")
# Set a foreign key with an object from a different database # Set a foreign key with an object from a different database
try: with self.assertRaises(ValueError):
with transaction.atomic(using='default'): dive.editor = marty
dive.editor = marty
self.fail("Shouldn't be able to assign across databases")
except ValueError:
pass
# Set a foreign key set with an object from a different database # Set a foreign key set with an object from a different database
try: with self.assertRaises(ValueError):
with transaction.atomic(using='default'): with transaction.atomic(using='default'):
marty.edited = [pro, dive] marty.edited = [pro, dive]
self.fail("Shouldn't be able to assign across databases")
except ValueError:
pass
# Add to a foreign key set with an object from a different database # Add to a foreign key set with an object from a different database
try: with self.assertRaises(ValueError):
with transaction.atomic(using='default'): with transaction.atomic(using='default'):
marty.edited.add(dive) marty.edited.add(dive)
self.fail("Shouldn't be able to assign across databases")
except ValueError:
pass
# BUT! if you assign a FK object when the base object hasn't # BUT! if you assign a FK object when the base object hasn't
# been saved yet, you implicitly assign the database for the # been saved yet, you implicitly assign the database for the
...@@ -634,11 +614,8 @@ class QueryTestCase(TestCase): ...@@ -634,11 +614,8 @@ class QueryTestCase(TestCase):
# Set a one-to-one relation with an object from a different database # Set a one-to-one relation with an object from a different database
alice_profile = UserProfile.objects.using('default').create(user=alice, flavor='chocolate') alice_profile = UserProfile.objects.using('default').create(user=alice, flavor='chocolate')
try: with self.assertRaises(ValueError):
bob.userprofile = alice_profile bob.userprofile = alice_profile
self.fail("Shouldn't be able to assign across databases")
except ValueError:
pass
# BUT! if you assign a FK object when the base object hasn't # BUT! if you assign a FK object when the base object hasn't
# been saved yet, you implicitly assign the database for the # been saved yet, you implicitly assign the database for the
...@@ -784,18 +761,13 @@ class QueryTestCase(TestCase): ...@@ -784,18 +761,13 @@ class QueryTestCase(TestCase):
Review.objects.using('other').create(source="Python Weekly", content_object=dive) Review.objects.using('other').create(source="Python Weekly", content_object=dive)
# Set a foreign key with an object from a different database # Set a foreign key with an object from a different database
try: with self.assertRaises(ValueError):
review1.content_object = dive review1.content_object = dive
self.fail("Shouldn't be able to assign across databases")
except ValueError:
pass
# Add to a foreign key set with an object from a different database # Add to a foreign key set with an object from a different database
try: with self.assertRaises(ValueError):
dive.reviews.add(review1) with transaction.atomic(using='other'):
self.fail("Shouldn't be able to assign across databases") dive.reviews.add(review1)
except ValueError:
pass
# BUT! if you assign a FK object when the base object hasn't # BUT! if you assign a FK object when the base object hasn't
# been saved yet, you implicitly assign the database for the # been saved yet, you implicitly assign the database for the
...@@ -892,12 +864,9 @@ class QueryTestCase(TestCase): ...@@ -892,12 +864,9 @@ class QueryTestCase(TestCase):
self.assertRaises(ValueError, str, qs.query) self.assertRaises(ValueError, str, qs.query)
# Evaluating the query shouldn't work, either # Evaluating the query shouldn't work, either
try: with self.assertRaises(ValueError):
for obj in qs: for obj in qs:
pass pass
self.fail('Iterating over query should raise ValueError')
except ValueError:
pass
def test_related_manager(self): def test_related_manager(self):
"Related managers return managers, not querysets" "Related managers return managers, not querysets"
...@@ -1041,12 +1010,9 @@ class RouterTestCase(TestCase): ...@@ -1041,12 +1010,9 @@ class RouterTestCase(TestCase):
# An update query will be routed to the default database # An update query will be routed to the default database
Book.objects.filter(title='Pro Django').update(pages=200) Book.objects.filter(title='Pro Django').update(pages=200)
try: with self.assertRaises(Book.DoesNotExist):
# By default, the get query will be directed to 'other' # By default, the get query will be directed to 'other'
Book.objects.get(title='Pro Django') Book.objects.get(title='Pro Django')
self.fail("Shouldn't be able to find the book")
except Book.DoesNotExist:
pass
# But the same query issued explicitly at a database will work. # But the same query issued explicitly at a database will work.
pro = Book.objects.using('default').get(title='Pro Django') pro = Book.objects.using('default').get(title='Pro Django')
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment