mirror of
				https://github.com/django/django.git
				synced 2025-10-25 14:46:09 +00:00 
			
		
		
		
	Fixed #16358 - Made memcache backend delete old value on a failure to set.
Default Memcached configuration allows for a maximum object of 1MB and will fail to set the key if it is too large. The key will be deleted from memcached if it fails to be set. This is needed to avoid an issue with cache_db session backend using the old value stored in memcached, instead of the newer value stored in the database.
This commit is contained in:
		
							
								
								
									
										4
									
								
								django/core/cache/backends/memcached.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								django/core/cache/backends/memcached.py
									
									
									
									
										vendored
									
									
								
							| @@ -86,7 +86,9 @@ class BaseMemcachedCache(six.with_metaclass(BaseMemcachedCacheMethods, 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._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) | ||||||
|  |             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) | ||||||
|   | |||||||
| @@ -729,6 +729,10 @@ Miscellaneous | |||||||
|  |  | ||||||
|   .. _universal newlines: http://www.python.org/dev/peps/pep-0278 |   .. _universal newlines: http://www.python.org/dev/peps/pep-0278 | ||||||
|  |  | ||||||
|  | * The Memcached cache backends ``MemcachedCache`` and ``PyLibMCCache`` will | ||||||
|  |   delete a key if ``set()`` fails. This is necessary to ensure the ``cache_db`` | ||||||
|  |   session store always fetches the most current session data. | ||||||
|  |  | ||||||
| .. _deprecated-features-1.8: | .. _deprecated-features-1.8: | ||||||
|  |  | ||||||
| Features deprecated in 1.8 | Features deprecated in 1.8 | ||||||
|   | |||||||
							
								
								
									
										18
									
								
								tests/cache/tests.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										18
									
								
								tests/cache/tests.py
									
									
									
									
										vendored
									
									
								
							| @@ -1133,6 +1133,24 @@ class MemcachedCacheTests(BaseCacheTests, TestCase): | |||||||
|         # culling isn't implemented, memcached deals with it. |         # culling isn't implemented, memcached deals with it. | ||||||
|         pass |         pass | ||||||
|  |  | ||||||
|  |     def test_memcached_deletes_key_on_failed_set(self): | ||||||
|  |         # By default memcached allows objects up to 1MB. For the cache_db session | ||||||
|  |         # backend to always use the current session, memcached needs to delete | ||||||
|  |         # the old key if it fails to set. | ||||||
|  |         # pylibmc doesn't seem to have SERVER_MAX_VALUE_LENGTH as far as I can | ||||||
|  |         # tell from a quick check of its source code. This is falling back to | ||||||
|  |         # the default value exposed by python-memcached on my system. | ||||||
|  |         max_value_length = getattr(cache._lib, 'SERVER_MAX_VALUE_LENGTH', 1048576) | ||||||
|  |  | ||||||
|  |         cache.set('small_value', 'a') | ||||||
|  |         self.assertEqual(cache.get('small_value'), 'a') | ||||||
|  |  | ||||||
|  |         large_value = 'a' * (max_value_length + 1) | ||||||
|  |         cache.set('small_value', large_value) | ||||||
|  |         # small_value should be deleted, or set if configured to accept larger values | ||||||
|  |         value = cache.get('small_value') | ||||||
|  |         self.assertTrue(value is None or value == large_value) | ||||||
|  |  | ||||||
|  |  | ||||||
| @override_settings(CACHES=caches_setting_for_tests( | @override_settings(CACHES=caches_setting_for_tests( | ||||||
|     BACKEND='django.core.cache.backends.filebased.FileBasedCache', |     BACKEND='django.core.cache.backends.filebased.FileBasedCache', | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user