Fixed #19343 -- Deadlock with TransactionTestCase + TEST_MIRROR + multi_db.

Thanks Jeremy Dunck for the review.
This commit is contained in:
Aymeric Augustin 2012-11-24 09:48:21 +01:00
parent f2d8004967
commit be64dd35fb
1 changed files with 14 additions and 23 deletions

View File

@ -416,6 +416,15 @@ class TransactionTestCase(SimpleTestCase):
self._urlconf_setup() self._urlconf_setup()
mail.outbox = [] mail.outbox = []
def _databases_names(self, include_mirrors=True):
# If the test case has a multi_db=True flag, act on all databases,
# including mirrors or not. Otherwise, just on the default DB.
if getattr(self, 'multi_db', False):
return [alias for alias in connections
if include_mirrors or not connections[alias].settings_dict['TEST_MIRROR']]
else:
return [DEFAULT_DB_ALIAS]
def _reset_sequences(self, db_name): def _reset_sequences(self, db_name):
conn = connections[db_name] conn = connections[db_name]
if conn.features.supports_sequence_reset: if conn.features.supports_sequence_reset:
@ -433,10 +442,7 @@ class TransactionTestCase(SimpleTestCase):
transaction.commit_unless_managed(using=db_name) transaction.commit_unless_managed(using=db_name)
def _fixture_setup(self): def _fixture_setup(self):
# If the test case has a multi_db=True flag, act on all databases. for db_name in self._databases_names(include_mirrors=False):
# Otherwise, just on the default DB.
db_names = connections if getattr(self, 'multi_db', False) else [DEFAULT_DB_ALIAS]
for db_name in db_names:
# Reset sequences # Reset sequences
if self.reset_sequences: if self.reset_sequences:
self._reset_sequences(db_name) self._reset_sequences(db_name)
@ -502,16 +508,12 @@ class TransactionTestCase(SimpleTestCase):
conn.close() conn.close()
def _fixture_teardown(self): def _fixture_teardown(self):
# If the test case has a multi_db=True flag, flush all databases.
# Otherwise, just flush default.
databases = connections if getattr(self, 'multi_db', False) else [DEFAULT_DB_ALIAS]
# Roll back any pending transactions in order to avoid a deadlock # Roll back any pending transactions in order to avoid a deadlock
# during flush when TEST_MIRROR is used (#18984). # during flush when TEST_MIRROR is used (#18984).
for conn in connections.all(): for conn in connections.all():
conn.rollback_unless_managed() conn.rollback_unless_managed()
for db in databases: for db in self._databases_names(include_mirrors=False):
call_command('flush', verbosity=0, interactive=False, database=db, call_command('flush', verbosity=0, interactive=False, database=db,
skip_validation=True, reset_sequences=False) skip_validation=True, reset_sequences=False)
@ -796,11 +798,7 @@ class TestCase(TransactionTestCase):
assert not self.reset_sequences, 'reset_sequences cannot be used on TestCase instances' assert not self.reset_sequences, 'reset_sequences cannot be used on TestCase instances'
# If the test case has a multi_db=True flag, setup all databases. for db_name in self._databases_names():
# Otherwise, just use default.
db_names = connections if getattr(self, 'multi_db', False) else [DEFAULT_DB_ALIAS]
for db_name in db_names:
transaction.enter_transaction_management(using=db_name) transaction.enter_transaction_management(using=db_name)
transaction.managed(True, using=db_name) transaction.managed(True, using=db_name)
disable_transaction_methods() disable_transaction_methods()
@ -808,7 +806,7 @@ class TestCase(TransactionTestCase):
from django.contrib.sites.models import Site from django.contrib.sites.models import Site
Site.objects.clear_cache() Site.objects.clear_cache()
for db in db_names: for db in self._databases_names(include_mirrors=False):
if hasattr(self, 'fixtures'): if hasattr(self, 'fixtures'):
call_command('loaddata', *self.fixtures, call_command('loaddata', *self.fixtures,
**{ **{
@ -822,15 +820,8 @@ class TestCase(TransactionTestCase):
if not connections_support_transactions(): if not connections_support_transactions():
return super(TestCase, self)._fixture_teardown() return super(TestCase, self)._fixture_teardown()
# If the test case has a multi_db=True flag, teardown all databases.
# Otherwise, just teardown default.
if getattr(self, 'multi_db', False):
databases = connections
else:
databases = [DEFAULT_DB_ALIAS]
restore_transaction_methods() restore_transaction_methods()
for db in databases: for db in self._databases_names():
transaction.rollback(using=db) transaction.rollback(using=db)
transaction.leave_transaction_management(using=db) transaction.leave_transaction_management(using=db)