Fix disco cache (no more weakrefs)

This commit is contained in:
shteef 2005-11-16 12:50:18 +00:00
parent c3012b4ec6
commit 9aa11f090c
1 changed files with 73 additions and 53 deletions

View File

@ -68,7 +68,6 @@ gtk.glade.textdomain (APP)
GTKGUI_GLADE = 'gtkgui.glade' GTKGUI_GLADE = 'gtkgui.glade'
_icon_cache = weakref.WeakValueDictionary()
# Dictionary mapping category, type pairs to browser class, image pairs. # Dictionary mapping category, type pairs to browser class, image pairs.
# This is a function, so we can call it after the classes are declared. # This is a function, so we can call it after the classes are declared.
@ -107,13 +106,67 @@ def _gen_agent_type_info():
# Category type to "human-readable" description string, and sort priority # Category type to "human-readable" description string, and sort priority
_cat_to_descr = { _cat_to_descr = {
'other': (_('Others'), 2), 'other': (_('Others'), 2),
'gateway': (_('Transports'), 0), 'gateway': (_('Transports'), 0),
'_jid': (_('Transports'), 0), '_jid': (_('Transports'), 0),
'conference': (_('Conference'), 1), 'conference': (_('Conference'), 1),
} }
class CacheDictionary:
'''A dictionary that keeps items around for only a specific time.
Lifetime is in minutes. Getrefresh specifies whether to refresh when
an item is merely accessed instead of set aswell.'''
def __init__(self, lifetime, getrefresh = True):
self.lifetime = lifetime * 1000 * 60
self.getrefresh = getrefresh
self.cache = {}
class CacheItem:
'''An object to store cache items and their timeouts.'''
def __init__(self, value):
self.value = value
self.source = None
def __call__(self):
return self.value
def _expire_timeout(self, key):
'''The timeout has expired, remove the object.'''
del self.cache[key]
return False
def _refresh_timeout(self, key):
'''The object was accessed, refresh the timeout.'''
item = self.cache[key]
if item.source:
gobject.source_remove(item.source)
source = gobject.timeout_add(self.lifetime, self._expire_timeout, key)
item.source = source
def __getitem__(self, key):
item = self.cache[key]
if self.getrefresh:
self._refresh_timeout(key)
return item()
def __setitem__(self, key, value):
item = self.CacheItem(value)
self.cache[key] = item
self._refresh_timeout(key)
def __delitem__(self, key):
item = self.cache[key]
if item.source:
gobject.source_remove(item.source)
del self.cache[key]
def __contains__(self, key):
return key in self.cache
has_key = __contains__
_icon_cache = CacheDictionary(15)
def get_agent_address(jid, node = None): def get_agent_address(jid, node = None):
'''Returns an agent's address for displaying in the GUI.''' '''Returns an agent's address for displaying in the GUI.'''
if node: if node:
@ -161,26 +214,9 @@ class ServicesCache:
ServiceCache instance.''' ServiceCache instance.'''
def __init__(self, account): def __init__(self, account):
self.account = account self.account = account
self._items = {} self._items = CacheDictionary(15, getrefresh = False)
self._info = {} self._info = CacheDictionary(15, getrefresh = False)
self._cbs = {} self._cbs = {}
self._cleancid = None
def _clean(self):
'''Purge outdated items in the cache.'''
now = time.time()
for key, value in self._info.items():
deltatime = now - value[0]
if deltatime > 1800: # 30 minutes.
del self._info[key]
for key, value in self._items.items():
deltatime = now - value[0]
if deltatime > 1800: # 30 minutes.
del self._items[key]
if self._info or self._items:
return True
self._cleancid = None
return False
def _clean_closure(self, cb, type, addr): def _clean_closure(self, cb, type, addr):
# A closure died, clean up # A closure died, clean up
@ -252,20 +288,11 @@ class ServicesCache:
addr = get_agent_address(jid, node) addr = get_agent_address(jid, node)
# Check the cache # Check the cache
if self._info.has_key(addr): if self._info.has_key(addr):
deltatime = time.time() - self._info[addr][0] args = self._info[addr] + args
if force or deltatime > 1800: # 30 minutes. cb(jid, node, *args)
# Cache is outdated return
del self._info[addr]
else:
# Cache hit
args = self._info[addr][1:] + args
cb(jid, node, *args)
return
if nofetch: if nofetch:
return return
# Cache cleaning callback
if not self._cleancid:
self._cleancid = gobject.timeout_add(1800000, self._clean)
# Create a closure object # Create a closure object
cbkey = ('info', addr) cbkey = ('info', addr)
@ -283,20 +310,11 @@ class ServicesCache:
addr = get_agent_address(jid, node) addr = get_agent_address(jid, node)
# Check the cache # Check the cache
if self._items.has_key(addr): if self._items.has_key(addr):
deltatime = time.time() - self._items[addr][0] args = (self._items[addr],) + args
if force or deltatime > 1800: # 30 minutes. cb(jid, node, *args)
# Cache is outdated return
del self._items[addr]
else:
# Cache hit
args = self._items[addr][1:] + args
cb(jid, node, *args)
return
if nofetch: if nofetch:
return return
# Cache cleaning callback
if not self._cleancid:
self._cleancid = gobject.timeout_add(1800000, self._clean)
# Create a closure object # Create a closure object
cbkey = ('items', addr) cbkey = ('items', addr)
@ -311,10 +329,10 @@ class ServicesCache:
def agent_info(self, jid, node, identities, features, data): def agent_info(self, jid, node, identities, features, data):
'''Callback for when we receive an agent's info.''' '''Callback for when we receive an agent's info.'''
# Store in cache
stamp = time.time()
addr = get_agent_address(jid, node) addr = get_agent_address(jid, node)
self._info[addr] = (stamp, identities, features, data)
# Store in cache
self._info[addr] = (identities, features, data)
# Call callbacks # Call callbacks
cbkey = ('info', addr) cbkey = ('info', addr)
@ -327,10 +345,10 @@ class ServicesCache:
def agent_items(self, jid, node, items): def agent_items(self, jid, node, items):
'''Callback for when we receive an agent's items.''' '''Callback for when we receive an agent's items.'''
# Store in cache
stamp = time.time()
addr = get_agent_address(jid, node) addr = get_agent_address(jid, node)
self._items[addr] = (stamp, items)
# Store in cache
self._items[addr] = items
# Call callbacks # Call callbacks
cbkey = ('items', addr) cbkey = ('items', addr)
@ -344,8 +362,9 @@ class ServicesCache:
def agent_info_error(self, jid): def agent_info_error(self, jid):
'''Callback for when a query fails. (even after the browse and agents '''Callback for when a query fails. (even after the browse and agents
namespaces)''' namespaces)'''
# Call callbacks
addr = get_agent_address(jid) addr = get_agent_address(jid)
# Call callbacks
cbkey = ('info', addr) cbkey = ('info', addr)
if self._cbs.has_key(cbkey): if self._cbs.has_key(cbkey):
for cb in self._cbs[cbkey]: for cb in self._cbs[cbkey]:
@ -357,8 +376,9 @@ class ServicesCache:
def agent_items_error(self, jid): def agent_items_error(self, jid):
'''Callback for when a query fails. (even after the browse and agents '''Callback for when a query fails. (even after the browse and agents
namespaces)''' namespaces)'''
# Call callbacks
addr = get_agent_address(jid) addr = get_agent_address(jid)
# Call callbacks
cbkey = ('items', addr) cbkey = ('items', addr)
if self._cbs.has_key(cbkey): if self._cbs.has_key(cbkey):
for cb in self._cbs[cbkey]: for cb in self._cbs[cbkey]: