2007-08-20 05:30:57 +08:00
|
|
|
try:
|
|
|
|
# Only exists in Python 2.4+
|
|
|
|
from threading import local
|
|
|
|
except ImportError:
|
|
|
|
# Import copy of _thread_local.py from Python 2.4
|
|
|
|
from django.utils._threading_local import local
|
|
|
|
|
|
|
|
class BaseDatabaseWrapper(local):
|
2007-08-20 06:29:57 +08:00
|
|
|
"""
|
|
|
|
Represents a database connection.
|
|
|
|
"""
|
|
|
|
ops = None
|
2007-08-20 05:30:57 +08:00
|
|
|
def __init__(self, **kwargs):
|
|
|
|
self.connection = None
|
|
|
|
self.queries = []
|
|
|
|
self.options = kwargs
|
|
|
|
|
|
|
|
def _commit(self):
|
|
|
|
if self.connection is not None:
|
|
|
|
return self.connection.commit()
|
|
|
|
|
|
|
|
def _rollback(self):
|
|
|
|
if self.connection is not None:
|
|
|
|
return self.connection.rollback()
|
|
|
|
|
|
|
|
def close(self):
|
|
|
|
if self.connection is not None:
|
|
|
|
self.connection.close()
|
|
|
|
self.connection = None
|
|
|
|
|
|
|
|
def cursor(self):
|
|
|
|
from django.conf import settings
|
|
|
|
cursor = self._cursor(settings)
|
|
|
|
if settings.DEBUG:
|
|
|
|
return self.make_debug_cursor(cursor)
|
|
|
|
return cursor
|
|
|
|
|
|
|
|
def make_debug_cursor(self, cursor):
|
|
|
|
from django.db.backends import util
|
|
|
|
return util.CursorDebugWrapper(cursor, self)
|
2007-08-20 06:29:57 +08:00
|
|
|
|
2007-08-20 10:20:33 +08:00
|
|
|
class BaseDatabaseFeatures(object):
|
|
|
|
allows_group_by_ordinal = True
|
|
|
|
allows_unique_and_pk = True
|
|
|
|
autoindexes_primary_keys = True
|
2007-11-04 13:05:24 +08:00
|
|
|
inline_fk_references = True
|
2007-08-20 10:20:33 +08:00
|
|
|
needs_datetime_string_cast = True
|
|
|
|
needs_upper_for_iops = False
|
|
|
|
supports_constraints = True
|
|
|
|
supports_tablespaces = False
|
|
|
|
uses_case_insensitive_names = False
|
2007-08-20 10:39:05 +08:00
|
|
|
uses_custom_queryset = False
|
2007-08-20 10:20:33 +08:00
|
|
|
|
2007-08-20 06:29:57 +08:00
|
|
|
class BaseDatabaseOperations(object):
|
|
|
|
"""
|
|
|
|
This class encapsulates all backend-specific differences, such as the way
|
|
|
|
a backend performs ordering or calculates the ID of a recently-inserted
|
|
|
|
row.
|
|
|
|
"""
|
2007-09-15 02:12:36 +08:00
|
|
|
def autoinc_sql(self, table, column):
|
2007-08-20 06:29:57 +08:00
|
|
|
"""
|
|
|
|
Returns any SQL needed to support auto-incrementing primary keys, or
|
|
|
|
None if no SQL is necessary.
|
|
|
|
|
|
|
|
This SQL is executed when a table is created.
|
|
|
|
"""
|
|
|
|
return None
|
2007-08-20 06:40:06 +08:00
|
|
|
|
|
|
|
def date_extract_sql(self, lookup_type, field_name):
|
|
|
|
"""
|
|
|
|
Given a lookup_type of 'year', 'month' or 'day', returns the SQL that
|
|
|
|
extracts a value from the given date field field_name.
|
|
|
|
"""
|
|
|
|
raise NotImplementedError()
|
2007-08-20 06:47:43 +08:00
|
|
|
|
|
|
|
def date_trunc_sql(self, lookup_type, field_name):
|
|
|
|
"""
|
|
|
|
Given a lookup_type of 'year', 'month' or 'day', returns the SQL that
|
|
|
|
truncates the given date field field_name to a DATE object with only
|
|
|
|
the given specificity.
|
|
|
|
"""
|
|
|
|
raise NotImplementedError()
|
2007-08-20 06:55:05 +08:00
|
|
|
|
|
|
|
def datetime_cast_sql(self):
|
|
|
|
"""
|
|
|
|
Returns the SQL necessary to cast a datetime value so that it will be
|
|
|
|
retrieved as a Python datetime object instead of a string.
|
2007-08-20 06:57:08 +08:00
|
|
|
|
|
|
|
This SQL should include a '%s' in place of the field's name. This
|
|
|
|
method should return None if no casting is necessary.
|
2007-08-20 06:55:05 +08:00
|
|
|
"""
|
|
|
|
return None
|
2007-08-20 07:03:38 +08:00
|
|
|
|
|
|
|
def deferrable_sql(self):
|
|
|
|
"""
|
|
|
|
Returns the SQL necessary to make a constraint "initially deferred"
|
|
|
|
during a CREATE TABLE statement.
|
|
|
|
"""
|
|
|
|
return ''
|
2007-08-20 07:07:34 +08:00
|
|
|
|
|
|
|
def drop_foreignkey_sql(self):
|
|
|
|
"""
|
|
|
|
Returns the SQL command that drops a foreign key.
|
|
|
|
"""
|
|
|
|
return "DROP CONSTRAINT"
|
2007-08-20 07:13:06 +08:00
|
|
|
|
2007-08-20 11:08:32 +08:00
|
|
|
def drop_sequence_sql(self, table):
|
|
|
|
"""
|
|
|
|
Returns any SQL necessary to drop the sequence for the given table.
|
|
|
|
Returns None if no SQL is necessary.
|
|
|
|
"""
|
|
|
|
return None
|
|
|
|
|
2007-08-20 11:03:40 +08:00
|
|
|
def field_cast_sql(self, db_type):
|
|
|
|
"""
|
|
|
|
Given a column type (e.g. 'BLOB', 'VARCHAR'), returns the SQL necessary
|
|
|
|
to cast it before using it in a WHERE statement. Note that the
|
|
|
|
resulting string should contain a '%s' placeholder for the column being
|
|
|
|
searched against.
|
|
|
|
"""
|
|
|
|
return '%s'
|
|
|
|
|
2007-08-20 07:13:06 +08:00
|
|
|
def fulltext_search_sql(self, field_name):
|
|
|
|
"""
|
|
|
|
Returns the SQL WHERE clause to use in order to perform a full-text
|
|
|
|
search of the given field_name. Note that the resulting string should
|
|
|
|
contain a '%s' placeholder for the value being searched against.
|
|
|
|
"""
|
|
|
|
raise NotImplementedError('Full-text search is not implemented for this database backend')
|
2007-08-20 07:18:43 +08:00
|
|
|
|
2007-10-24 03:00:31 +08:00
|
|
|
def last_executed_query(self, cursor, sql, params):
|
|
|
|
"""
|
|
|
|
Returns a string of the query last executed by the given cursor, with
|
|
|
|
placeholders replaced with actual values.
|
|
|
|
|
|
|
|
`sql` is the raw query containing placeholders, and `params` is the
|
|
|
|
sequence of parameters. These are used by default, but this method
|
|
|
|
exists for database backends to provide a better implementation
|
|
|
|
according to their own quoting schemes.
|
|
|
|
"""
|
|
|
|
from django.utils.encoding import smart_unicode, force_unicode
|
|
|
|
|
|
|
|
# Convert params to contain Unicode values.
|
|
|
|
to_unicode = lambda s: force_unicode(s, strings_only=True)
|
|
|
|
if isinstance(params, (list, tuple)):
|
|
|
|
u_params = tuple([to_unicode(val) for val in params])
|
|
|
|
else:
|
|
|
|
u_params = dict([(to_unicode(k), to_unicode(v)) for k, v in params.items()])
|
|
|
|
|
|
|
|
return smart_unicode(sql) % u_params
|
|
|
|
|
2007-08-20 07:18:43 +08:00
|
|
|
def last_insert_id(self, cursor, table_name, pk_name):
|
|
|
|
"""
|
|
|
|
Given a cursor object that has just performed an INSERT statement into
|
|
|
|
a table that has an auto-incrementing ID, returns the newly created ID.
|
|
|
|
|
|
|
|
This method also receives the table name and the name of the primary-key
|
|
|
|
column.
|
|
|
|
"""
|
|
|
|
return cursor.lastrowid
|
2007-08-20 07:24:59 +08:00
|
|
|
|
|
|
|
def limit_offset_sql(self, limit, offset=None):
|
|
|
|
"""
|
|
|
|
Returns a LIMIT/OFFSET SQL clause, given a limit and optional offset.
|
|
|
|
"""
|
|
|
|
# 'LIMIT 40 OFFSET 20'
|
|
|
|
sql = "LIMIT %s" % limit
|
|
|
|
if offset and offset != 0:
|
|
|
|
sql += " OFFSET %s" % offset
|
|
|
|
return sql
|
2007-08-20 07:53:39 +08:00
|
|
|
|
|
|
|
def max_name_length(self):
|
|
|
|
"""
|
|
|
|
Returns the maximum length of table and column names, or None if there
|
|
|
|
is no limit.
|
|
|
|
"""
|
|
|
|
return None
|
2007-08-20 07:59:06 +08:00
|
|
|
|
|
|
|
def pk_default_value(self):
|
|
|
|
"""
|
|
|
|
Returns the value to use during an INSERT statement to specify that
|
|
|
|
the field should use its default value.
|
|
|
|
"""
|
|
|
|
return 'DEFAULT'
|
2007-08-20 08:04:20 +08:00
|
|
|
|
2007-08-20 10:39:05 +08:00
|
|
|
def query_set_class(self, DefaultQuerySet):
|
|
|
|
"""
|
|
|
|
Given the default QuerySet class, returns a custom QuerySet class
|
|
|
|
to use for this backend. Returns None if a custom QuerySet isn't used.
|
|
|
|
See also BaseDatabaseFeatures.uses_custom_queryset, which regulates
|
|
|
|
whether this method is called at all.
|
|
|
|
"""
|
|
|
|
return None
|
|
|
|
|
2007-08-20 09:03:33 +08:00
|
|
|
def quote_name(self, name):
|
|
|
|
"""
|
|
|
|
Returns a quoted version of the given table, index or column name. Does
|
|
|
|
not quote the given name if it's already been quoted.
|
|
|
|
"""
|
|
|
|
raise NotImplementedError()
|
|
|
|
|
2007-08-20 08:04:20 +08:00
|
|
|
def random_function_sql(self):
|
|
|
|
"""
|
|
|
|
Returns a SQL expression that returns a random value.
|
|
|
|
"""
|
|
|
|
return 'RANDOM()'
|
2007-08-20 08:15:53 +08:00
|
|
|
|
|
|
|
def sql_flush(self, style, tables, sequences):
|
|
|
|
"""
|
|
|
|
Returns a list of SQL statements required to remove all data from
|
|
|
|
the given database tables (without actually removing the tables
|
|
|
|
themselves).
|
|
|
|
|
|
|
|
The `style` argument is a Style object as returned by either
|
|
|
|
color_style() or no_style() in django.core.management.color.
|
|
|
|
"""
|
|
|
|
raise NotImplementedError()
|
2007-08-20 08:21:10 +08:00
|
|
|
|
|
|
|
def sequence_reset_sql(self, style, model_list):
|
|
|
|
"""
|
|
|
|
Returns a list of the SQL statements required to reset sequences for
|
|
|
|
the given models.
|
|
|
|
|
|
|
|
The `style` argument is a Style object as returned by either
|
|
|
|
color_style() or no_style() in django.core.management.color.
|
|
|
|
"""
|
|
|
|
return [] # No sequence reset required by default.
|
2007-08-20 08:24:03 +08:00
|
|
|
|
|
|
|
def start_transaction_sql(self):
|
|
|
|
"""
|
|
|
|
Returns the SQL statement required to start a transaction.
|
|
|
|
"""
|
|
|
|
return "BEGIN;"
|
2007-08-20 08:30:19 +08:00
|
|
|
|
|
|
|
def tablespace_sql(self, tablespace, inline=False):
|
|
|
|
"""
|
|
|
|
Returns the tablespace SQL, or None if the backend doesn't use
|
|
|
|
tablespaces.
|
|
|
|
"""
|
|
|
|
return None
|