dephne compatibility of LocalCacheProxy
This commit is contained in:
parent
80cf20d8f2
commit
2b3bb581bc
1 changed files with 16 additions and 11 deletions
27
src/c3nav/mapdata/utils/cache/local.py
vendored
27
src/c3nav/mapdata/utils/cache/local.py
vendored
|
@ -1,4 +1,5 @@
|
||||||
from collections import OrderedDict
|
from collections import OrderedDict
|
||||||
|
from contextvars import ContextVar
|
||||||
|
|
||||||
from django.core.cache import cache
|
from django.core.cache import cache
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
|
@ -10,42 +11,46 @@ class NoneFromCache:
|
||||||
|
|
||||||
class LocalCacheProxy:
|
class LocalCacheProxy:
|
||||||
# django cache, buffered using a LRU cache
|
# django cache, buffered using a LRU cache
|
||||||
# only usable for stuff that never changes, obviously
|
# only usable for stuff that never needs to know about changes made by other cache clients, obviously
|
||||||
# todo: ensure thread-safety, compatible with async + daphne etc
|
|
||||||
def __init__(self, maxsize=128):
|
def __init__(self, maxsize=128):
|
||||||
self._maxsize = maxsize
|
self._maxsize = maxsize
|
||||||
self._mapupdate = None
|
self._mapupdate = None
|
||||||
self._items = OrderedDict()
|
self._items: ContextVar[OrderedDict] = ContextVar("cache items")
|
||||||
|
# do not use a default of a dict, this can lead to same instance in different contexts
|
||||||
|
# we don't particularly care about this for LocalCacheProxy,
|
||||||
|
# but we DEFINITELY care about this for the local request cache.
|
||||||
|
# Most importantly, this is why the clear function always sets a new dictionary to be extra sure.
|
||||||
|
self.clear()
|
||||||
|
|
||||||
def get(self, key, default=None):
|
def get(self, key, default=None):
|
||||||
if self._mapupdate is None:
|
if self._mapupdate is None:
|
||||||
self._check_mapupdate()
|
self._check_mapupdate()
|
||||||
try:
|
try:
|
||||||
# first check out cache
|
# first check out cache
|
||||||
result = self._items[key]
|
result = self._items.get()[key]
|
||||||
except KeyError:
|
except KeyError:
|
||||||
# not in our cache
|
# not in our cache
|
||||||
result = cache.get(key, default=NoneFromCache)
|
result = cache.get(key, default=NoneFromCache)
|
||||||
if result is not NoneFromCache:
|
if result is not NoneFromCache:
|
||||||
self._items[key] = result
|
self._items.get()[key] = result
|
||||||
self._prune()
|
self._prune()
|
||||||
else:
|
else:
|
||||||
result = default
|
result = default
|
||||||
else:
|
else:
|
||||||
self._items.move_to_end(key, last=True)
|
self._items.get().move_to_end(key, last=True)
|
||||||
return result
|
return result
|
||||||
|
|
||||||
def _prune(self):
|
def _prune(self):
|
||||||
# remove old items
|
# remove old items
|
||||||
while len(self._items) > self._maxsize:
|
while len(self._items.get()) > self._maxsize:
|
||||||
self._items.pop(next(iter(self._items.keys())))
|
self._items.get().pop(next(iter(self._items.get().keys())))
|
||||||
|
|
||||||
def _check_mapupdate(self):
|
def _check_mapupdate(self):
|
||||||
# todo: thanks to enable_globally() we shouldn't need this any more
|
# todo: thanks to enable_globally() we shouldn't need this any more
|
||||||
from c3nav.mapdata.models import MapUpdate
|
from c3nav.mapdata.models import MapUpdate
|
||||||
mapupdate = MapUpdate.current_cache_key()
|
mapupdate = MapUpdate.current_cache_key()
|
||||||
if self._mapupdate != mapupdate:
|
if self._mapupdate != mapupdate:
|
||||||
self._items = OrderedDict()
|
self.clear()
|
||||||
self._mapupdate = mapupdate
|
self._mapupdate = mapupdate
|
||||||
|
|
||||||
enabled = False
|
enabled = False
|
||||||
|
@ -61,11 +66,11 @@ class LocalCacheProxy:
|
||||||
self._check_mapupdate()
|
self._check_mapupdate()
|
||||||
cache.set(key, value, expire)
|
cache.set(key, value, expire)
|
||||||
if LocalCacheProxy.enabled:
|
if LocalCacheProxy.enabled:
|
||||||
self._items[key] = value
|
self._items.get()[key] = value
|
||||||
self._prune()
|
self._prune()
|
||||||
|
|
||||||
def clear(self):
|
def clear(self):
|
||||||
self._items.clear()
|
self._items.set(OrderedDict())
|
||||||
|
|
||||||
|
|
||||||
class RequestLocalCacheProxy(LocalCacheProxy):
|
class RequestLocalCacheProxy(LocalCacheProxy):
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue