2018-12-27 17:24:53 +01:00
|
|
|
from collections import OrderedDict
|
|
|
|
|
|
|
|
from django.core.cache import cache
|
2024-12-27 18:50:36 +01:00
|
|
|
from django.conf import settings
|
2018-12-27 22:02:07 +01:00
|
|
|
|
2018-12-27 17:24:53 +01:00
|
|
|
|
|
|
|
class NoneFromCache:
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class LocalCacheProxy:
|
|
|
|
# django cache, buffered using a LRU cache
|
|
|
|
# only usable for stuff that never changes, obviously
|
2023-12-07 17:28:13 +01:00
|
|
|
# todo: ensure thread-safety, compatible with async + daphne etc
|
2018-12-27 17:24:53 +01:00
|
|
|
def __init__(self, maxsize=128):
|
|
|
|
self._maxsize = maxsize
|
2018-12-27 22:02:07 +01:00
|
|
|
self._mapupdate = None
|
2018-12-27 17:24:53 +01:00
|
|
|
self._items = OrderedDict()
|
|
|
|
|
|
|
|
def get(self, key, default=None):
|
2018-12-28 17:29:49 +01:00
|
|
|
if self._mapupdate is None:
|
|
|
|
self._check_mapupdate()
|
2018-12-27 17:24:53 +01:00
|
|
|
try:
|
|
|
|
# first check out cache
|
|
|
|
result = self._items[key]
|
|
|
|
except KeyError:
|
|
|
|
# not in our cache
|
|
|
|
result = cache.get(key, default=NoneFromCache)
|
|
|
|
if result is not NoneFromCache:
|
|
|
|
self._items[key] = result
|
|
|
|
self._prune()
|
|
|
|
else:
|
|
|
|
result = default
|
|
|
|
else:
|
|
|
|
self._items.move_to_end(key, last=True)
|
|
|
|
return result
|
|
|
|
|
|
|
|
def _prune(self):
|
|
|
|
# remove old items
|
|
|
|
while len(self._items) > self._maxsize:
|
|
|
|
self._items.pop(next(iter(self._items.keys())))
|
|
|
|
|
2018-12-27 22:02:07 +01:00
|
|
|
def _check_mapupdate(self):
|
2025-04-17 22:25:23 +02:00
|
|
|
# todo: thanks to enable_globally() we shouldn't need this any more
|
2024-12-27 18:50:36 +01:00
|
|
|
from c3nav.mapdata.models import MapUpdate
|
2018-12-27 22:02:07 +01:00
|
|
|
mapupdate = MapUpdate.current_cache_key()
|
|
|
|
if self._mapupdate != mapupdate:
|
2018-12-27 22:21:17 +01:00
|
|
|
self._items = OrderedDict()
|
2018-12-27 22:02:07 +01:00
|
|
|
self._mapupdate = mapupdate
|
|
|
|
|
2025-04-17 22:25:23 +02:00
|
|
|
enabled = False
|
|
|
|
@classmethod
|
|
|
|
def enable_globally(cls):
|
|
|
|
"""
|
|
|
|
This gets called when the per request cache middleware is loaded.
|
|
|
|
We don't want local cache proxies to work outside of requests.
|
|
|
|
"""
|
2025-04-17 23:18:19 +02:00
|
|
|
LocalCacheProxy.enabled = True
|
2025-04-17 22:25:23 +02:00
|
|
|
|
2018-12-27 17:24:53 +01:00
|
|
|
def set(self, key, value, expire):
|
2018-12-27 22:02:07 +01:00
|
|
|
self._check_mapupdate()
|
2018-12-27 17:24:53 +01:00
|
|
|
cache.set(key, value, expire)
|
2025-04-17 22:25:23 +02:00
|
|
|
if LocalCacheProxy.enabled:
|
|
|
|
self._items[key] = value
|
2018-12-27 17:24:53 +01:00
|
|
|
self._prune()
|
2024-12-27 18:50:36 +01:00
|
|
|
|
|
|
|
def clear(self):
|
|
|
|
self._items.clear()
|
|
|
|
|
|
|
|
|
|
|
|
class RequestLocalCacheProxy(LocalCacheProxy):
|
|
|
|
""" this is a subclass without prune, to be cleared after every request """
|
|
|
|
def _prune(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def _check_mapupdate(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
per_request_cache = RequestLocalCacheProxy(maxsize=settings.CACHE_SIZE_LOCATIONS)
|