summaryrefslogtreecommitdiff
path: root/tests/get_or_create
diff options
context:
space:
mode:
authorMichael Sanders <michael.sanders@arm.com>2018-08-01 10:52:28 +0100
committerTim Graham <timograham@gmail.com>2018-08-02 17:07:48 -0400
commit271542dad1686c438f658aa6220982495db09797 (patch)
treeccdcdfbe3bb2dd8c27d7f237cbaf877b21074088 /tests/get_or_create
parent743d28f5539b17d6e39bd37c6e3df5628b470cac (diff)
downloaddjango-271542dad1686c438f658aa6220982495db09797.tar.gz
Fixed #29499 -- Fixed race condition in QuerySet.update_or_create().
A race condition happened when the object didn't already exist and another process/thread created the object before update_or_create() did and then attempted to update the object, also before update_or_create() saved the object. The update by the other process/thread could be lost.
Diffstat (limited to 'tests/get_or_create')
-rw-r--r--tests/get_or_create/models.py2
-rw-r--r--tests/get_or_create/tests.py58
2 files changed, 59 insertions, 1 deletions
diff --git a/tests/get_or_create/models.py b/tests/get_or_create/models.py
index 4a33a809bb..6510bb9464 100644
--- a/tests/get_or_create/models.py
+++ b/tests/get_or_create/models.py
@@ -2,7 +2,7 @@ from django.db import models
class Person(models.Model):
- first_name = models.CharField(max_length=100)
+ first_name = models.CharField(max_length=100, unique=True)
last_name = models.CharField(max_length=100)
birthday = models.DateField()
defaults = models.TextField()
diff --git a/tests/get_or_create/tests.py b/tests/get_or_create/tests.py
index e4647d2ab5..194d4159b6 100644
--- a/tests/get_or_create/tests.py
+++ b/tests/get_or_create/tests.py
@@ -535,6 +535,64 @@ class UpdateOrCreateTransactionTests(TransactionTestCase):
self.assertGreater(after_update - before_start, timedelta(seconds=0.5))
self.assertEqual(updated_person.last_name, 'NotLennon')
+ @skipUnlessDBFeature('has_select_for_update')
+ @skipUnlessDBFeature('supports_transactions')
+ def test_creation_in_transaction(self):
+ """
+ Objects are selected and updated in a transaction to avoid race
+ conditions. This test checks the behavior of update_or_create() when
+ the object doesn't already exist, but another thread creates the
+ object before update_or_create() does and then attempts to update the
+ object, also before update_or_create(). It forces update_or_create() to
+ hold the lock in another thread for a relatively long time so that it
+ can update while it holds the lock. The updated field isn't a field in
+ 'defaults', so update_or_create() shouldn't have an effect on it.
+ """
+ lock_status = {'lock_count': 0}
+
+ def birthday_sleep():
+ lock_status['lock_count'] += 1
+ time.sleep(0.5)
+ return date(1940, 10, 10)
+
+ def update_birthday_slowly():
+ try:
+ Person.objects.update_or_create(first_name='John', defaults={'birthday': birthday_sleep})
+ finally:
+ # Avoid leaking connection for Oracle
+ connection.close()
+
+ def lock_wait(expected_lock_count):
+ # timeout after ~0.5 seconds
+ for i in range(20):
+ time.sleep(0.025)
+ if lock_status['lock_count'] == expected_lock_count:
+ return True
+ self.skipTest('Database took too long to lock the row')
+
+ # update_or_create in a separate thread.
+ t = Thread(target=update_birthday_slowly)
+ before_start = datetime.now()
+ t.start()
+ lock_wait(1)
+ # Create object *after* initial attempt by update_or_create to get obj
+ # but before creation attempt.
+ Person.objects.create(first_name='John', last_name='Lennon', birthday=date(1940, 10, 9))
+ lock_wait(2)
+ # At this point, the thread is pausing for 0.5 seconds, so now attempt
+ # to modify object before update_or_create() calls save(). This should
+ # be blocked until after the save().
+ Person.objects.filter(first_name='John').update(last_name='NotLennon')
+ after_update = datetime.now()
+ # Wait for thread to finish
+ t.join()
+ # Check call to update_or_create() succeeded and the subsequent
+ # (blocked) call to update().
+ updated_person = Person.objects.get(first_name='John')
+ self.assertEqual(updated_person.birthday, date(1940, 10, 10)) # set by update_or_create()
+ self.assertEqual(updated_person.last_name, 'NotLennon') # set by update()
+ self.assertGreater(after_update - before_start, timedelta(seconds=1))
+
class InvalidCreateArgumentsTests(TransactionTestCase):
available_apps = ['get_or_create']