[2.2.x] Fixed CVE-2020-13254 -- Enforced cache key validation in memcached backends.
This commit is contained in:
parent
6d61860b22
commit
07e59caa02
|
@ -17,13 +17,13 @@ from threading import local
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.core import signals
|
from django.core import signals
|
||||||
from django.core.cache.backends.base import (
|
from django.core.cache.backends.base import (
|
||||||
BaseCache, CacheKeyWarning, InvalidCacheBackendError,
|
BaseCache, CacheKeyWarning, InvalidCacheBackendError, InvalidCacheKey,
|
||||||
)
|
)
|
||||||
from django.utils.module_loading import import_string
|
from django.utils.module_loading import import_string
|
||||||
|
|
||||||
__all__ = [
|
__all__ = [
|
||||||
'cache', 'caches', 'DEFAULT_CACHE_ALIAS', 'InvalidCacheBackendError',
|
'cache', 'caches', 'DEFAULT_CACHE_ALIAS', 'InvalidCacheBackendError',
|
||||||
'CacheKeyWarning', 'BaseCache',
|
'CacheKeyWarning', 'BaseCache', 'InvalidCacheKey',
|
||||||
]
|
]
|
||||||
|
|
||||||
DEFAULT_CACHE_ALIAS = 'default'
|
DEFAULT_CACHE_ALIAS = 'default'
|
||||||
|
|
|
@ -14,6 +14,10 @@ class CacheKeyWarning(RuntimeWarning):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class InvalidCacheKey(ValueError):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
# Stub class to ensure not passing in a `timeout` argument results in
|
# Stub class to ensure not passing in a `timeout` argument results in
|
||||||
# the default timeout
|
# the default timeout
|
||||||
DEFAULT_TIMEOUT = object()
|
DEFAULT_TIMEOUT = object()
|
||||||
|
@ -242,18 +246,8 @@ class BaseCache:
|
||||||
backend. This encourages (but does not force) writing backend-portable
|
backend. This encourages (but does not force) writing backend-portable
|
||||||
cache code.
|
cache code.
|
||||||
"""
|
"""
|
||||||
if len(key) > MEMCACHE_MAX_KEY_LENGTH:
|
for warning in memcache_key_warnings(key):
|
||||||
warnings.warn(
|
warnings.warn(warning, CacheKeyWarning)
|
||||||
'Cache key will cause errors if used with memcached: %r '
|
|
||||||
'(longer than %s)' % (key, MEMCACHE_MAX_KEY_LENGTH), CacheKeyWarning
|
|
||||||
)
|
|
||||||
for char in key:
|
|
||||||
if ord(char) < 33 or ord(char) == 127:
|
|
||||||
warnings.warn(
|
|
||||||
'Cache key contains characters that will cause errors if '
|
|
||||||
'used with memcached: %r' % key, CacheKeyWarning
|
|
||||||
)
|
|
||||||
break
|
|
||||||
|
|
||||||
def incr_version(self, key, delta=1, version=None):
|
def incr_version(self, key, delta=1, version=None):
|
||||||
"""
|
"""
|
||||||
|
@ -281,3 +275,18 @@ class BaseCache:
|
||||||
def close(self, **kwargs):
|
def close(self, **kwargs):
|
||||||
"""Close the cache connection"""
|
"""Close the cache connection"""
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
def memcache_key_warnings(key):
|
||||||
|
if len(key) > MEMCACHE_MAX_KEY_LENGTH:
|
||||||
|
yield (
|
||||||
|
'Cache key will cause errors if used with memcached: %r '
|
||||||
|
'(longer than %s)' % (key, MEMCACHE_MAX_KEY_LENGTH)
|
||||||
|
)
|
||||||
|
for char in key:
|
||||||
|
if ord(char) < 33 or ord(char) == 127:
|
||||||
|
yield (
|
||||||
|
'Cache key contains characters that will cause errors if '
|
||||||
|
'used with memcached: %r' % key, CacheKeyWarning
|
||||||
|
)
|
||||||
|
break
|
||||||
|
|
|
@ -4,7 +4,9 @@ import pickle
|
||||||
import re
|
import re
|
||||||
import time
|
import time
|
||||||
|
|
||||||
from django.core.cache.backends.base import DEFAULT_TIMEOUT, BaseCache
|
from django.core.cache.backends.base import (
|
||||||
|
DEFAULT_TIMEOUT, BaseCache, InvalidCacheKey, memcache_key_warnings,
|
||||||
|
)
|
||||||
from django.utils.functional import cached_property
|
from django.utils.functional import cached_property
|
||||||
|
|
||||||
|
|
||||||
|
@ -64,10 +66,12 @@ class BaseMemcachedCache(BaseCache):
|
||||||
|
|
||||||
def add(self, key, value, timeout=DEFAULT_TIMEOUT, version=None):
|
def add(self, key, value, timeout=DEFAULT_TIMEOUT, version=None):
|
||||||
key = self.make_key(key, version=version)
|
key = self.make_key(key, version=version)
|
||||||
|
self.validate_key(key)
|
||||||
return self._cache.add(key, value, self.get_backend_timeout(timeout))
|
return self._cache.add(key, value, self.get_backend_timeout(timeout))
|
||||||
|
|
||||||
def get(self, key, default=None, version=None):
|
def get(self, key, default=None, version=None):
|
||||||
key = self.make_key(key, version=version)
|
key = self.make_key(key, version=version)
|
||||||
|
self.validate_key(key)
|
||||||
val = self._cache.get(key)
|
val = self._cache.get(key)
|
||||||
if val is None:
|
if val is None:
|
||||||
return default
|
return default
|
||||||
|
@ -75,16 +79,20 @@ class BaseMemcachedCache(BaseCache):
|
||||||
|
|
||||||
def set(self, key, value, timeout=DEFAULT_TIMEOUT, version=None):
|
def set(self, key, value, timeout=DEFAULT_TIMEOUT, version=None):
|
||||||
key = self.make_key(key, version=version)
|
key = self.make_key(key, version=version)
|
||||||
|
self.validate_key(key)
|
||||||
if not self._cache.set(key, value, self.get_backend_timeout(timeout)):
|
if not self._cache.set(key, value, self.get_backend_timeout(timeout)):
|
||||||
# make sure the key doesn't keep its old value in case of failure to set (memcached's 1MB limit)
|
# make sure the key doesn't keep its old value in case of failure to set (memcached's 1MB limit)
|
||||||
self._cache.delete(key)
|
self._cache.delete(key)
|
||||||
|
|
||||||
def delete(self, key, version=None):
|
def delete(self, key, version=None):
|
||||||
key = self.make_key(key, version=version)
|
key = self.make_key(key, version=version)
|
||||||
|
self.validate_key(key)
|
||||||
self._cache.delete(key)
|
self._cache.delete(key)
|
||||||
|
|
||||||
def get_many(self, keys, version=None):
|
def get_many(self, keys, version=None):
|
||||||
key_map = {self.make_key(key, version=version): key for key in keys}
|
key_map = {self.make_key(key, version=version): key for key in keys}
|
||||||
|
for key in key_map:
|
||||||
|
self.validate_key(key)
|
||||||
ret = self._cache.get_multi(key_map.keys())
|
ret = self._cache.get_multi(key_map.keys())
|
||||||
return {key_map[k]: v for k, v in ret.items()}
|
return {key_map[k]: v for k, v in ret.items()}
|
||||||
|
|
||||||
|
@ -94,6 +102,7 @@ class BaseMemcachedCache(BaseCache):
|
||||||
|
|
||||||
def incr(self, key, delta=1, version=None):
|
def incr(self, key, delta=1, version=None):
|
||||||
key = self.make_key(key, version=version)
|
key = self.make_key(key, version=version)
|
||||||
|
self.validate_key(key)
|
||||||
# memcached doesn't support a negative delta
|
# memcached doesn't support a negative delta
|
||||||
if delta < 0:
|
if delta < 0:
|
||||||
return self._cache.decr(key, -delta)
|
return self._cache.decr(key, -delta)
|
||||||
|
@ -112,6 +121,7 @@ class BaseMemcachedCache(BaseCache):
|
||||||
|
|
||||||
def decr(self, key, delta=1, version=None):
|
def decr(self, key, delta=1, version=None):
|
||||||
key = self.make_key(key, version=version)
|
key = self.make_key(key, version=version)
|
||||||
|
self.validate_key(key)
|
||||||
# memcached doesn't support a negative delta
|
# memcached doesn't support a negative delta
|
||||||
if delta < 0:
|
if delta < 0:
|
||||||
return self._cache.incr(key, -delta)
|
return self._cache.incr(key, -delta)
|
||||||
|
@ -133,6 +143,7 @@ class BaseMemcachedCache(BaseCache):
|
||||||
original_keys = {}
|
original_keys = {}
|
||||||
for key, value in data.items():
|
for key, value in data.items():
|
||||||
safe_key = self.make_key(key, version=version)
|
safe_key = self.make_key(key, version=version)
|
||||||
|
self.validate_key(safe_key)
|
||||||
safe_data[safe_key] = value
|
safe_data[safe_key] = value
|
||||||
original_keys[safe_key] = key
|
original_keys[safe_key] = key
|
||||||
failed_keys = self._cache.set_multi(safe_data, self.get_backend_timeout(timeout))
|
failed_keys = self._cache.set_multi(safe_data, self.get_backend_timeout(timeout))
|
||||||
|
@ -144,6 +155,10 @@ class BaseMemcachedCache(BaseCache):
|
||||||
def clear(self):
|
def clear(self):
|
||||||
self._cache.flush_all()
|
self._cache.flush_all()
|
||||||
|
|
||||||
|
def validate_key(self, key):
|
||||||
|
for warning in memcache_key_warnings(key):
|
||||||
|
raise InvalidCacheKey(warning)
|
||||||
|
|
||||||
|
|
||||||
class MemcachedCache(BaseMemcachedCache):
|
class MemcachedCache(BaseMemcachedCache):
|
||||||
"An implementation of a cache binding using python-memcached"
|
"An implementation of a cache binding using python-memcached"
|
||||||
|
|
|
@ -6,6 +6,14 @@ Django 2.2.13 release notes
|
||||||
|
|
||||||
Django 2.2.13 fixes two security issues and a regression in 2.2.12.
|
Django 2.2.13 fixes two security issues and a regression in 2.2.12.
|
||||||
|
|
||||||
|
CVE-2020-13254: Potential data leakage via malformed memcached keys
|
||||||
|
===================================================================
|
||||||
|
|
||||||
|
In cases where a memcached backend does not perform key validation, passing
|
||||||
|
malformed cache keys could result in a key collision, and potential data
|
||||||
|
leakage. In order to avoid this vulnerability, key validation is added to the
|
||||||
|
memcached cache backends.
|
||||||
|
|
||||||
CVE-2020-13596: Possible XSS via admin ``ForeignKeyRawIdWidget``
|
CVE-2020-13596: Possible XSS via admin ``ForeignKeyRawIdWidget``
|
||||||
================================================================
|
================================================================
|
||||||
|
|
||||||
|
|
|
@ -15,7 +15,7 @@ from unittest import mock
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.core import management, signals
|
from django.core import management, signals
|
||||||
from django.core.cache import (
|
from django.core.cache import (
|
||||||
DEFAULT_CACHE_ALIAS, CacheKeyWarning, cache, caches,
|
DEFAULT_CACHE_ALIAS, CacheKeyWarning, InvalidCacheKey, cache, caches,
|
||||||
)
|
)
|
||||||
from django.core.cache.utils import make_template_fragment_key
|
from django.core.cache.utils import make_template_fragment_key
|
||||||
from django.db import close_old_connections, connection, connections
|
from django.db import close_old_connections, connection, connections
|
||||||
|
@ -605,10 +605,10 @@ class BaseCacheTests:
|
||||||
|
|
||||||
def _perform_invalid_key_test(self, key, expected_warning):
|
def _perform_invalid_key_test(self, key, expected_warning):
|
||||||
"""
|
"""
|
||||||
All the builtin backends (except memcached, see below) should warn on
|
All the builtin backends should warn (except memcached that should
|
||||||
keys that would be refused by memcached. This encourages portable
|
error) on keys that would be refused by memcached. This encourages
|
||||||
caching code without making it too difficult to use production backends
|
portable caching code without making it too difficult to use production
|
||||||
with more liberal key rules. Refs #6447.
|
backends with more liberal key rules. Refs #6447.
|
||||||
"""
|
"""
|
||||||
# mimic custom ``make_key`` method being defined since the default will
|
# mimic custom ``make_key`` method being defined since the default will
|
||||||
# never show the below warnings
|
# never show the below warnings
|
||||||
|
@ -1251,24 +1251,14 @@ class BaseMemcachedTests(BaseCacheTests):
|
||||||
with self.settings(CACHES={'default': params}):
|
with self.settings(CACHES={'default': params}):
|
||||||
self.assertEqual(cache._servers, ['server1.tld', 'server2:11211'])
|
self.assertEqual(cache._servers, ['server1.tld', 'server2:11211'])
|
||||||
|
|
||||||
def test_invalid_key_characters(self):
|
def _perform_invalid_key_test(self, key, expected_warning):
|
||||||
"""
|
"""
|
||||||
On memcached, we don't introduce a duplicate key validation
|
Whilst other backends merely warn, memcached should raise for an
|
||||||
step (for speed reasons), we just let the memcached API
|
invalid key.
|
||||||
library raise its own exception on bad keys. Refs #6447.
|
|
||||||
|
|
||||||
In order to be memcached-API-library agnostic, we only assert
|
|
||||||
that a generic exception of some kind is raised.
|
|
||||||
"""
|
"""
|
||||||
# memcached does not allow whitespace or control characters in keys
|
msg = expected_warning.replace(key, ':1:%s' % key)
|
||||||
# when using the ascii protocol.
|
with self.assertRaisesMessage(InvalidCacheKey, msg):
|
||||||
with self.assertRaises(Exception):
|
cache.set(key, 'value')
|
||||||
cache.set('key with spaces', 'value')
|
|
||||||
|
|
||||||
def test_invalid_key_length(self):
|
|
||||||
# memcached limits key length to 250
|
|
||||||
with self.assertRaises(Exception):
|
|
||||||
cache.set('a' * 251, 'value')
|
|
||||||
|
|
||||||
def test_default_never_expiring_timeout(self):
|
def test_default_never_expiring_timeout(self):
|
||||||
# Regression test for #22845
|
# Regression test for #22845
|
||||||
|
@ -1377,15 +1367,6 @@ class PyLibMCCacheTests(BaseMemcachedTests, TestCase):
|
||||||
# libmemcached manages its own connections.
|
# libmemcached manages its own connections.
|
||||||
should_disconnect_on_close = False
|
should_disconnect_on_close = False
|
||||||
|
|
||||||
# By default, pylibmc/libmemcached don't verify keys client-side and so
|
|
||||||
# this test triggers a server-side bug that causes later tests to fail
|
|
||||||
# (#19914). The `verify_keys` behavior option could be set to True (which
|
|
||||||
# would avoid triggering the server-side bug), however this test would
|
|
||||||
# still fail due to https://github.com/lericson/pylibmc/issues/219.
|
|
||||||
@unittest.skip("triggers a memcached-server bug, causing subsequent tests to fail")
|
|
||||||
def test_invalid_key_characters(self):
|
|
||||||
pass
|
|
||||||
|
|
||||||
@override_settings(CACHES=caches_setting_for_tests(
|
@override_settings(CACHES=caches_setting_for_tests(
|
||||||
base=PyLibMCCache_params,
|
base=PyLibMCCache_params,
|
||||||
exclude=memcached_excluded_caches,
|
exclude=memcached_excluded_caches,
|
||||||
|
|
Loading…
Reference in New Issue