diff --git a/django/core/management/commands/createcachetable.py b/django/core/management/commands/createcachetable.py index 67b327b9fa..a71ff945a9 100644 --- a/django/core/management/commands/createcachetable.py +++ b/django/core/management/commands/createcachetable.py @@ -1,7 +1,7 @@ from optparse import make_option from django.conf import settings -from django.core.cache import create_cache +from django.core.cache import caches from django.core.cache.backends.db import BaseDatabaseCache from django.core.management.base import BaseCommand, CommandError from django.db import connections, router, transaction, models, DEFAULT_DB_ALIAS @@ -30,7 +30,7 @@ class Command(BaseCommand): self.create_table(db, tablename) else: for cache_alias in settings.CACHES: - cache = create_cache(cache_alias) + cache = caches[cache_alias] if isinstance(cache, BaseDatabaseCache): self.create_table(db, cache._table) diff --git a/django/middleware/cache.py b/django/middleware/cache.py index ae4277c766..a99d428ad9 100644 --- a/django/middleware/cache.py +++ b/django/middleware/cache.py @@ -46,7 +46,7 @@ More details about how the caching works: import warnings from django.conf import settings -from django.core.cache import create_cache, caches, DEFAULT_CACHE_ALIAS +from django.core.cache import caches, DEFAULT_CACHE_ALIAS from django.utils.cache import get_cache_key, learn_cache_key, patch_response_headers, get_max_age @@ -196,4 +196,4 @@ class CacheMiddleware(UpdateCacheMiddleware, FetchFromCacheMiddleware): msg = "CACHE_MIDDLEWARE_ANONYMOUS_ONLY has been deprecated and will be removed in Django 1.8." warnings.warn(msg, DeprecationWarning, stacklevel=1) - self.cache = create_cache(self.cache_alias) + self.cache = caches[self.cache_alias] diff --git a/django/test/signals.py b/django/test/signals.py index de8802b381..4af408ea9d 100644 --- a/django/test/signals.py +++ b/django/test/signals.py @@ -1,5 +1,6 @@ import os import time +import threading import warnings from django.conf import settings @@ -19,6 +20,13 @@ setting_changed = Signal(providing_args=["setting", "value", "enter"]) COMPLEX_OVERRIDE_SETTINGS = set(['DATABASES']) +@receiver(setting_changed) +def clear_cache_handlers(**kwargs): + if kwargs['setting'] == 'CACHES': + from django.core.cache import caches + caches._caches = threading.local() + + @receiver(setting_changed) def update_connections_time_zone(**kwargs): if kwargs['setting'] == 'TIME_ZONE': diff --git a/tests/cache/tests.py b/tests/cache/tests.py index 2c834e7d08..4ebcd078de 100644 --- a/tests/cache/tests.py +++ b/tests/cache/tests.py @@ -8,6 +8,7 @@ import os import pickle import random import re +import shutil import string import tempfile import threading @@ -17,9 +18,7 @@ import warnings from django.conf import settings from django.core import management -from django.core.cache import create_cache, caches -from django.core.cache.backends.base import (CacheKeyWarning, - InvalidCacheBackendError) +from django.core.cache import cache, caches, CacheKeyWarning, InvalidCacheBackendError from django.db import connection, router, transaction from django.core.cache.utils import make_template_fragment_key from django.http import HttpResponse, StreamingHttpResponse @@ -28,7 +27,8 @@ from django.middleware.cache import (FetchFromCacheMiddleware, from django.template import Template from django.template.response import TemplateResponse from django.test import TestCase, TransactionTestCase, RequestFactory -from django.test.utils import override_settings, IgnoreDeprecationWarningsMixin +from django.test.utils import (override_settings, IgnoreDeprecationWarningsMixin, + IgnorePendingDeprecationWarningsMixin) from django.utils import six from django.utils import timezone from django.utils import translation @@ -50,72 +50,73 @@ class C: return 24 -class DummyCacheTests(unittest.TestCase): +@override_settings(CACHES={ + 'default': { + 'BACKEND': 'django.core.cache.backends.dummy.DummyCache', + } +}) +class DummyCacheTests(TestCase): # The Dummy cache backend doesn't really behave like a test backend, - # so it has different test requirements. - backend_name = 'django.core.cache.backends.dummy.DummyCache' - - def setUp(self): - self.cache = create_cache(self.backend_name) + # so it has its own test case. def test_simple(self): "Dummy cache backend ignores cache set calls" - self.cache.set("key", "value") - self.assertEqual(self.cache.get("key"), None) + cache.set("key", "value") + self.assertEqual(cache.get("key"), None) def test_add(self): "Add doesn't do anything in dummy cache backend" - self.cache.add("addkey1", "value") - result = self.cache.add("addkey1", "newvalue") + cache.add("addkey1", "value") + result = cache.add("addkey1", "newvalue") self.assertEqual(result, True) - self.assertEqual(self.cache.get("addkey1"), None) + self.assertEqual(cache.get("addkey1"), None) def test_non_existent(self): "Non-existent keys aren't found in the dummy cache backend" - self.assertEqual(self.cache.get("does_not_exist"), None) - self.assertEqual(self.cache.get("does_not_exist", "bang!"), "bang!") + self.assertEqual(cache.get("does_not_exist"), None) + self.assertEqual(cache.get("does_not_exist", "bang!"), "bang!") def test_get_many(self): "get_many returns nothing for the dummy cache backend" - self.cache.set('a', 'a') - self.cache.set('b', 'b') - self.cache.set('c', 'c') - self.cache.set('d', 'd') - self.assertEqual(self.cache.get_many(['a', 'c', 'd']), {}) - self.assertEqual(self.cache.get_many(['a', 'b', 'e']), {}) + cache.set('a', 'a') + cache.set('b', 'b') + cache.set('c', 'c') + cache.set('d', 'd') + self.assertEqual(cache.get_many(['a', 'c', 'd']), {}) + self.assertEqual(cache.get_many(['a', 'b', 'e']), {}) def test_delete(self): "Cache deletion is transparently ignored on the dummy cache backend" - self.cache.set("key1", "spam") - self.cache.set("key2", "eggs") - self.assertEqual(self.cache.get("key1"), None) - self.cache.delete("key1") - self.assertEqual(self.cache.get("key1"), None) - self.assertEqual(self.cache.get("key2"), None) + cache.set("key1", "spam") + cache.set("key2", "eggs") + self.assertEqual(cache.get("key1"), None) + cache.delete("key1") + self.assertEqual(cache.get("key1"), None) + self.assertEqual(cache.get("key2"), None) def test_has_key(self): "The has_key method doesn't ever return True for the dummy cache backend" - self.cache.set("hello1", "goodbye1") - self.assertEqual(self.cache.has_key("hello1"), False) - self.assertEqual(self.cache.has_key("goodbye1"), False) + cache.set("hello1", "goodbye1") + self.assertEqual(cache.has_key("hello1"), False) + self.assertEqual(cache.has_key("goodbye1"), False) def test_in(self): "The in operator doesn't ever return True for the dummy cache backend" - self.cache.set("hello2", "goodbye2") - self.assertEqual("hello2" in self.cache, False) - self.assertEqual("goodbye2" in self.cache, False) + cache.set("hello2", "goodbye2") + self.assertEqual("hello2" in cache, False) + self.assertEqual("goodbye2" in cache, False) def test_incr(self): "Dummy cache values can't be incremented" - self.cache.set('answer', 42) - self.assertRaises(ValueError, self.cache.incr, 'answer') - self.assertRaises(ValueError, self.cache.incr, 'does_not_exist') + cache.set('answer', 42) + self.assertRaises(ValueError, cache.incr, 'answer') + self.assertRaises(ValueError, cache.incr, 'does_not_exist') def test_decr(self): "Dummy cache values can't be decremented" - self.cache.set('answer', 42) - self.assertRaises(ValueError, self.cache.decr, 'answer') - self.assertRaises(ValueError, self.cache.decr, 'does_not_exist') + cache.set('answer', 42) + self.assertRaises(ValueError, cache.decr, 'answer') + self.assertRaises(ValueError, cache.decr, 'does_not_exist') def test_data_types(self): "All data types are ignored equally by the dummy cache" @@ -128,21 +129,21 @@ class DummyCacheTests(unittest.TestCase): 'function': f, 'class': C, } - self.cache.set("stuff", stuff) - self.assertEqual(self.cache.get("stuff"), None) + cache.set("stuff", stuff) + self.assertEqual(cache.get("stuff"), None) def test_expiration(self): "Expiration has no effect on the dummy cache" - self.cache.set('expire1', 'very quickly', 1) - self.cache.set('expire2', 'very quickly', 1) - self.cache.set('expire3', 'very quickly', 1) + cache.set('expire1', 'very quickly', 1) + cache.set('expire2', 'very quickly', 1) + cache.set('expire3', 'very quickly', 1) time.sleep(2) - self.assertEqual(self.cache.get("expire1"), None) + self.assertEqual(cache.get("expire1"), None) - self.cache.add("expire2", "newvalue") - self.assertEqual(self.cache.get("expire2"), None) - self.assertEqual(self.cache.has_key("expire3"), False) + cache.add("expire2", "newvalue") + self.assertEqual(cache.get("expire2"), None) + self.assertEqual(cache.has_key("expire3"), False) def test_unicode(self): "Unicode values are ignored by the dummy cache" @@ -153,33 +154,57 @@ class DummyCacheTests(unittest.TestCase): 'ascii2': {'x': 1} } for (key, value) in stuff.items(): - self.cache.set(key, value) - self.assertEqual(self.cache.get(key), None) + cache.set(key, value) + self.assertEqual(cache.get(key), None) def test_set_many(self): "set_many does nothing for the dummy cache backend" - self.cache.set_many({'a': 1, 'b': 2}) - self.cache.set_many({'a': 1, 'b': 2}, timeout=2, version='1') + cache.set_many({'a': 1, 'b': 2}) + cache.set_many({'a': 1, 'b': 2}, timeout=2, version='1') def test_delete_many(self): "delete_many does nothing for the dummy cache backend" - self.cache.delete_many(['a', 'b']) + cache.delete_many(['a', 'b']) def test_clear(self): "clear does nothing for the dummy cache backend" - self.cache.clear() + cache.clear() def test_incr_version(self): "Dummy cache versions can't be incremented" - self.cache.set('answer', 42) - self.assertRaises(ValueError, self.cache.incr_version, 'answer') - self.assertRaises(ValueError, self.cache.incr_version, 'does_not_exist') + cache.set('answer', 42) + self.assertRaises(ValueError, cache.incr_version, 'answer') + self.assertRaises(ValueError, cache.incr_version, 'does_not_exist') def test_decr_version(self): "Dummy cache versions can't be decremented" - self.cache.set('answer', 42) - self.assertRaises(ValueError, self.cache.decr_version, 'answer') - self.assertRaises(ValueError, self.cache.decr_version, 'does_not_exist') + cache.set('answer', 42) + self.assertRaises(ValueError, cache.decr_version, 'answer') + self.assertRaises(ValueError, cache.decr_version, 'does_not_exist') + + +def custom_key_func(key, key_prefix, version): + "A customized cache key function" + return 'CUSTOM-' + '-'.join([key_prefix, str(version), key]) + + +_caches_setting_base = { + 'default': {}, + 'prefix': {'KEY_PREFIX': 'cacheprefix'}, + 'v2': {'VERSION': 2}, + 'custom_key': {'KEY_FUNCTION': custom_key_func}, + 'custom_key2': {'KEY_FUNCTION': 'cache.tests.custom_key_func'}, + 'cull': {'OPTIONS': {'MAX_ENTRIES': 30}}, + 'zero_cull': {'OPTIONS': {'CULL_FREQUENCY': 0, 'MAX_ENTRIES': 30}}, + 'other': {'LOCATION': 'other'}, +} + + +def caches_setting_for_tests(**params): + setting = dict((k, v.copy()) for k, v in _caches_setting_base.items()) + for cache_params in setting.values(): + cache_params.update(params) + return setting class BaseCacheTests(object): @@ -188,89 +213,92 @@ class BaseCacheTests(object): def setUp(self): self.factory = RequestFactory() + def tearDown(self): + cache.clear() + def test_simple(self): # Simple cache set/get works - self.cache.set("key", "value") - self.assertEqual(self.cache.get("key"), "value") + cache.set("key", "value") + self.assertEqual(cache.get("key"), "value") def test_add(self): # A key can be added to a cache - self.cache.add("addkey1", "value") - result = self.cache.add("addkey1", "newvalue") + cache.add("addkey1", "value") + result = cache.add("addkey1", "newvalue") self.assertEqual(result, False) - self.assertEqual(self.cache.get("addkey1"), "value") + self.assertEqual(cache.get("addkey1"), "value") def test_prefix(self): # Test for same cache key conflicts between shared backend - self.cache.set('somekey', 'value') + cache.set('somekey', 'value') # should not be set in the prefixed cache - self.assertFalse(self.prefix_cache.has_key('somekey')) + self.assertFalse(caches['prefix'].has_key('somekey')) - self.prefix_cache.set('somekey', 'value2') + caches['prefix'].set('somekey', 'value2') - self.assertEqual(self.cache.get('somekey'), 'value') - self.assertEqual(self.prefix_cache.get('somekey'), 'value2') + self.assertEqual(cache.get('somekey'), 'value') + self.assertEqual(caches['prefix'].get('somekey'), 'value2') def test_non_existent(self): # Non-existent cache keys return as None/default # get with non-existent keys - self.assertEqual(self.cache.get("does_not_exist"), None) - self.assertEqual(self.cache.get("does_not_exist", "bang!"), "bang!") + self.assertEqual(cache.get("does_not_exist"), None) + self.assertEqual(cache.get("does_not_exist", "bang!"), "bang!") def test_get_many(self): # Multiple cache keys can be returned using get_many - self.cache.set('a', 'a') - self.cache.set('b', 'b') - self.cache.set('c', 'c') - self.cache.set('d', 'd') - self.assertEqual(self.cache.get_many(['a', 'c', 'd']), {'a': 'a', 'c': 'c', 'd': 'd'}) - self.assertEqual(self.cache.get_many(['a', 'b', 'e']), {'a': 'a', 'b': 'b'}) + cache.set('a', 'a') + cache.set('b', 'b') + cache.set('c', 'c') + cache.set('d', 'd') + self.assertEqual(cache.get_many(['a', 'c', 'd']), {'a': 'a', 'c': 'c', 'd': 'd'}) + self.assertEqual(cache.get_many(['a', 'b', 'e']), {'a': 'a', 'b': 'b'}) def test_delete(self): # Cache keys can be deleted - self.cache.set("key1", "spam") - self.cache.set("key2", "eggs") - self.assertEqual(self.cache.get("key1"), "spam") - self.cache.delete("key1") - self.assertEqual(self.cache.get("key1"), None) - self.assertEqual(self.cache.get("key2"), "eggs") + cache.set("key1", "spam") + cache.set("key2", "eggs") + self.assertEqual(cache.get("key1"), "spam") + cache.delete("key1") + self.assertEqual(cache.get("key1"), None) + self.assertEqual(cache.get("key2"), "eggs") def test_has_key(self): # The cache can be inspected for cache keys - self.cache.set("hello1", "goodbye1") - self.assertEqual(self.cache.has_key("hello1"), True) - self.assertEqual(self.cache.has_key("goodbye1"), False) + cache.set("hello1", "goodbye1") + self.assertEqual(cache.has_key("hello1"), True) + self.assertEqual(cache.has_key("goodbye1"), False) def test_in(self): # The in operator can be used to inspect cache contents - self.cache.set("hello2", "goodbye2") - self.assertEqual("hello2" in self.cache, True) - self.assertEqual("goodbye2" in self.cache, False) + cache.set("hello2", "goodbye2") + self.assertEqual("hello2" in cache, True) + self.assertEqual("goodbye2" in cache, False) def test_incr(self): # Cache values can be incremented - self.cache.set('answer', 41) - self.assertEqual(self.cache.incr('answer'), 42) - self.assertEqual(self.cache.get('answer'), 42) - self.assertEqual(self.cache.incr('answer', 10), 52) - self.assertEqual(self.cache.get('answer'), 52) - self.assertEqual(self.cache.incr('answer', -10), 42) - self.assertRaises(ValueError, self.cache.incr, 'does_not_exist') + cache.set('answer', 41) + self.assertEqual(cache.incr('answer'), 42) + self.assertEqual(cache.get('answer'), 42) + self.assertEqual(cache.incr('answer', 10), 52) + self.assertEqual(cache.get('answer'), 52) + self.assertEqual(cache.incr('answer', -10), 42) + self.assertRaises(ValueError, cache.incr, 'does_not_exist') def test_decr(self): # Cache values can be decremented - self.cache.set('answer', 43) - self.assertEqual(self.cache.decr('answer'), 42) - self.assertEqual(self.cache.get('answer'), 42) - self.assertEqual(self.cache.decr('answer', 10), 32) - self.assertEqual(self.cache.get('answer'), 32) - self.assertEqual(self.cache.decr('answer', -10), 42) - self.assertRaises(ValueError, self.cache.decr, 'does_not_exist') + cache.set('answer', 43) + self.assertEqual(cache.decr('answer'), 42) + self.assertEqual(cache.get('answer'), 42) + self.assertEqual(cache.decr('answer', 10), 32) + self.assertEqual(cache.get('answer'), 32) + self.assertEqual(cache.decr('answer', -10), 42) + self.assertRaises(ValueError, cache.decr, 'does_not_exist') def test_close(self): - self.assertTrue(hasattr(self.cache, 'close')) - self.cache.close() + self.assertTrue(hasattr(cache, 'close')) + cache.close() def test_data_types(self): # Many different data types can be cached @@ -283,8 +311,8 @@ class BaseCacheTests(object): 'function': f, 'class': C, } - self.cache.set("stuff", stuff) - self.assertEqual(self.cache.get("stuff"), stuff) + cache.set("stuff", stuff) + self.assertEqual(cache.get("stuff"), stuff) def test_cache_read_for_model_instance(self): # Don't want fields with callable as default to be called on cache read @@ -293,8 +321,8 @@ class BaseCacheTests(object): my_poll = Poll.objects.create(question="Well?") self.assertEqual(Poll.objects.count(), 1) pub_date = my_poll.pub_date - self.cache.set('question', my_poll) - cached_poll = self.cache.get('question') + cache.set('question', my_poll) + cached_poll = cache.get('question') self.assertEqual(cached_poll.pub_date, pub_date) # We only want the default expensive calculation run once self.assertEqual(expensive_calculation.num_runs, 1) @@ -308,7 +336,7 @@ class BaseCacheTests(object): defer_qs = Poll.objects.all().defer('question') self.assertEqual(defer_qs.count(), 1) self.assertEqual(expensive_calculation.num_runs, 1) - self.cache.set('deferred_queryset', defer_qs) + cache.set('deferred_queryset', defer_qs) # cache set should not re-evaluate default functions self.assertEqual(expensive_calculation.num_runs, 1) @@ -320,25 +348,25 @@ class BaseCacheTests(object): self.assertEqual(expensive_calculation.num_runs, 1) defer_qs = Poll.objects.all().defer('question') self.assertEqual(defer_qs.count(), 1) - self.cache.set('deferred_queryset', defer_qs) + cache.set('deferred_queryset', defer_qs) self.assertEqual(expensive_calculation.num_runs, 1) runs_before_cache_read = expensive_calculation.num_runs - self.cache.get('deferred_queryset') + cache.get('deferred_queryset') # We only want the default expensive calculation run on creation and set self.assertEqual(expensive_calculation.num_runs, runs_before_cache_read) def test_expiration(self): # Cache values can be set to expire - self.cache.set('expire1', 'very quickly', 1) - self.cache.set('expire2', 'very quickly', 1) - self.cache.set('expire3', 'very quickly', 1) + cache.set('expire1', 'very quickly', 1) + cache.set('expire2', 'very quickly', 1) + cache.set('expire3', 'very quickly', 1) time.sleep(2) - self.assertEqual(self.cache.get("expire1"), None) + self.assertEqual(cache.get("expire1"), None) - self.cache.add("expire2", "newvalue") - self.assertEqual(self.cache.get("expire2"), "newvalue") - self.assertEqual(self.cache.has_key("expire3"), False) + cache.add("expire2", "newvalue") + self.assertEqual(cache.get("expire2"), "newvalue") + self.assertEqual(cache.has_key("expire3"), False) def test_unicode(self): # Unicode values can be cached @@ -350,21 +378,21 @@ class BaseCacheTests(object): } # Test `set` for (key, value) in stuff.items(): - self.cache.set(key, value) - self.assertEqual(self.cache.get(key), value) + cache.set(key, value) + self.assertEqual(cache.get(key), value) # Test `add` for (key, value) in stuff.items(): - self.cache.delete(key) - self.cache.add(key, value) - self.assertEqual(self.cache.get(key), value) + cache.delete(key) + cache.add(key, value) + self.assertEqual(cache.get(key), value) # Test `set_many` for (key, value) in stuff.items(): - self.cache.delete(key) - self.cache.set_many(stuff) + cache.delete(key) + cache.set_many(stuff) for (key, value) in stuff.items(): - self.assertEqual(self.cache.get(key), value) + self.assertEqual(cache.get(key), value) def test_binary_string(self): # Binary strings should be cacheable @@ -373,53 +401,53 @@ class BaseCacheTests(object): compressed_value = compress(value.encode()) # Test set - self.cache.set('binary1', compressed_value) - compressed_result = self.cache.get('binary1') + cache.set('binary1', compressed_value) + compressed_result = cache.get('binary1') self.assertEqual(compressed_value, compressed_result) self.assertEqual(value, decompress(compressed_result).decode()) # Test add - self.cache.add('binary1-add', compressed_value) - compressed_result = self.cache.get('binary1-add') + cache.add('binary1-add', compressed_value) + compressed_result = cache.get('binary1-add') self.assertEqual(compressed_value, compressed_result) self.assertEqual(value, decompress(compressed_result).decode()) # Test set_many - self.cache.set_many({'binary1-set_many': compressed_value}) - compressed_result = self.cache.get('binary1-set_many') + cache.set_many({'binary1-set_many': compressed_value}) + compressed_result = cache.get('binary1-set_many') self.assertEqual(compressed_value, compressed_result) self.assertEqual(value, decompress(compressed_result).decode()) def test_set_many(self): # Multiple keys can be set using set_many - self.cache.set_many({"key1": "spam", "key2": "eggs"}) - self.assertEqual(self.cache.get("key1"), "spam") - self.assertEqual(self.cache.get("key2"), "eggs") + cache.set_many({"key1": "spam", "key2": "eggs"}) + self.assertEqual(cache.get("key1"), "spam") + self.assertEqual(cache.get("key2"), "eggs") def test_set_many_expiration(self): # set_many takes a second ``timeout`` parameter - self.cache.set_many({"key1": "spam", "key2": "eggs"}, 1) + cache.set_many({"key1": "spam", "key2": "eggs"}, 1) time.sleep(2) - self.assertEqual(self.cache.get("key1"), None) - self.assertEqual(self.cache.get("key2"), None) + self.assertEqual(cache.get("key1"), None) + self.assertEqual(cache.get("key2"), None) def test_delete_many(self): # Multiple keys can be deleted using delete_many - self.cache.set("key1", "spam") - self.cache.set("key2", "eggs") - self.cache.set("key3", "ham") - self.cache.delete_many(["key1", "key2"]) - self.assertEqual(self.cache.get("key1"), None) - self.assertEqual(self.cache.get("key2"), None) - self.assertEqual(self.cache.get("key3"), "ham") + cache.set("key1", "spam") + cache.set("key2", "eggs") + cache.set("key3", "ham") + cache.delete_many(["key1", "key2"]) + self.assertEqual(cache.get("key1"), None) + self.assertEqual(cache.get("key2"), None) + self.assertEqual(cache.get("key3"), "ham") def test_clear(self): # The cache can be emptied using clear - self.cache.set("key1", "spam") - self.cache.set("key2", "eggs") - self.cache.clear() - self.assertEqual(self.cache.get("key1"), None) - self.assertEqual(self.cache.get("key2"), None) + cache.set("key1", "spam") + cache.set("key2", "eggs") + cache.clear() + self.assertEqual(cache.get("key1"), None) + self.assertEqual(cache.get("key2"), None) def test_long_timeout(self): ''' @@ -427,63 +455,67 @@ class BaseCacheTests(object): it is an absolute expiration timestamp instead of a relative offset. Test that we honour this convention. Refs #12399. ''' - self.cache.set('key1', 'eggs', 60 * 60 * 24 * 30 + 1) # 30 days + 1 second - self.assertEqual(self.cache.get('key1'), 'eggs') + cache.set('key1', 'eggs', 60 * 60 * 24 * 30 + 1) # 30 days + 1 second + self.assertEqual(cache.get('key1'), 'eggs') - self.cache.add('key2', 'ham', 60 * 60 * 24 * 30 + 1) - self.assertEqual(self.cache.get('key2'), 'ham') + cache.add('key2', 'ham', 60 * 60 * 24 * 30 + 1) + self.assertEqual(cache.get('key2'), 'ham') - self.cache.set_many({'key3': 'sausage', 'key4': 'lobster bisque'}, 60 * 60 * 24 * 30 + 1) - self.assertEqual(self.cache.get('key3'), 'sausage') - self.assertEqual(self.cache.get('key4'), 'lobster bisque') + cache.set_many({'key3': 'sausage', 'key4': 'lobster bisque'}, 60 * 60 * 24 * 30 + 1) + self.assertEqual(cache.get('key3'), 'sausage') + self.assertEqual(cache.get('key4'), 'lobster bisque') def test_forever_timeout(self): ''' Passing in None into timeout results in a value that is cached forever ''' - self.cache.set('key1', 'eggs', None) - self.assertEqual(self.cache.get('key1'), 'eggs') + cache.set('key1', 'eggs', None) + self.assertEqual(cache.get('key1'), 'eggs') - self.cache.add('key2', 'ham', None) - self.assertEqual(self.cache.get('key2'), 'ham') + cache.add('key2', 'ham', None) + self.assertEqual(cache.get('key2'), 'ham') - self.cache.set_many({'key3': 'sausage', 'key4': 'lobster bisque'}, None) - self.assertEqual(self.cache.get('key3'), 'sausage') - self.assertEqual(self.cache.get('key4'), 'lobster bisque') + cache.set_many({'key3': 'sausage', 'key4': 'lobster bisque'}, None) + self.assertEqual(cache.get('key3'), 'sausage') + self.assertEqual(cache.get('key4'), 'lobster bisque') def test_zero_timeout(self): ''' Passing in None into timeout results in a value that is cached forever ''' - self.cache.set('key1', 'eggs', 0) - self.assertEqual(self.cache.get('key1'), None) + cache.set('key1', 'eggs', 0) + self.assertEqual(cache.get('key1'), None) - self.cache.add('key2', 'ham', 0) - self.assertEqual(self.cache.get('key2'), None) + cache.add('key2', 'ham', 0) + self.assertEqual(cache.get('key2'), None) - self.cache.set_many({'key3': 'sausage', 'key4': 'lobster bisque'}, 0) - self.assertEqual(self.cache.get('key3'), None) - self.assertEqual(self.cache.get('key4'), None) + cache.set_many({'key3': 'sausage', 'key4': 'lobster bisque'}, 0) + self.assertEqual(cache.get('key3'), None) + self.assertEqual(cache.get('key4'), None) def test_float_timeout(self): # Make sure a timeout given as a float doesn't crash anything. - self.cache.set("key1", "spam", 100.2) - self.assertEqual(self.cache.get("key1"), "spam") + cache.set("key1", "spam", 100.2) + self.assertEqual(cache.get("key1"), "spam") - def perform_cull_test(self, initial_count, final_count): - """This is implemented as a utility method, because only some of the backends - implement culling. The culling algorithm also varies slightly, so the final - number of entries will vary between backends""" - # Create initial cache key entries. This will overflow the cache, causing a cull + def _perform_cull_test(self, cull_cache, initial_count, final_count): + # Create initial cache key entries. This will overflow the cache, + # causing a cull. for i in range(1, initial_count): - self.cache.set('cull%d' % i, 'value', 1000) + cull_cache.set('cull%d' % i, 'value', 1000) count = 0 # Count how many keys are left in the cache. for i in range(1, initial_count): - if self.cache.has_key('cull%d' % i): + if cull_cache.has_key('cull%d' % i): count = count + 1 self.assertEqual(count, final_count) + def test_cull(self): + self._perform_cull_test(caches['cull'], 50, 29) + + def test_zero_cull(self): + self._perform_cull_test(caches['zero_cull'], 50, 19) + def test_invalid_keys(self): """ All the builtin backends (except memcached, see below) should warn on @@ -497,302 +529,302 @@ class BaseCacheTests(object): def func(key, *args): return key - old_func = self.cache.key_func - self.cache.key_func = func + old_func = cache.key_func + cache.key_func = func try: with warnings.catch_warnings(record=True) as w: warnings.simplefilter("always") # memcached does not allow whitespace or control characters in keys - self.cache.set('key with spaces', 'value') + cache.set('key with spaces', 'value') self.assertEqual(len(w), 2) self.assertIsInstance(w[0].message, CacheKeyWarning) with warnings.catch_warnings(record=True) as w: warnings.simplefilter("always") # memcached limits key length to 250 - self.cache.set('a' * 251, 'value') + cache.set('a' * 251, 'value') self.assertEqual(len(w), 1) self.assertIsInstance(w[0].message, CacheKeyWarning) finally: - self.cache.key_func = old_func + cache.key_func = old_func def test_cache_versioning_get_set(self): # set, using default version = 1 - self.cache.set('answer1', 42) - self.assertEqual(self.cache.get('answer1'), 42) - self.assertEqual(self.cache.get('answer1', version=1), 42) - self.assertEqual(self.cache.get('answer1', version=2), None) + cache.set('answer1', 42) + self.assertEqual(cache.get('answer1'), 42) + self.assertEqual(cache.get('answer1', version=1), 42) + self.assertEqual(cache.get('answer1', version=2), None) - self.assertEqual(self.v2_cache.get('answer1'), None) - self.assertEqual(self.v2_cache.get('answer1', version=1), 42) - self.assertEqual(self.v2_cache.get('answer1', version=2), None) + self.assertEqual(caches['v2'].get('answer1'), None) + self.assertEqual(caches['v2'].get('answer1', version=1), 42) + self.assertEqual(caches['v2'].get('answer1', version=2), None) # set, default version = 1, but manually override version = 2 - self.cache.set('answer2', 42, version=2) - self.assertEqual(self.cache.get('answer2'), None) - self.assertEqual(self.cache.get('answer2', version=1), None) - self.assertEqual(self.cache.get('answer2', version=2), 42) + cache.set('answer2', 42, version=2) + self.assertEqual(cache.get('answer2'), None) + self.assertEqual(cache.get('answer2', version=1), None) + self.assertEqual(cache.get('answer2', version=2), 42) - self.assertEqual(self.v2_cache.get('answer2'), 42) - self.assertEqual(self.v2_cache.get('answer2', version=1), None) - self.assertEqual(self.v2_cache.get('answer2', version=2), 42) + self.assertEqual(caches['v2'].get('answer2'), 42) + self.assertEqual(caches['v2'].get('answer2', version=1), None) + self.assertEqual(caches['v2'].get('answer2', version=2), 42) # v2 set, using default version = 2 - self.v2_cache.set('answer3', 42) - self.assertEqual(self.cache.get('answer3'), None) - self.assertEqual(self.cache.get('answer3', version=1), None) - self.assertEqual(self.cache.get('answer3', version=2), 42) + caches['v2'].set('answer3', 42) + self.assertEqual(cache.get('answer3'), None) + self.assertEqual(cache.get('answer3', version=1), None) + self.assertEqual(cache.get('answer3', version=2), 42) - self.assertEqual(self.v2_cache.get('answer3'), 42) - self.assertEqual(self.v2_cache.get('answer3', version=1), None) - self.assertEqual(self.v2_cache.get('answer3', version=2), 42) + self.assertEqual(caches['v2'].get('answer3'), 42) + self.assertEqual(caches['v2'].get('answer3', version=1), None) + self.assertEqual(caches['v2'].get('answer3', version=2), 42) # v2 set, default version = 2, but manually override version = 1 - self.v2_cache.set('answer4', 42, version=1) - self.assertEqual(self.cache.get('answer4'), 42) - self.assertEqual(self.cache.get('answer4', version=1), 42) - self.assertEqual(self.cache.get('answer4', version=2), None) + caches['v2'].set('answer4', 42, version=1) + self.assertEqual(cache.get('answer4'), 42) + self.assertEqual(cache.get('answer4', version=1), 42) + self.assertEqual(cache.get('answer4', version=2), None) - self.assertEqual(self.v2_cache.get('answer4'), None) - self.assertEqual(self.v2_cache.get('answer4', version=1), 42) - self.assertEqual(self.v2_cache.get('answer4', version=2), None) + self.assertEqual(caches['v2'].get('answer4'), None) + self.assertEqual(caches['v2'].get('answer4', version=1), 42) + self.assertEqual(caches['v2'].get('answer4', version=2), None) def test_cache_versioning_add(self): # add, default version = 1, but manually override version = 2 - self.cache.add('answer1', 42, version=2) - self.assertEqual(self.cache.get('answer1', version=1), None) - self.assertEqual(self.cache.get('answer1', version=2), 42) + cache.add('answer1', 42, version=2) + self.assertEqual(cache.get('answer1', version=1), None) + self.assertEqual(cache.get('answer1', version=2), 42) - self.cache.add('answer1', 37, version=2) - self.assertEqual(self.cache.get('answer1', version=1), None) - self.assertEqual(self.cache.get('answer1', version=2), 42) + cache.add('answer1', 37, version=2) + self.assertEqual(cache.get('answer1', version=1), None) + self.assertEqual(cache.get('answer1', version=2), 42) - self.cache.add('answer1', 37, version=1) - self.assertEqual(self.cache.get('answer1', version=1), 37) - self.assertEqual(self.cache.get('answer1', version=2), 42) + cache.add('answer1', 37, version=1) + self.assertEqual(cache.get('answer1', version=1), 37) + self.assertEqual(cache.get('answer1', version=2), 42) # v2 add, using default version = 2 - self.v2_cache.add('answer2', 42) - self.assertEqual(self.cache.get('answer2', version=1), None) - self.assertEqual(self.cache.get('answer2', version=2), 42) + caches['v2'].add('answer2', 42) + self.assertEqual(cache.get('answer2', version=1), None) + self.assertEqual(cache.get('answer2', version=2), 42) - self.v2_cache.add('answer2', 37) - self.assertEqual(self.cache.get('answer2', version=1), None) - self.assertEqual(self.cache.get('answer2', version=2), 42) + caches['v2'].add('answer2', 37) + self.assertEqual(cache.get('answer2', version=1), None) + self.assertEqual(cache.get('answer2', version=2), 42) - self.v2_cache.add('answer2', 37, version=1) - self.assertEqual(self.cache.get('answer2', version=1), 37) - self.assertEqual(self.cache.get('answer2', version=2), 42) + caches['v2'].add('answer2', 37, version=1) + self.assertEqual(cache.get('answer2', version=1), 37) + self.assertEqual(cache.get('answer2', version=2), 42) # v2 add, default version = 2, but manually override version = 1 - self.v2_cache.add('answer3', 42, version=1) - self.assertEqual(self.cache.get('answer3', version=1), 42) - self.assertEqual(self.cache.get('answer3', version=2), None) + caches['v2'].add('answer3', 42, version=1) + self.assertEqual(cache.get('answer3', version=1), 42) + self.assertEqual(cache.get('answer3', version=2), None) - self.v2_cache.add('answer3', 37, version=1) - self.assertEqual(self.cache.get('answer3', version=1), 42) - self.assertEqual(self.cache.get('answer3', version=2), None) + caches['v2'].add('answer3', 37, version=1) + self.assertEqual(cache.get('answer3', version=1), 42) + self.assertEqual(cache.get('answer3', version=2), None) - self.v2_cache.add('answer3', 37) - self.assertEqual(self.cache.get('answer3', version=1), 42) - self.assertEqual(self.cache.get('answer3', version=2), 37) + caches['v2'].add('answer3', 37) + self.assertEqual(cache.get('answer3', version=1), 42) + self.assertEqual(cache.get('answer3', version=2), 37) def test_cache_versioning_has_key(self): - self.cache.set('answer1', 42) + cache.set('answer1', 42) # has_key - self.assertTrue(self.cache.has_key('answer1')) - self.assertTrue(self.cache.has_key('answer1', version=1)) - self.assertFalse(self.cache.has_key('answer1', version=2)) + self.assertTrue(cache.has_key('answer1')) + self.assertTrue(cache.has_key('answer1', version=1)) + self.assertFalse(cache.has_key('answer1', version=2)) - self.assertFalse(self.v2_cache.has_key('answer1')) - self.assertTrue(self.v2_cache.has_key('answer1', version=1)) - self.assertFalse(self.v2_cache.has_key('answer1', version=2)) + self.assertFalse(caches['v2'].has_key('answer1')) + self.assertTrue(caches['v2'].has_key('answer1', version=1)) + self.assertFalse(caches['v2'].has_key('answer1', version=2)) def test_cache_versioning_delete(self): - self.cache.set('answer1', 37, version=1) - self.cache.set('answer1', 42, version=2) - self.cache.delete('answer1') - self.assertEqual(self.cache.get('answer1', version=1), None) - self.assertEqual(self.cache.get('answer1', version=2), 42) + cache.set('answer1', 37, version=1) + cache.set('answer1', 42, version=2) + cache.delete('answer1') + self.assertEqual(cache.get('answer1', version=1), None) + self.assertEqual(cache.get('answer1', version=2), 42) - self.cache.set('answer2', 37, version=1) - self.cache.set('answer2', 42, version=2) - self.cache.delete('answer2', version=2) - self.assertEqual(self.cache.get('answer2', version=1), 37) - self.assertEqual(self.cache.get('answer2', version=2), None) + cache.set('answer2', 37, version=1) + cache.set('answer2', 42, version=2) + cache.delete('answer2', version=2) + self.assertEqual(cache.get('answer2', version=1), 37) + self.assertEqual(cache.get('answer2', version=2), None) - self.cache.set('answer3', 37, version=1) - self.cache.set('answer3', 42, version=2) - self.v2_cache.delete('answer3') - self.assertEqual(self.cache.get('answer3', version=1), 37) - self.assertEqual(self.cache.get('answer3', version=2), None) + cache.set('answer3', 37, version=1) + cache.set('answer3', 42, version=2) + caches['v2'].delete('answer3') + self.assertEqual(cache.get('answer3', version=1), 37) + self.assertEqual(cache.get('answer3', version=2), None) - self.cache.set('answer4', 37, version=1) - self.cache.set('answer4', 42, version=2) - self.v2_cache.delete('answer4', version=1) - self.assertEqual(self.cache.get('answer4', version=1), None) - self.assertEqual(self.cache.get('answer4', version=2), 42) + cache.set('answer4', 37, version=1) + cache.set('answer4', 42, version=2) + caches['v2'].delete('answer4', version=1) + self.assertEqual(cache.get('answer4', version=1), None) + self.assertEqual(cache.get('answer4', version=2), 42) def test_cache_versioning_incr_decr(self): - self.cache.set('answer1', 37, version=1) - self.cache.set('answer1', 42, version=2) - self.cache.incr('answer1') - self.assertEqual(self.cache.get('answer1', version=1), 38) - self.assertEqual(self.cache.get('answer1', version=2), 42) - self.cache.decr('answer1') - self.assertEqual(self.cache.get('answer1', version=1), 37) - self.assertEqual(self.cache.get('answer1', version=2), 42) + cache.set('answer1', 37, version=1) + cache.set('answer1', 42, version=2) + cache.incr('answer1') + self.assertEqual(cache.get('answer1', version=1), 38) + self.assertEqual(cache.get('answer1', version=2), 42) + cache.decr('answer1') + self.assertEqual(cache.get('answer1', version=1), 37) + self.assertEqual(cache.get('answer1', version=2), 42) - self.cache.set('answer2', 37, version=1) - self.cache.set('answer2', 42, version=2) - self.cache.incr('answer2', version=2) - self.assertEqual(self.cache.get('answer2', version=1), 37) - self.assertEqual(self.cache.get('answer2', version=2), 43) - self.cache.decr('answer2', version=2) - self.assertEqual(self.cache.get('answer2', version=1), 37) - self.assertEqual(self.cache.get('answer2', version=2), 42) + cache.set('answer2', 37, version=1) + cache.set('answer2', 42, version=2) + cache.incr('answer2', version=2) + self.assertEqual(cache.get('answer2', version=1), 37) + self.assertEqual(cache.get('answer2', version=2), 43) + cache.decr('answer2', version=2) + self.assertEqual(cache.get('answer2', version=1), 37) + self.assertEqual(cache.get('answer2', version=2), 42) - self.cache.set('answer3', 37, version=1) - self.cache.set('answer3', 42, version=2) - self.v2_cache.incr('answer3') - self.assertEqual(self.cache.get('answer3', version=1), 37) - self.assertEqual(self.cache.get('answer3', version=2), 43) - self.v2_cache.decr('answer3') - self.assertEqual(self.cache.get('answer3', version=1), 37) - self.assertEqual(self.cache.get('answer3', version=2), 42) + cache.set('answer3', 37, version=1) + cache.set('answer3', 42, version=2) + caches['v2'].incr('answer3') + self.assertEqual(cache.get('answer3', version=1), 37) + self.assertEqual(cache.get('answer3', version=2), 43) + caches['v2'].decr('answer3') + self.assertEqual(cache.get('answer3', version=1), 37) + self.assertEqual(cache.get('answer3', version=2), 42) - self.cache.set('answer4', 37, version=1) - self.cache.set('answer4', 42, version=2) - self.v2_cache.incr('answer4', version=1) - self.assertEqual(self.cache.get('answer4', version=1), 38) - self.assertEqual(self.cache.get('answer4', version=2), 42) - self.v2_cache.decr('answer4', version=1) - self.assertEqual(self.cache.get('answer4', version=1), 37) - self.assertEqual(self.cache.get('answer4', version=2), 42) + cache.set('answer4', 37, version=1) + cache.set('answer4', 42, version=2) + caches['v2'].incr('answer4', version=1) + self.assertEqual(cache.get('answer4', version=1), 38) + self.assertEqual(cache.get('answer4', version=2), 42) + caches['v2'].decr('answer4', version=1) + self.assertEqual(cache.get('answer4', version=1), 37) + self.assertEqual(cache.get('answer4', version=2), 42) def test_cache_versioning_get_set_many(self): # set, using default version = 1 - self.cache.set_many({'ford1': 37, 'arthur1': 42}) - self.assertEqual(self.cache.get_many(['ford1', 'arthur1']), + cache.set_many({'ford1': 37, 'arthur1': 42}) + self.assertEqual(cache.get_many(['ford1', 'arthur1']), {'ford1': 37, 'arthur1': 42}) - self.assertEqual(self.cache.get_many(['ford1', 'arthur1'], version=1), + self.assertEqual(cache.get_many(['ford1', 'arthur1'], version=1), {'ford1': 37, 'arthur1': 42}) - self.assertEqual(self.cache.get_many(['ford1', 'arthur1'], version=2), {}) + self.assertEqual(cache.get_many(['ford1', 'arthur1'], version=2), {}) - self.assertEqual(self.v2_cache.get_many(['ford1', 'arthur1']), {}) - self.assertEqual(self.v2_cache.get_many(['ford1', 'arthur1'], version=1), + self.assertEqual(caches['v2'].get_many(['ford1', 'arthur1']), {}) + self.assertEqual(caches['v2'].get_many(['ford1', 'arthur1'], version=1), {'ford1': 37, 'arthur1': 42}) - self.assertEqual(self.v2_cache.get_many(['ford1', 'arthur1'], version=2), {}) + self.assertEqual(caches['v2'].get_many(['ford1', 'arthur1'], version=2), {}) # set, default version = 1, but manually override version = 2 - self.cache.set_many({'ford2': 37, 'arthur2': 42}, version=2) - self.assertEqual(self.cache.get_many(['ford2', 'arthur2']), {}) - self.assertEqual(self.cache.get_many(['ford2', 'arthur2'], version=1), {}) - self.assertEqual(self.cache.get_many(['ford2', 'arthur2'], version=2), + cache.set_many({'ford2': 37, 'arthur2': 42}, version=2) + self.assertEqual(cache.get_many(['ford2', 'arthur2']), {}) + self.assertEqual(cache.get_many(['ford2', 'arthur2'], version=1), {}) + self.assertEqual(cache.get_many(['ford2', 'arthur2'], version=2), {'ford2': 37, 'arthur2': 42}) - self.assertEqual(self.v2_cache.get_many(['ford2', 'arthur2']), + self.assertEqual(caches['v2'].get_many(['ford2', 'arthur2']), {'ford2': 37, 'arthur2': 42}) - self.assertEqual(self.v2_cache.get_many(['ford2', 'arthur2'], version=1), {}) - self.assertEqual(self.v2_cache.get_many(['ford2', 'arthur2'], version=2), + self.assertEqual(caches['v2'].get_many(['ford2', 'arthur2'], version=1), {}) + self.assertEqual(caches['v2'].get_many(['ford2', 'arthur2'], version=2), {'ford2': 37, 'arthur2': 42}) # v2 set, using default version = 2 - self.v2_cache.set_many({'ford3': 37, 'arthur3': 42}) - self.assertEqual(self.cache.get_many(['ford3', 'arthur3']), {}) - self.assertEqual(self.cache.get_many(['ford3', 'arthur3'], version=1), {}) - self.assertEqual(self.cache.get_many(['ford3', 'arthur3'], version=2), + caches['v2'].set_many({'ford3': 37, 'arthur3': 42}) + self.assertEqual(cache.get_many(['ford3', 'arthur3']), {}) + self.assertEqual(cache.get_many(['ford3', 'arthur3'], version=1), {}) + self.assertEqual(cache.get_many(['ford3', 'arthur3'], version=2), {'ford3': 37, 'arthur3': 42}) - self.assertEqual(self.v2_cache.get_many(['ford3', 'arthur3']), + self.assertEqual(caches['v2'].get_many(['ford3', 'arthur3']), {'ford3': 37, 'arthur3': 42}) - self.assertEqual(self.v2_cache.get_many(['ford3', 'arthur3'], version=1), {}) - self.assertEqual(self.v2_cache.get_many(['ford3', 'arthur3'], version=2), + self.assertEqual(caches['v2'].get_many(['ford3', 'arthur3'], version=1), {}) + self.assertEqual(caches['v2'].get_many(['ford3', 'arthur3'], version=2), {'ford3': 37, 'arthur3': 42}) # v2 set, default version = 2, but manually override version = 1 - self.v2_cache.set_many({'ford4': 37, 'arthur4': 42}, version=1) - self.assertEqual(self.cache.get_many(['ford4', 'arthur4']), + caches['v2'].set_many({'ford4': 37, 'arthur4': 42}, version=1) + self.assertEqual(cache.get_many(['ford4', 'arthur4']), {'ford4': 37, 'arthur4': 42}) - self.assertEqual(self.cache.get_many(['ford4', 'arthur4'], version=1), + self.assertEqual(cache.get_many(['ford4', 'arthur4'], version=1), {'ford4': 37, 'arthur4': 42}) - self.assertEqual(self.cache.get_many(['ford4', 'arthur4'], version=2), {}) + self.assertEqual(cache.get_many(['ford4', 'arthur4'], version=2), {}) - self.assertEqual(self.v2_cache.get_many(['ford4', 'arthur4']), {}) - self.assertEqual(self.v2_cache.get_many(['ford4', 'arthur4'], version=1), + self.assertEqual(caches['v2'].get_many(['ford4', 'arthur4']), {}) + self.assertEqual(caches['v2'].get_many(['ford4', 'arthur4'], version=1), {'ford4': 37, 'arthur4': 42}) - self.assertEqual(self.v2_cache.get_many(['ford4', 'arthur4'], version=2), {}) + self.assertEqual(caches['v2'].get_many(['ford4', 'arthur4'], version=2), {}) def test_incr_version(self): - self.cache.set('answer', 42, version=2) - self.assertEqual(self.cache.get('answer'), None) - self.assertEqual(self.cache.get('answer', version=1), None) - self.assertEqual(self.cache.get('answer', version=2), 42) - self.assertEqual(self.cache.get('answer', version=3), None) + cache.set('answer', 42, version=2) + self.assertEqual(cache.get('answer'), None) + self.assertEqual(cache.get('answer', version=1), None) + self.assertEqual(cache.get('answer', version=2), 42) + self.assertEqual(cache.get('answer', version=3), None) - self.assertEqual(self.cache.incr_version('answer', version=2), 3) - self.assertEqual(self.cache.get('answer'), None) - self.assertEqual(self.cache.get('answer', version=1), None) - self.assertEqual(self.cache.get('answer', version=2), None) - self.assertEqual(self.cache.get('answer', version=3), 42) + self.assertEqual(cache.incr_version('answer', version=2), 3) + self.assertEqual(cache.get('answer'), None) + self.assertEqual(cache.get('answer', version=1), None) + self.assertEqual(cache.get('answer', version=2), None) + self.assertEqual(cache.get('answer', version=3), 42) - self.v2_cache.set('answer2', 42) - self.assertEqual(self.v2_cache.get('answer2'), 42) - self.assertEqual(self.v2_cache.get('answer2', version=1), None) - self.assertEqual(self.v2_cache.get('answer2', version=2), 42) - self.assertEqual(self.v2_cache.get('answer2', version=3), None) + caches['v2'].set('answer2', 42) + self.assertEqual(caches['v2'].get('answer2'), 42) + self.assertEqual(caches['v2'].get('answer2', version=1), None) + self.assertEqual(caches['v2'].get('answer2', version=2), 42) + self.assertEqual(caches['v2'].get('answer2', version=3), None) - self.assertEqual(self.v2_cache.incr_version('answer2'), 3) - self.assertEqual(self.v2_cache.get('answer2'), None) - self.assertEqual(self.v2_cache.get('answer2', version=1), None) - self.assertEqual(self.v2_cache.get('answer2', version=2), None) - self.assertEqual(self.v2_cache.get('answer2', version=3), 42) + self.assertEqual(caches['v2'].incr_version('answer2'), 3) + self.assertEqual(caches['v2'].get('answer2'), None) + self.assertEqual(caches['v2'].get('answer2', version=1), None) + self.assertEqual(caches['v2'].get('answer2', version=2), None) + self.assertEqual(caches['v2'].get('answer2', version=3), 42) - self.assertRaises(ValueError, self.cache.incr_version, 'does_not_exist') + self.assertRaises(ValueError, cache.incr_version, 'does_not_exist') def test_decr_version(self): - self.cache.set('answer', 42, version=2) - self.assertEqual(self.cache.get('answer'), None) - self.assertEqual(self.cache.get('answer', version=1), None) - self.assertEqual(self.cache.get('answer', version=2), 42) + cache.set('answer', 42, version=2) + self.assertEqual(cache.get('answer'), None) + self.assertEqual(cache.get('answer', version=1), None) + self.assertEqual(cache.get('answer', version=2), 42) - self.assertEqual(self.cache.decr_version('answer', version=2), 1) - self.assertEqual(self.cache.get('answer'), 42) - self.assertEqual(self.cache.get('answer', version=1), 42) - self.assertEqual(self.cache.get('answer', version=2), None) + self.assertEqual(cache.decr_version('answer', version=2), 1) + self.assertEqual(cache.get('answer'), 42) + self.assertEqual(cache.get('answer', version=1), 42) + self.assertEqual(cache.get('answer', version=2), None) - self.v2_cache.set('answer2', 42) - self.assertEqual(self.v2_cache.get('answer2'), 42) - self.assertEqual(self.v2_cache.get('answer2', version=1), None) - self.assertEqual(self.v2_cache.get('answer2', version=2), 42) + caches['v2'].set('answer2', 42) + self.assertEqual(caches['v2'].get('answer2'), 42) + self.assertEqual(caches['v2'].get('answer2', version=1), None) + self.assertEqual(caches['v2'].get('answer2', version=2), 42) - self.assertEqual(self.v2_cache.decr_version('answer2'), 1) - self.assertEqual(self.v2_cache.get('answer2'), None) - self.assertEqual(self.v2_cache.get('answer2', version=1), 42) - self.assertEqual(self.v2_cache.get('answer2', version=2), None) + self.assertEqual(caches['v2'].decr_version('answer2'), 1) + self.assertEqual(caches['v2'].get('answer2'), None) + self.assertEqual(caches['v2'].get('answer2', version=1), 42) + self.assertEqual(caches['v2'].get('answer2', version=2), None) - self.assertRaises(ValueError, self.cache.decr_version, 'does_not_exist', version=2) + self.assertRaises(ValueError, cache.decr_version, 'does_not_exist', version=2) def test_custom_key_func(self): # Two caches with different key functions aren't visible to each other - self.cache.set('answer1', 42) - self.assertEqual(self.cache.get('answer1'), 42) - self.assertEqual(self.custom_key_cache.get('answer1'), None) - self.assertEqual(self.custom_key_cache2.get('answer1'), None) + cache.set('answer1', 42) + self.assertEqual(cache.get('answer1'), 42) + self.assertEqual(caches['custom_key'].get('answer1'), None) + self.assertEqual(caches['custom_key2'].get('answer1'), None) - self.custom_key_cache.set('answer2', 42) - self.assertEqual(self.cache.get('answer2'), None) - self.assertEqual(self.custom_key_cache.get('answer2'), 42) - self.assertEqual(self.custom_key_cache2.get('answer2'), 42) + caches['custom_key'].set('answer2', 42) + self.assertEqual(cache.get('answer2'), None) + self.assertEqual(caches['custom_key'].get('answer2'), 42) + self.assertEqual(caches['custom_key2'].get('answer2'), 42) def test_cache_write_unpickable_object(self): update_middleware = UpdateCacheMiddleware() - update_middleware.cache = self.cache + update_middleware.cache = cache fetch_middleware = FetchFromCacheMiddleware() - fetch_middleware.cache = self.cache + fetch_middleware.cache = cache request = self.factory.get('/cache/test') request._cache_update_cache = True @@ -818,46 +850,36 @@ class BaseCacheTests(object): self.assertEqual(get_cache_data.cookies, response.cookies) -def custom_key_func(key, key_prefix, version): - "A customized cache key function" - return 'CUSTOM-' + '-'.join([key_prefix, str(version), key]) - - -@override_settings( - CACHES={ - 'default': { - 'BACKEND': 'django.core.cache.backends.db.DatabaseCache', - 'LOCATION': 'test cache table', - }, - }, -) +@override_settings(CACHES=caches_setting_for_tests( + BACKEND='django.core.cache.backends.db.DatabaseCache', + # Spaces are used in the table name to ensure quoting/escaping is working + LOCATION='test cache table' +)) class DBCacheTests(BaseCacheTests, TransactionTestCase): available_apps = ['cache'] - backend_name = 'django.core.cache.backends.db.DatabaseCache' def setUp(self): - self.factory = RequestFactory() - # Spaces are used in the table name to ensure quoting/escaping is working - self._table_name = 'test cache table' - management.call_command('createcachetable', verbosity=0, interactive=False) - self.cache = create_cache(self.backend_name, LOCATION=self._table_name, OPTIONS={'MAX_ENTRIES': 30}) - self.prefix_cache = create_cache(self.backend_name, LOCATION=self._table_name, KEY_PREFIX='cacheprefix') - self.v2_cache = create_cache(self.backend_name, LOCATION=self._table_name, VERSION=2) - self.custom_key_cache = create_cache(self.backend_name, LOCATION=self._table_name, KEY_FUNCTION=custom_key_func) - self.custom_key_cache2 = create_cache(self.backend_name, LOCATION=self._table_name, KEY_FUNCTION='cache.tests.custom_key_func') + # The super calls needs to happen first for the settings override. + super(DBCacheTests, self).setUp() + self.create_table() def tearDown(self): - cursor = connection.cursor() - cursor.execute('DROP TABLE %s' % connection.ops.quote_name(self._table_name)) - connection.commit() + # The super call needs to happen first because it uses the database. + super(DBCacheTests, self).tearDown() + self.drop_table() - def test_cull(self): - self.perform_cull_test(50, 29) + def create_table(self): + management.call_command('createcachetable', verbosity=0, interactive=False) + + def drop_table(self): + cursor = connection.cursor() + table_name = connection.ops.quote_name('test cache table') + cursor.execute('DROP TABLE %s' % table_name) + cursor.close() def test_zero_cull(self): - self.cache = create_cache(self.backend_name, LOCATION=self._table_name, OPTIONS={'MAX_ENTRIES': 30, 'CULL_FREQUENCY': 0}) - self.perform_cull_test(50, 18) + self._perform_cull_test(caches['zero_cull'], 50, 18) def test_second_call_doesnt_crash(self): stdout = six.StringIO() @@ -866,30 +888,30 @@ class DBCacheTests(BaseCacheTests, TransactionTestCase): stdout=stdout ) self.assertEqual(stdout.getvalue(), - "Cache table '%s' already exists.\n" % self._table_name) + "Cache table 'test cache table' already exists.\n" * len(settings.CACHES)) def test_createcachetable_with_table_argument(self): """ Delete and recreate cache table with legacy behavior (explicitly specifying the table name). """ - self.tearDown() + self.drop_table() stdout = six.StringIO() management.call_command( 'createcachetable', - self._table_name, + 'test cache table', verbosity=2, stdout=stdout ) self.assertEqual(stdout.getvalue(), - "Cache table '%s' created.\n" % self._table_name) + "Cache table 'test cache table' created.\n") def test_clear_commits_transaction(self): # Ensure the database transaction is committed (#19896) - self.cache.set("key1", "spam") - self.cache.clear() + cache.set("key1", "spam") + cache.clear() transaction.rollback() - self.assertEqual(self.cache.get("key1"), None) + self.assertEqual(cache.get("key1"), None) @override_settings(USE_TZ=True) @@ -946,86 +968,59 @@ class CreateCacheTableForDBCacheTests(TestCase): router.routers = old_routers -class LocMemCacheTests(unittest.TestCase, BaseCacheTests): - backend_name = 'django.core.cache.backends.locmem.LocMemCache' +@override_settings(CACHES=caches_setting_for_tests( + BACKEND='django.core.cache.backends.locmem.LocMemCache', +)) +class LocMemCacheTests(BaseCacheTests, TestCase): def setUp(self): - self.factory = RequestFactory() - self.cache = create_cache(self.backend_name, OPTIONS={'MAX_ENTRIES': 30}) - self.prefix_cache = create_cache(self.backend_name, KEY_PREFIX='cacheprefix') - self.v2_cache = create_cache(self.backend_name, VERSION=2) - self.custom_key_cache = create_cache(self.backend_name, OPTIONS={'MAX_ENTRIES': 30}, KEY_FUNCTION=custom_key_func) - self.custom_key_cache2 = create_cache(self.backend_name, OPTIONS={'MAX_ENTRIES': 30}, KEY_FUNCTION='cache.tests.custom_key_func') + super(LocMemCacheTests, self).setUp() # LocMem requires a hack to make the other caches # share a data store with the 'normal' cache. - self.prefix_cache._cache = self.cache._cache - self.prefix_cache._expire_info = self.cache._expire_info + caches['prefix']._cache = cache._cache + caches['prefix']._expire_info = cache._expire_info - self.v2_cache._cache = self.cache._cache - self.v2_cache._expire_info = self.cache._expire_info + caches['v2']._cache = cache._cache + caches['v2']._expire_info = cache._expire_info - self.custom_key_cache._cache = self.cache._cache - self.custom_key_cache._expire_info = self.cache._expire_info + caches['custom_key']._cache = cache._cache + caches['custom_key']._expire_info = cache._expire_info - self.custom_key_cache2._cache = self.cache._cache - self.custom_key_cache2._expire_info = self.cache._expire_info - - def tearDown(self): - self.cache.clear() - - def test_cull(self): - self.perform_cull_test(50, 29) - - def test_zero_cull(self): - self.cache = create_cache(self.backend_name, OPTIONS={'MAX_ENTRIES': 30, 'CULL_FREQUENCY': 0}) - self.perform_cull_test(50, 19) + caches['custom_key2']._cache = cache._cache + caches['custom_key2']._expire_info = cache._expire_info def test_multiple_caches(self): "Check that multiple locmem caches are isolated" - mirror_cache = create_cache(self.backend_name) - other_cache = create_cache(self.backend_name, LOCATION='other') - - self.cache.set('value1', 42) - self.assertEqual(mirror_cache.get('value1'), 42) - self.assertEqual(other_cache.get('value1'), None) + cache.set('value', 42) + self.assertEqual(caches['default'].get('value'), 42) + self.assertEqual(caches['other'].get('value'), None) def test_incr_decr_timeout(self): """incr/decr does not modify expiry time (matches memcached behavior)""" key = 'value' - _key = self.cache.make_key(key) - self.cache.set(key, 1, timeout=self.cache.default_timeout * 10) - expire = self.cache._expire_info[_key] - self.cache.incr(key) - self.assertEqual(expire, self.cache._expire_info[_key]) - self.cache.decr(key) - self.assertEqual(expire, self.cache._expire_info[_key]) + _key = cache.make_key(key) + cache.set(key, 1, timeout=cache.default_timeout * 10) + expire = cache._expire_info[_key] + cache.incr(key) + self.assertEqual(expire, cache._expire_info[_key]) + cache.decr(key) + self.assertEqual(expire, cache._expire_info[_key]) # memcached backend isn't guaranteed to be available. # To check the memcached backend, the test settings file will # need to contain at least one cache backend setting that points at # your memcache server. -@unittest.skipUnless( - any(cache['BACKEND'].startswith('django.core.cache.backends.memcached.') - for cache in settings.CACHES.values()), - "memcached not available") -class MemcachedCacheTests(unittest.TestCase, BaseCacheTests): +memcached_params = None +for _cache_params in settings.CACHES.values(): + if _cache_params['BACKEND'].startswith('django.core.cache.backends.memcached.'): + memcached_params = _cache_params - def setUp(self): - self.factory = RequestFactory() - for cache_key, cache in settings.CACHES.items(): - if cache['BACKEND'].startswith('django.core.cache.backends.memcached.'): - break - random_prefix = ''.join(random.choice(string.ascii_letters) for x in range(10)) - self.cache = caches[cache_key] - self.prefix_cache = create_cache(cache_key, KEY_PREFIX=random_prefix) - self.v2_cache = create_cache(cache_key, VERSION=2) - self.custom_key_cache = create_cache(cache_key, KEY_FUNCTION=custom_key_func) - self.custom_key_cache2 = create_cache(cache_key, KEY_FUNCTION='cache.tests.custom_key_func') - def tearDown(self): - self.cache.clear() +@unittest.skipIf(memcached_params is None, "memcached not available") +@override_settings(CACHES=caches_setting_for_tests(**memcached_params)) +class MemcachedCacheTests(BaseCacheTests, TestCase): def test_invalid_keys(self): """ @@ -1038,9 +1033,9 @@ class MemcachedCacheTests(unittest.TestCase, BaseCacheTests): """ # memcached does not allow whitespace or control characters in keys - self.assertRaises(Exception, self.cache.set, 'key with spaces', 'value') + self.assertRaises(Exception, cache.set, 'key with spaces', 'value') # memcached limits key length to 250 - self.assertRaises(Exception, self.cache.set, 'a' * 251, 'value') + self.assertRaises(Exception, cache.set, 'a' * 251, 'value') # Explicitly display a skipped test if no configured cache uses MemcachedCache @unittest.skipUnless( @@ -1054,55 +1049,59 @@ class MemcachedCacheTests(unittest.TestCase, BaseCacheTests): self.assertEqual(caches[cache_key]._cache.pickleProtocol, pickle.HIGHEST_PROTOCOL) + def test_cull(self): + # culling isn't implemented, memcached deals with it. + pass -class FileBasedCacheTests(unittest.TestCase, BaseCacheTests): + def test_zero_cull(self): + # culling isn't implemented, memcached deals with it. + pass + + +@override_settings(CACHES=caches_setting_for_tests( + BACKEND='django.core.cache.backends.filebased.FileBasedCache', +)) +class FileBasedCacheTests(BaseCacheTests, TestCase): """ Specific test cases for the file-based cache. """ - backend_name = 'django.core.cache.backends.filebased.FileBasedCache' def setUp(self): - self.factory = RequestFactory() + super(FileBasedCacheTests, self).setUp() self.dirname = tempfile.mkdtemp() - self.cache = create_cache(self.backend_name, LOCATION=self.dirname, OPTIONS={'MAX_ENTRIES': 30}) - self.prefix_cache = create_cache(self.backend_name, LOCATION=self.dirname, KEY_PREFIX='cacheprefix') - self.v2_cache = create_cache(self.backend_name, LOCATION=self.dirname, VERSION=2) - self.custom_key_cache = create_cache(self.backend_name, LOCATION=self.dirname, KEY_FUNCTION=custom_key_func) - self.custom_key_cache2 = create_cache(self.backend_name, LOCATION=self.dirname, KEY_FUNCTION='cache.tests.custom_key_func') + for cache_params in settings.CACHES.values(): + cache_params.update({'LOCATION': self.dirname}) def tearDown(self): - self.cache.clear() - os.rmdir(self.dirname) - - def test_cull(self): - self.perform_cull_test(50, 29) + shutil.rmtree(self.dirname) + super(FileBasedCacheTests, self).tearDown() def test_ignores_non_cache_files(self): fname = os.path.join(self.dirname, 'not-a-cache-file') with open(fname, 'w'): os.utime(fname, None) - self.cache.clear() + cache.clear() self.assertTrue(os.path.exists(fname), 'Expected cache.clear to ignore non cache files') os.remove(fname) def test_clear_does_not_remove_cache_dir(self): - self.cache.clear() + cache.clear() self.assertTrue(os.path.exists(self.dirname), 'Expected cache.clear to keep the cache dir') def test_creates_cache_dir_if_nonexistent(self): os.rmdir(self.dirname) - self.cache.set('foo', 'bar') + cache.set('foo', 'bar') os.path.exists(self.dirname) - def test_zero_cull(self): - # Regression test for #15806 - self.cache = create_cache(self.backend_name, LOCATION=self.dirname, OPTIONS={'MAX_ENTRIES': 30, 'CULL_FREQUENCY': 0}) - self.perform_cull_test(50, 19) - -class CustomCacheKeyValidationTests(unittest.TestCase): +@override_settings(CACHES={ + 'default': { + 'BACKEND': 'cache.liberal_backend.CacheClass', + }, +}) +class CustomCacheKeyValidationTests(TestCase): """ Tests for the ability to mixin a custom ``validate_key`` method to a custom cache backend that otherwise inherits from a builtin @@ -1110,8 +1109,6 @@ class CustomCacheKeyValidationTests(unittest.TestCase): """ def test_custom_key_validation(self): - cache = create_cache('cache.liberal_backend.CacheClass') - # this key is both longer than 250 characters, and has spaces key = 'some key with spaces' * 15 val = 'a value' @@ -1119,26 +1116,27 @@ class CustomCacheKeyValidationTests(unittest.TestCase): self.assertEqual(cache.get(key), val) -class GetCacheTests(unittest.TestCase): +class GetCacheTests(IgnorePendingDeprecationWarningsMixin, TestCase): def test_simple(self): - from django.core.cache import caches, DEFAULT_CACHE_ALIAS + from django.core.cache import caches, DEFAULT_CACHE_ALIAS, get_cache self.assertIsInstance( caches[DEFAULT_CACHE_ALIAS], - create_cache('default').__class__ + get_cache('default').__class__ ) - cache = create_cache( + cache = get_cache( 'django.core.cache.backends.dummy.DummyCache', **{'TIMEOUT': 120} ) self.assertEqual(cache.default_timeout, 120) - self.assertRaises(InvalidCacheBackendError, create_cache, 'does_not_exist') + self.assertRaises(InvalidCacheBackendError, get_cache, 'does_not_exist') def test_close(self): + from django.core.cache import get_cache from django.core import signals - cache = create_cache('cache.closeable_cache.CacheClass') + cache = get_cache('cache.closeable_cache.CacheClass') self.assertFalse(cache.closed) signals.request_finished.send(self.__class__) self.assertTrue(cache.closed) @@ -1159,11 +1157,8 @@ class CacheUtils(TestCase): def setUp(self): self.path = '/cache/test/' - self.cache = caches['default'] self.factory = RequestFactory() - def tearDown(self): - self.cache.clear() def test_patch_vary_headers(self): headers = ( @@ -1267,11 +1262,10 @@ class CacheHEADTest(TestCase): def setUp(self): self.path = '/cache/test/' - self.cache = caches['default'] self.factory = RequestFactory() def tearDown(self): - self.cache.clear() + cache.clear() def _set_cache(self, request, msg): response = HttpResponse() @@ -1320,11 +1314,10 @@ class CacheI18nTest(TestCase): def setUp(self): self.path = '/cache/test/' - self.cache = create_cache('default') self.factory = RequestFactory() def tearDown(self): - self.cache.clear() + cache.clear() @override_settings(USE_I18N=True, USE_L10N=False, USE_TZ=False) def test_cache_key_i18n_translation(self): @@ -1587,8 +1580,8 @@ class CacheMiddlewareTest(IgnoreDeprecationWarningsMixin, TestCase): def setUp(self): super(CacheMiddlewareTest, self).setUp() self.factory = RequestFactory() - self.default_cache = create_cache('default') - self.other_cache = create_cache('other') + self.default_cache = caches['default'] + self.other_cache = caches['other'] def tearDown(self): self.default_cache.clear() @@ -1807,11 +1800,10 @@ class TestWithTemplateResponse(TestCase): """ def setUp(self): self.path = '/cache/test/' - self.cache = create_cache('default') self.factory = RequestFactory() def tearDown(self): - self.cache.clear() + cache.clear() def test_patch_vary_headers(self): headers = (