2017-02-12 04:37:49 +08:00
|
|
|
"""Tests for django.db.utils."""
|
|
|
|
import unittest
|
|
|
|
|
|
|
|
from django.core.exceptions import ImproperlyConfigured
|
2019-08-20 15:54:41 +08:00
|
|
|
from django.db import DEFAULT_DB_ALIAS, ProgrammingError, connection
|
2017-10-22 23:30:42 +08:00
|
|
|
from django.db.utils import ConnectionHandler, load_backend
|
2017-02-12 04:37:49 +08:00
|
|
|
from django.test import SimpleTestCase, TestCase
|
2017-10-22 23:30:42 +08:00
|
|
|
from django.utils.connection import ConnectionDoesNotExist
|
2017-02-12 04:37:49 +08:00
|
|
|
|
|
|
|
|
|
|
|
class ConnectionHandlerTests(SimpleTestCase):
|
|
|
|
|
|
|
|
def test_connection_handler_no_databases(self):
|
2019-01-09 23:58:25 +08:00
|
|
|
"""
|
|
|
|
Empty DATABASES and empty 'default' settings default to the dummy
|
|
|
|
backend.
|
|
|
|
"""
|
|
|
|
for DATABASES in (
|
|
|
|
{}, # Empty DATABASES setting.
|
|
|
|
{'default': {}}, # Empty 'default' database.
|
|
|
|
):
|
|
|
|
with self.subTest(DATABASES=DATABASES):
|
|
|
|
self.assertImproperlyConfigured(DATABASES)
|
|
|
|
|
|
|
|
def assertImproperlyConfigured(self, DATABASES):
|
2017-02-12 04:37:49 +08:00
|
|
|
conns = ConnectionHandler(DATABASES)
|
|
|
|
self.assertEqual(conns[DEFAULT_DB_ALIAS].settings_dict['ENGINE'], 'django.db.backends.dummy')
|
|
|
|
msg = (
|
|
|
|
'settings.DATABASES is improperly configured. Please supply the '
|
|
|
|
'ENGINE value. Check settings documentation for more details.'
|
|
|
|
)
|
|
|
|
with self.assertRaisesMessage(ImproperlyConfigured, msg):
|
|
|
|
conns[DEFAULT_DB_ALIAS].ensure_connection()
|
|
|
|
|
2018-12-17 02:09:08 +08:00
|
|
|
def test_no_default_database(self):
|
|
|
|
DATABASES = {'other': {}}
|
|
|
|
conns = ConnectionHandler(DATABASES)
|
|
|
|
msg = "You must define a 'default' database."
|
|
|
|
with self.assertRaisesMessage(ImproperlyConfigured, msg):
|
|
|
|
conns['other'].ensure_connection()
|
|
|
|
|
2020-12-08 14:05:49 +08:00
|
|
|
def test_nonexistent_alias(self):
|
2017-10-22 23:30:42 +08:00
|
|
|
msg = "The connection 'nonexistent' doesn't exist."
|
2020-12-08 14:05:49 +08:00
|
|
|
conns = ConnectionHandler({
|
|
|
|
DEFAULT_DB_ALIAS: {'ENGINE': 'django.db.backends.dummy'},
|
|
|
|
})
|
|
|
|
with self.assertRaisesMessage(ConnectionDoesNotExist, msg):
|
|
|
|
conns['nonexistent']
|
|
|
|
|
|
|
|
def test_ensure_defaults_nonexistent_alias(self):
|
2017-10-22 23:30:42 +08:00
|
|
|
msg = "The connection 'nonexistent' doesn't exist."
|
2020-12-08 14:05:49 +08:00
|
|
|
conns = ConnectionHandler({
|
|
|
|
DEFAULT_DB_ALIAS: {'ENGINE': 'django.db.backends.dummy'},
|
|
|
|
})
|
|
|
|
with self.assertRaisesMessage(ConnectionDoesNotExist, msg):
|
|
|
|
conns.ensure_defaults('nonexistent')
|
|
|
|
|
|
|
|
def test_prepare_test_settings_nonexistent_alias(self):
|
2017-10-22 23:30:42 +08:00
|
|
|
msg = "The connection 'nonexistent' doesn't exist."
|
2020-12-08 14:05:49 +08:00
|
|
|
conns = ConnectionHandler({
|
|
|
|
DEFAULT_DB_ALIAS: {'ENGINE': 'django.db.backends.dummy'},
|
|
|
|
})
|
|
|
|
with self.assertRaisesMessage(ConnectionDoesNotExist, msg):
|
|
|
|
conns.prepare_test_settings('nonexistent')
|
|
|
|
|
2017-02-12 04:37:49 +08:00
|
|
|
|
|
|
|
class DatabaseErrorWrapperTests(TestCase):
|
|
|
|
|
|
|
|
@unittest.skipUnless(connection.vendor == 'postgresql', 'PostgreSQL test')
|
|
|
|
def test_reraising_backend_specific_database_exception(self):
|
2017-11-28 21:12:28 +08:00
|
|
|
with connection.cursor() as cursor:
|
|
|
|
msg = 'table "X" does not exist'
|
|
|
|
with self.assertRaisesMessage(ProgrammingError, msg) as cm:
|
|
|
|
cursor.execute('DROP TABLE "X"')
|
2017-02-12 04:37:49 +08:00
|
|
|
self.assertNotEqual(type(cm.exception), type(cm.exception.__cause__))
|
|
|
|
self.assertIsNotNone(cm.exception.__cause__)
|
|
|
|
self.assertIsNotNone(cm.exception.__cause__.pgcode)
|
|
|
|
self.assertIsNotNone(cm.exception.__cause__.pgerror)
|
|
|
|
|
|
|
|
|
|
|
|
class LoadBackendTests(SimpleTestCase):
|
|
|
|
|
|
|
|
def test_load_backend_invalid_name(self):
|
|
|
|
msg = (
|
2020-11-16 04:34:35 +08:00
|
|
|
"'foo' isn't an available database backend or couldn't be "
|
|
|
|
"imported. Check the above exception. To use one of the built-in "
|
|
|
|
"backends, use 'django.db.backends.XXX', where XXX is one of:\n"
|
2017-02-12 04:37:49 +08:00
|
|
|
" 'mysql', 'oracle', 'postgresql', 'sqlite3'"
|
|
|
|
)
|
|
|
|
with self.assertRaisesMessage(ImproperlyConfigured, msg) as cm:
|
|
|
|
load_backend('foo')
|
|
|
|
self.assertEqual(str(cm.exception.__cause__), "No module named 'foo'")
|