Source code for gavo.base.caches
"""
Accessor functions for the various immutables we have.
The main purpose of this module is to keep caches of resource descriptors,
and other items the parsing of which may take some time.
All you need to do is provide a function taking a "key" (a string, most
likely) and returning the object. Then call
base.caches.makeCache(<accessorName>, <function>)
After that, clients can call
base.caches.<accessorName>(key)
You can additionally provide an isDirty(res) function when calling makeCache.
This can return True if the resource is out of date and should be reloaded.
An alternative interface to registering caches is the registerCache function
(see there).
"""
#c Copyright 2008-2023, the GAVO project <gavo@ari.uni-heidelberg.de>
#c
#c This program is free software, covered by the GNU GPL. See the
#c COPYING file in the source distribution.
[docs]class CacheRegistry:
"""is a registry for caches kept to be able to clear them.
A cache is assumed to be a dictionary here.
"""
def __init__(self):
self.knownCaches = []
[docs] def clearall(self):
for cache in self.knownCaches:
for key in list(cache.keys()):
del cache[key]
[docs] def clearForName(self, key):
for cache in self.knownCaches:
if key in cache:
del cache[key]
[docs] def register(self, cache):
self.knownCaches.append(cache)
_cacheRegistry = CacheRegistry()
clearCaches = _cacheRegistry.clearall
clearForName = _cacheRegistry.clearForName
def _makeCache(creator, isDirty):
"""returns a callable that memoizes the results of creator.
The creator has to be a function taking an id and returning the
designated object.
The whole thing is thread-safe only when the creators are. It is
possible that arbitrarily many creators for the same id run. Only one
will win in the end.
Race conditions are possible when exceptions occur, but then creators
behaviour should only depend on id, and so it shouldn't matter.
isDirty can be a function returning true when the cache should be
cleared. The function is passed the current resource. If isDirty
is None, no such check is performed.
"""
cache = {}
_cacheRegistry.register(cache)
def func(id):
if isDirty is not None and id in cache and isDirty(cache[id]):
clearForName(id)
if not id in cache:
try:
cache[id] = creator(id)
except Exception as exc:
cache[id] = exc
raise
if isinstance(cache[id], Exception):
raise cache[id]
else:
return cache[id]
return func
[docs]def registerCache(name, cacheDict, creationFunction):
"""registers a custom cache.
This function makes creationFunction available as base.caches.name,
and it registers cacheDict with the cache manager such that cacheDict
is cleared as necessary.
creationFunction must manage cacheDict itself, and of course it
must always use the instance passed to registerCache.
This is for "magic" things like getRD that has to deal with aliases
and such. For normal use, use makeCache.
"""
globals()[name] = creationFunction
_cacheRegistry.register(cacheDict)
[docs]def makeCache(name, callable, isDirty=None):
"""creates a new function name to cache results to calls to callable.
isDirty can be a function returning true when the cache should be
cleared. The function is passed the current resource.
"""
globals()[name] = _makeCache(callable, isDirty)