mirror of
https://github.com/django/django.git
synced 2025-01-10 10:26:34 +00:00
325 lines
12 KiB
Python
325 lines
12 KiB
Python
import os
|
|
import re
|
|
import tempfile
|
|
import threading
|
|
import unittest
|
|
from contextlib import contextmanager
|
|
from pathlib import Path
|
|
from unittest import mock
|
|
|
|
from django.core.exceptions import ImproperlyConfigured
|
|
from django.db import (
|
|
DEFAULT_DB_ALIAS,
|
|
NotSupportedError,
|
|
connection,
|
|
connections,
|
|
transaction,
|
|
)
|
|
from django.db.models import Aggregate, Avg, StdDev, Sum, Variance
|
|
from django.db.utils import ConnectionHandler
|
|
from django.test import SimpleTestCase, TestCase, TransactionTestCase, override_settings
|
|
from django.test.utils import CaptureQueriesContext, isolate_apps
|
|
|
|
from ..models import Item, Object, Square
|
|
|
|
|
|
@unittest.skipUnless(connection.vendor == "sqlite", "SQLite tests")
|
|
class Tests(TestCase):
|
|
longMessage = True
|
|
|
|
def test_aggregation(self):
|
|
"""Raise NotSupportedError when aggregating on date/time fields."""
|
|
for aggregate in (Sum, Avg, Variance, StdDev):
|
|
with self.assertRaises(NotSupportedError):
|
|
Item.objects.aggregate(aggregate("time"))
|
|
with self.assertRaises(NotSupportedError):
|
|
Item.objects.aggregate(aggregate("date"))
|
|
with self.assertRaises(NotSupportedError):
|
|
Item.objects.aggregate(aggregate("last_modified"))
|
|
with self.assertRaises(NotSupportedError):
|
|
Item.objects.aggregate(
|
|
**{
|
|
"complex": aggregate("last_modified")
|
|
+ aggregate("last_modified")
|
|
}
|
|
)
|
|
|
|
def test_distinct_aggregation(self):
|
|
class DistinctAggregate(Aggregate):
|
|
allow_distinct = True
|
|
|
|
aggregate = DistinctAggregate("first", "second", distinct=True)
|
|
msg = (
|
|
"SQLite doesn't support DISTINCT on aggregate functions accepting "
|
|
"multiple arguments."
|
|
)
|
|
with self.assertRaisesMessage(NotSupportedError, msg):
|
|
connection.ops.check_expression_support(aggregate)
|
|
|
|
def test_distinct_aggregation_multiple_args_no_distinct(self):
|
|
# Aggregate functions accept multiple arguments when DISTINCT isn't
|
|
# used, e.g. GROUP_CONCAT().
|
|
class DistinctAggregate(Aggregate):
|
|
allow_distinct = True
|
|
|
|
aggregate = DistinctAggregate("first", "second", distinct=False)
|
|
connection.ops.check_expression_support(aggregate)
|
|
|
|
def test_memory_db_test_name(self):
|
|
"""A named in-memory db should be allowed where supported."""
|
|
from django.db.backends.sqlite3.base import DatabaseWrapper
|
|
|
|
settings_dict = {
|
|
"TEST": {
|
|
"NAME": "file:memorydb_test?mode=memory&cache=shared",
|
|
}
|
|
}
|
|
creation = DatabaseWrapper(settings_dict).creation
|
|
self.assertEqual(
|
|
creation._get_test_db_name(),
|
|
creation.connection.settings_dict["TEST"]["NAME"],
|
|
)
|
|
|
|
def test_regexp_function(self):
|
|
tests = (
|
|
("test", r"[0-9]+", False),
|
|
("test", r"[a-z]+", True),
|
|
("test", None, None),
|
|
(None, r"[a-z]+", None),
|
|
(None, None, None),
|
|
)
|
|
for string, pattern, expected in tests:
|
|
with self.subTest((string, pattern)):
|
|
with connection.cursor() as cursor:
|
|
cursor.execute("SELECT %s REGEXP %s", [string, pattern])
|
|
value = cursor.fetchone()[0]
|
|
value = bool(value) if value in {0, 1} else value
|
|
self.assertIs(value, expected)
|
|
|
|
def test_pathlib_name(self):
|
|
with tempfile.TemporaryDirectory() as tmp:
|
|
settings_dict = {
|
|
"default": {
|
|
"ENGINE": "django.db.backends.sqlite3",
|
|
"NAME": Path(tmp) / "test.db",
|
|
},
|
|
}
|
|
connections = ConnectionHandler(settings_dict)
|
|
connections["default"].ensure_connection()
|
|
connections["default"].close()
|
|
self.assertTrue(os.path.isfile(os.path.join(tmp, "test.db")))
|
|
|
|
@mock.patch.object(connection, "get_database_version", return_value=(3, 30))
|
|
def test_check_database_version_supported(self, mocked_get_database_version):
|
|
msg = "SQLite 3.31 or later is required (found 3.30)."
|
|
with self.assertRaisesMessage(NotSupportedError, msg):
|
|
connection.check_database_version_supported()
|
|
self.assertTrue(mocked_get_database_version.called)
|
|
|
|
def test_init_command(self):
|
|
settings_dict = {
|
|
"default": {
|
|
"ENGINE": "django.db.backends.sqlite3",
|
|
"NAME": ":memory:",
|
|
"OPTIONS": {
|
|
"init_command": "PRAGMA synchronous=3; PRAGMA cache_size=2000;",
|
|
},
|
|
}
|
|
}
|
|
connections = ConnectionHandler(settings_dict)
|
|
connections["default"].ensure_connection()
|
|
try:
|
|
with connections["default"].cursor() as cursor:
|
|
cursor.execute("PRAGMA synchronous")
|
|
value = cursor.fetchone()[0]
|
|
self.assertEqual(value, 3)
|
|
cursor.execute("PRAGMA cache_size")
|
|
value = cursor.fetchone()[0]
|
|
self.assertEqual(value, 2000)
|
|
finally:
|
|
connections["default"]._close()
|
|
|
|
|
|
@unittest.skipUnless(connection.vendor == "sqlite", "SQLite tests")
|
|
@isolate_apps("backends")
|
|
class SchemaTests(TransactionTestCase):
|
|
available_apps = ["backends"]
|
|
|
|
def test_autoincrement(self):
|
|
"""
|
|
auto_increment fields are created with the AUTOINCREMENT keyword
|
|
in order to be monotonically increasing (#10164).
|
|
"""
|
|
with connection.schema_editor(collect_sql=True) as editor:
|
|
editor.create_model(Square)
|
|
statements = editor.collected_sql
|
|
match = re.search('"id" ([^,]+),', statements[0])
|
|
self.assertIsNotNone(match)
|
|
self.assertEqual(
|
|
"integer NOT NULL PRIMARY KEY AUTOINCREMENT",
|
|
match[1],
|
|
"Wrong SQL used to create an auto-increment column on SQLite",
|
|
)
|
|
|
|
def test_disable_constraint_checking_failure_disallowed(self):
|
|
"""
|
|
SQLite schema editor is not usable within an outer transaction if
|
|
foreign key constraint checks are not disabled beforehand.
|
|
"""
|
|
msg = (
|
|
"SQLite schema editor cannot be used while foreign key "
|
|
"constraint checks are enabled. Make sure to disable them "
|
|
"before entering a transaction.atomic() context because "
|
|
"SQLite does not support disabling them in the middle of "
|
|
"a multi-statement transaction."
|
|
)
|
|
with self.assertRaisesMessage(NotSupportedError, msg):
|
|
with transaction.atomic(), connection.schema_editor(atomic=True):
|
|
pass
|
|
|
|
def test_constraint_checks_disabled_atomic_allowed(self):
|
|
"""
|
|
SQLite schema editor is usable within an outer transaction as long as
|
|
foreign key constraints checks are disabled beforehand.
|
|
"""
|
|
|
|
def constraint_checks_enabled():
|
|
with connection.cursor() as cursor:
|
|
return bool(cursor.execute("PRAGMA foreign_keys").fetchone()[0])
|
|
|
|
with connection.constraint_checks_disabled(), transaction.atomic():
|
|
with connection.schema_editor(atomic=True):
|
|
self.assertFalse(constraint_checks_enabled())
|
|
self.assertFalse(constraint_checks_enabled())
|
|
self.assertTrue(constraint_checks_enabled())
|
|
|
|
|
|
@unittest.skipUnless(connection.vendor == "sqlite", "Test only for SQLite")
|
|
@override_settings(DEBUG=True)
|
|
class LastExecutedQueryTest(TestCase):
|
|
def test_no_interpolation(self):
|
|
# This shouldn't raise an exception (#17158)
|
|
query = "SELECT strftime('%Y', 'now');"
|
|
with connection.cursor() as cursor:
|
|
cursor.execute(query)
|
|
self.assertEqual(connection.queries[-1]["sql"], query)
|
|
|
|
def test_parameter_quoting(self):
|
|
# The implementation of last_executed_queries isn't optimal. It's
|
|
# worth testing that parameters are quoted (#14091).
|
|
query = "SELECT %s"
|
|
params = ["\"'\\"]
|
|
with connection.cursor() as cursor:
|
|
cursor.execute(query, params)
|
|
# Note that the single quote is repeated
|
|
substituted = "SELECT '\"''\\'"
|
|
self.assertEqual(connection.queries[-1]["sql"], substituted)
|
|
|
|
def test_large_number_of_parameters(self):
|
|
# If SQLITE_MAX_VARIABLE_NUMBER (default = 999) has been changed to be
|
|
# greater than SQLITE_MAX_COLUMN (default = 2000), last_executed_query
|
|
# can hit the SQLITE_MAX_COLUMN limit (#26063).
|
|
with connection.cursor() as cursor:
|
|
sql = "SELECT MAX(%s)" % ", ".join(["%s"] * 2001)
|
|
params = list(range(2001))
|
|
# This should not raise an exception.
|
|
cursor.db.ops.last_executed_query(cursor.cursor, sql, params)
|
|
|
|
|
|
@unittest.skipUnless(connection.vendor == "sqlite", "SQLite tests")
|
|
class EscapingChecks(TestCase):
|
|
"""
|
|
All tests in this test case are also run with settings.DEBUG=True in
|
|
EscapingChecksDebug test case, to also test CursorDebugWrapper.
|
|
"""
|
|
|
|
def test_parameter_escaping(self):
|
|
# '%s' escaping support for sqlite3 (#13648).
|
|
with connection.cursor() as cursor:
|
|
cursor.execute("select strftime('%s', date('now'))")
|
|
response = cursor.fetchall()[0][0]
|
|
# response should be an non-zero integer
|
|
self.assertTrue(int(response))
|
|
|
|
|
|
@unittest.skipUnless(connection.vendor == "sqlite", "SQLite tests")
|
|
@override_settings(DEBUG=True)
|
|
class EscapingChecksDebug(EscapingChecks):
|
|
pass
|
|
|
|
|
|
@unittest.skipUnless(connection.vendor == "sqlite", "SQLite tests")
|
|
class ThreadSharing(TransactionTestCase):
|
|
available_apps = ["backends"]
|
|
|
|
def test_database_sharing_in_threads(self):
|
|
thread_connections = []
|
|
|
|
def create_object():
|
|
Object.objects.create()
|
|
thread_connections.append(connections[DEFAULT_DB_ALIAS].connection)
|
|
|
|
main_connection = connections[DEFAULT_DB_ALIAS].connection
|
|
try:
|
|
create_object()
|
|
thread = threading.Thread(target=create_object)
|
|
thread.start()
|
|
thread.join()
|
|
self.assertEqual(Object.objects.count(), 2)
|
|
finally:
|
|
for conn in thread_connections:
|
|
if conn is not main_connection:
|
|
conn.close()
|
|
|
|
|
|
@unittest.skipUnless(connection.vendor == "sqlite", "SQLite tests")
|
|
class TestTransactionMode(SimpleTestCase):
|
|
databases = {"default"}
|
|
|
|
def test_default_transaction_mode(self):
|
|
with CaptureQueriesContext(connection) as captured_queries:
|
|
with transaction.atomic():
|
|
pass
|
|
|
|
begin_query, commit_query = captured_queries
|
|
self.assertEqual(begin_query["sql"], "BEGIN")
|
|
self.assertEqual(commit_query["sql"], "COMMIT")
|
|
|
|
def test_invalid_transaction_mode(self):
|
|
msg = (
|
|
"settings.DATABASES['default']['OPTIONS']['transaction_mode'] is "
|
|
"improperly configured to 'invalid'. Use one of 'DEFERRED', 'EXCLUSIVE', "
|
|
"'IMMEDIATE', or None."
|
|
)
|
|
with self.change_transaction_mode("invalid") as new_connection:
|
|
with self.assertRaisesMessage(ImproperlyConfigured, msg):
|
|
new_connection.ensure_connection()
|
|
|
|
def test_valid_transaction_modes(self):
|
|
valid_transaction_modes = ("deferred", "immediate", "exclusive")
|
|
for transaction_mode in valid_transaction_modes:
|
|
with (
|
|
self.subTest(transaction_mode=transaction_mode),
|
|
self.change_transaction_mode(transaction_mode) as new_connection,
|
|
CaptureQueriesContext(new_connection) as captured_queries,
|
|
):
|
|
new_connection.set_autocommit(
|
|
False, force_begin_transaction_with_broken_autocommit=True
|
|
)
|
|
new_connection.commit()
|
|
expected_transaction_mode = transaction_mode.upper()
|
|
begin_sql = captured_queries[0]["sql"]
|
|
self.assertEqual(begin_sql, f"BEGIN {expected_transaction_mode}")
|
|
|
|
@contextmanager
|
|
def change_transaction_mode(self, transaction_mode):
|
|
new_connection = connection.copy()
|
|
new_connection.settings_dict["OPTIONS"] = {
|
|
**new_connection.settings_dict["OPTIONS"],
|
|
"transaction_mode": transaction_mode,
|
|
}
|
|
try:
|
|
yield new_connection
|
|
finally:
|
|
new_connection._close()
|