Fixed #18330 - Made cache culling 3rd party db backend friendly

This is Ian Kelly's patch from #15580 with minor modifications.
This commit is contained in:
Anssi Kääriäinen 2012-07-05 17:20:48 +03:00
parent 0f49b2bce2
commit d3c2eb103f
3 changed files with 20 additions and 11 deletions

View File

@ -167,17 +167,9 @@ class DatabaseCache(BaseDatabaseCache):
num = cursor.fetchone()[0] num = cursor.fetchone()[0]
if num > self._max_entries: if num > self._max_entries:
cull_num = num / self._cull_frequency cull_num = num / self._cull_frequency
if connections[db].vendor == 'oracle': cursor.execute(
# Oracle doesn't support LIMIT + OFFSET connections[db].ops.cache_key_culling_sql() % table,
cursor.execute("""SELECT cache_key FROM [cull_num])
(SELECT ROW_NUMBER() OVER (ORDER BY cache_key) AS counter, cache_key FROM %s)
WHERE counter > %%s AND COUNTER <= %%s""" % table, [cull_num, cull_num + 1])
else:
# This isn't standard SQL, it's likely to break
# with some non officially supported databases
cursor.execute("SELECT cache_key FROM %s "
"ORDER BY cache_key "
"LIMIT 1 OFFSET %%s" % table, [cull_num])
cursor.execute("DELETE FROM %s " cursor.execute("DELETE FROM %s "
"WHERE cache_key < %%s" % table, "WHERE cache_key < %%s" % table,
[cursor.fetchone()[0]]) [cursor.fetchone()[0]])

View File

@ -475,6 +475,16 @@ class BaseDatabaseOperations(object):
""" """
return None return None
def cache_key_culling_sql(self):
"""
Returns a SQL query that retrieves the first cache key greater than the
n smallest.
This is used by the 'db' cache backend to determine where to start
culling.
"""
return "SELECT cache_key FROM %s ORDER BY cache_key LIMIT 1 OFFSET %%s"
def date_extract_sql(self, lookup_type, field_name): def date_extract_sql(self, lookup_type, field_name):
""" """
Given a lookup_type of 'year', 'month' or 'day', returns the SQL that Given a lookup_type of 'year', 'month' or 'day', returns the SQL that

View File

@ -118,6 +118,13 @@ WHEN (new.%(col_name)s IS NULL)
/""" % locals() /""" % locals()
return sequence_sql, trigger_sql return sequence_sql, trigger_sql
def cache_key_culling_sql(self):
return """
SELECT cache_key
FROM (SELECT cache_key, rank() OVER (ORDER BY cache_key) AS rank FROM %s)
WHERE rank = %%s + 1
"""
def date_extract_sql(self, lookup_type, field_name): def date_extract_sql(self, lookup_type, field_name):
# http://download-east.oracle.com/docs/cd/B10501_01/server.920/a96540/functions42a.htm#1017163 # http://download-east.oracle.com/docs/cd/B10501_01/server.920/a96540/functions42a.htm#1017163
if lookup_type == 'week_day': if lookup_type == 'week_day':