Skip to content
This repository has been archived by the owner on Apr 26, 2024. It is now read-only.

Fix LruCache. Make TreeCache track its own size. #538

Merged
merged 6 commits into from
Jan 29, 2016
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
10 changes: 4 additions & 6 deletions synapse/util/caches/lrucache.py
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ class LruCache(object):
"""
def __init__(self, max_size, keylen=1, cache_type=dict):
cache = cache_type()
self.size = 0
self.cache = cache # Used for introspection.
list_root = []
list_root[:] = [list_root, list_root, None, None]

Expand All @@ -60,7 +60,6 @@ def add_node(key, value):
prev_node[NEXT] = node
next_node[PREV] = node
cache[key] = node
self.size += 1

def move_node_to_front(node):
prev_node = node[PREV]
Expand All @@ -79,7 +78,6 @@ def delete_node(node):
next_node = node[NEXT]
prev_node[NEXT] = next_node
next_node[PREV] = prev_node
self.size -= 1

@synchronized
def cache_get(key, default=None):
Expand All @@ -98,7 +96,7 @@ def cache_set(key, value):
node[VALUE] = value
else:
add_node(key, value)
if self.size > max_size:
if len(cache) > max_size:
todelete = list_root[PREV]
delete_node(todelete)
cache.pop(todelete[KEY], None)
Expand All @@ -110,7 +108,7 @@ def cache_set_default(key, value):
return node[VALUE]
else:
add_node(key, value)
if self.size > max_size:
if len(cache) > max_size:
todelete = list_root[PREV]
delete_node(todelete)
cache.pop(todelete[KEY], None)
Expand Down Expand Up @@ -145,7 +143,7 @@ def cache_clear():

@synchronized
def cache_len():
return self.size
return len(cache)

@synchronized
def cache_contains(key):
Expand Down
36 changes: 34 additions & 2 deletions synapse/util/caches/treecache.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ class TreeCache(object):
Keys must be tuples.
"""
def __init__(self):
self.size = 0
self.root = {}

def __setitem__(self, key, value):
Expand All @@ -20,17 +21,19 @@ def set(self, key, value):
node = self.root
for k in key[:-1]:
node = node.setdefault(k, {})
node[key[-1]] = value
node[key[-1]] = _Entry(value)
self.size += 1

def get(self, key, default=None):
node = self.root
for k in key[:-1]:
node = node.get(k, None)
if node is None:
return default
return node.get(key[-1], default)
return node.get(key[-1], _Entry(default)).value

def clear(self):
self.size = 0
self.root = {}

def pop(self, key, default=None):
Expand All @@ -57,4 +60,33 @@ def pop(self, key, default=None):
break
node_and_keys[i+1][0].pop(k)

popped, cnt = _strip_and_count_entires(popped)
self.size -= cnt
return popped
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If you are going to enumerate the popped tree here then you might as well return a list of popped entries for the LruCache to remove from its linked list.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'd quite like to keep the pop method returning a single item when given a full key, as that matches expectations.


def __len__(self):
return self.size


class _Entry(object):
__slots__ = ["value"]

def __init__(self, value):
self.value = value


def _strip_and_count_entires(d):
"""Takes an _Entry or dict with leaves of _Entry's, and either returns the
value or a dictionary with _Entry's replaced by their values.

Also returns the count of _Entry's
"""
if isinstance(d, dict):
cnt = 0
for key, value in d.items():
v, n = _strip_and_count_entires(value)
d[key] = v
cnt += n
return d, cnt
else:
return d.value, 1
7 changes: 7 additions & 0 deletions tests/util/test_lrucache.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@
from synapse.util.caches.lrucache import LruCache
from synapse.util.caches.treecache import TreeCache


class LruCacheTestCase(unittest.TestCase):

def test_get_set(self):
Expand Down Expand Up @@ -72,3 +73,9 @@ def test_del_multi(self):
self.assertEquals(cache.get(("vehicles", "car")), "vroom")
self.assertEquals(cache.get(("vehicles", "train")), "chuff")
# Man from del_multi say "Yes".

def test_clear(self):
cache = LruCache(1)
cache["key"] = 1
cache.clear()
self.assertEquals(len(cache), 0)
12 changes: 12 additions & 0 deletions tests/util/test_treecache.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ def test_get_set_onelevel(self):
cache[("b",)] = "B"
self.assertEquals(cache.get(("a",)), "A")
self.assertEquals(cache.get(("b",)), "B")
self.assertEquals(len(cache), 2)

def test_pop_onelevel(self):
cache = TreeCache()
Expand All @@ -33,6 +34,7 @@ def test_pop_onelevel(self):
self.assertEquals(cache.pop(("a",)), "A")
self.assertEquals(cache.pop(("a",)), None)
self.assertEquals(cache.get(("b",)), "B")
self.assertEquals(len(cache), 1)

def test_get_set_twolevel(self):
cache = TreeCache()
Expand All @@ -42,6 +44,7 @@ def test_get_set_twolevel(self):
self.assertEquals(cache.get(("a", "a")), "AA")
self.assertEquals(cache.get(("a", "b")), "AB")
self.assertEquals(cache.get(("b", "a")), "BA")
self.assertEquals(len(cache), 3)

def test_pop_twolevel(self):
cache = TreeCache()
Expand All @@ -53,6 +56,7 @@ def test_pop_twolevel(self):
self.assertEquals(cache.get(("a", "b")), "AB")
self.assertEquals(cache.pop(("b", "a")), "BA")
self.assertEquals(cache.pop(("b", "a")), None)
self.assertEquals(len(cache), 1)

def test_pop_mixedlevel(self):
cache = TreeCache()
Expand All @@ -64,3 +68,11 @@ def test_pop_mixedlevel(self):
self.assertEquals(cache.get(("a", "a")), None)
self.assertEquals(cache.get(("a", "b")), None)
self.assertEquals(cache.get(("b", "a")), "BA")
self.assertEquals(len(cache), 1)

def test_clear(self):
cache = TreeCache()
cache[("a",)] = "A"
cache[("b",)] = "B"
cache.clear()
self.assertEquals(len(cache), 0)