2017-02-11 20:37:49 +00:00
|
|
|
import unittest
|
|
|
|
from io import StringIO
|
|
|
|
from unittest import mock
|
|
|
|
|
2019-08-20 07:54:41 +00:00
|
|
|
from django.db import DatabaseError, connection
|
2017-02-11 20:37:49 +00:00
|
|
|
from django.db.backends.oracle.creation import DatabaseCreation
|
|
|
|
from django.test import TestCase
|
|
|
|
|
|
|
|
|
2022-02-03 19:24:19 +00:00
|
|
|
@unittest.skipUnless(connection.vendor == "oracle", "Oracle tests")
|
|
|
|
@mock.patch.object(DatabaseCreation, "_maindb_connection", return_value=connection)
|
|
|
|
@mock.patch("sys.stdout", new_callable=StringIO)
|
|
|
|
@mock.patch("sys.stderr", new_callable=StringIO)
|
2017-02-11 20:37:49 +00:00
|
|
|
class DatabaseCreationTests(TestCase):
|
2022-02-03 19:24:19 +00:00
|
|
|
def _execute_raise_user_already_exists(
|
|
|
|
self, cursor, statements, parameters, verbosity, allow_quiet_fail=False
|
|
|
|
):
|
2017-02-11 20:37:49 +00:00
|
|
|
# Raise "user already exists" only in test user creation
|
2022-02-03 19:24:19 +00:00
|
|
|
if statements and statements[0].startswith("CREATE USER"):
|
|
|
|
raise DatabaseError(
|
|
|
|
"ORA-01920: user name 'string' conflicts with another user or role name"
|
|
|
|
)
|
2017-02-11 20:37:49 +00:00
|
|
|
|
|
|
|
def _execute_raise_tablespace_already_exists(
|
|
|
|
self, cursor, statements, parameters, verbosity, allow_quiet_fail=False
|
|
|
|
):
|
|
|
|
raise DatabaseError("ORA-01543: tablespace 'string' already exists")
|
|
|
|
|
|
|
|
def _execute_raise_insufficient_privileges(
|
|
|
|
self, cursor, statements, parameters, verbosity, allow_quiet_fail=False
|
|
|
|
):
|
|
|
|
raise DatabaseError("ORA-01031: insufficient privileges")
|
|
|
|
|
|
|
|
def _test_database_passwd(self):
|
|
|
|
# Mocked to avoid test user password changed
|
2022-02-03 19:24:19 +00:00
|
|
|
return connection.settings_dict["SAVED_PASSWORD"]
|
2017-02-11 20:37:49 +00:00
|
|
|
|
|
|
|
def patch_execute_statements(self, execute_statements):
|
2022-02-03 19:24:19 +00:00
|
|
|
return mock.patch.object(
|
|
|
|
DatabaseCreation, "_execute_statements", execute_statements
|
|
|
|
)
|
2017-02-11 20:37:49 +00:00
|
|
|
|
2022-02-03 19:24:19 +00:00
|
|
|
@mock.patch.object(DatabaseCreation, "_test_user_create", return_value=False)
|
2017-02-11 20:37:49 +00:00
|
|
|
def test_create_test_db(self, *mocked_objects):
|
|
|
|
creation = DatabaseCreation(connection)
|
|
|
|
# Simulate test database creation raising "tablespace already exists"
|
2022-02-03 19:24:19 +00:00
|
|
|
with self.patch_execute_statements(
|
|
|
|
self._execute_raise_tablespace_already_exists
|
|
|
|
):
|
|
|
|
with mock.patch("builtins.input", return_value="no"):
|
2017-02-11 20:37:49 +00:00
|
|
|
with self.assertRaises(SystemExit):
|
|
|
|
# SystemExit is raised if the user answers "no" to the
|
|
|
|
# prompt asking if it's okay to delete the test tablespace.
|
|
|
|
creation._create_test_db(verbosity=0, keepdb=False)
|
|
|
|
# "Tablespace already exists" error is ignored when keepdb is on
|
|
|
|
creation._create_test_db(verbosity=0, keepdb=True)
|
|
|
|
# Simulate test database creation raising unexpected error
|
|
|
|
with self.patch_execute_statements(self._execute_raise_insufficient_privileges):
|
|
|
|
with self.assertRaises(SystemExit):
|
|
|
|
creation._create_test_db(verbosity=0, keepdb=False)
|
|
|
|
with self.assertRaises(SystemExit):
|
|
|
|
creation._create_test_db(verbosity=0, keepdb=True)
|
|
|
|
|
2022-02-03 19:24:19 +00:00
|
|
|
@mock.patch.object(DatabaseCreation, "_test_database_create", return_value=False)
|
2017-02-11 20:37:49 +00:00
|
|
|
def test_create_test_user(self, *mocked_objects):
|
|
|
|
creation = DatabaseCreation(connection)
|
2022-02-03 19:24:19 +00:00
|
|
|
with mock.patch.object(
|
|
|
|
DatabaseCreation, "_test_database_passwd", self._test_database_passwd
|
|
|
|
):
|
2017-02-11 20:37:49 +00:00
|
|
|
# Simulate test user creation raising "user already exists"
|
|
|
|
with self.patch_execute_statements(self._execute_raise_user_already_exists):
|
2022-02-03 19:24:19 +00:00
|
|
|
with mock.patch("builtins.input", return_value="no"):
|
2017-02-11 20:37:49 +00:00
|
|
|
with self.assertRaises(SystemExit):
|
|
|
|
# SystemExit is raised if the user answers "no" to the
|
|
|
|
# prompt asking if it's okay to delete the test user.
|
|
|
|
creation._create_test_db(verbosity=0, keepdb=False)
|
|
|
|
# "User already exists" error is ignored when keepdb is on
|
|
|
|
creation._create_test_db(verbosity=0, keepdb=True)
|
|
|
|
# Simulate test user creation raising unexpected error
|
2022-02-03 19:24:19 +00:00
|
|
|
with self.patch_execute_statements(
|
|
|
|
self._execute_raise_insufficient_privileges
|
|
|
|
):
|
2017-02-11 20:37:49 +00:00
|
|
|
with self.assertRaises(SystemExit):
|
|
|
|
creation._create_test_db(verbosity=0, keepdb=False)
|
|
|
|
with self.assertRaises(SystemExit):
|
|
|
|
creation._create_test_db(verbosity=0, keepdb=True)
|
2018-11-06 21:13:57 +00:00
|
|
|
|
|
|
|
def test_oracle_managed_files(self, *mocked_objects):
|
2022-02-03 19:24:19 +00:00
|
|
|
def _execute_capture_statements(
|
|
|
|
self, cursor, statements, parameters, verbosity, allow_quiet_fail=False
|
|
|
|
):
|
2018-11-06 21:13:57 +00:00
|
|
|
self.tblspace_sqls = statements
|
|
|
|
|
|
|
|
creation = DatabaseCreation(connection)
|
|
|
|
# Simulate test database creation with Oracle Managed File (OMF)
|
|
|
|
# tablespaces.
|
2022-02-03 19:24:19 +00:00
|
|
|
with mock.patch.object(
|
|
|
|
DatabaseCreation, "_test_database_oracle_managed_files", return_value=True
|
|
|
|
):
|
2018-11-06 21:13:57 +00:00
|
|
|
with self.patch_execute_statements(_execute_capture_statements):
|
|
|
|
with connection.cursor() as cursor:
|
2022-02-03 19:24:19 +00:00
|
|
|
creation._execute_test_db_creation(
|
|
|
|
cursor, creation._get_test_db_params(), verbosity=0
|
|
|
|
)
|
2018-11-06 21:13:57 +00:00
|
|
|
tblspace_sql, tblspace_tmp_sql = creation.tblspace_sqls
|
|
|
|
# Datafile names shouldn't appear.
|
2022-02-03 19:24:19 +00:00
|
|
|
self.assertIn("DATAFILE SIZE", tblspace_sql)
|
|
|
|
self.assertIn("TEMPFILE SIZE", tblspace_tmp_sql)
|
2018-11-06 21:13:57 +00:00
|
|
|
# REUSE cannot be used with OMF.
|
2022-02-03 19:24:19 +00:00
|
|
|
self.assertNotIn("REUSE", tblspace_sql)
|
|
|
|
self.assertNotIn("REUSE", tblspace_tmp_sql)
|