[Zodb-checkins] CVS: Zope3/src/persistence - cache.py:1.3

Jeremy Hylton jeremy@zope.com
Fri, 7 Mar 2003 15:26:47 -0500


Update of /cvs-repository/Zope3/src/persistence
In directory cvs.zope.org:/tmp/cvs-serv29075/persistence

Modified Files:
	cache.py 
Log Message:
Implement a new throw-away LRU cache.

The old cache, through a variety of confusions, was guaranteed never
to evict anything after the first call to incrgc().  Eliminate the
minimize() and full_sweep() methods that were only called by the tests
anyway.


=== Zope3/src/persistence/cache.py 1.2 => 1.3 ===
--- Zope3/src/persistence/cache.py:1.2	Wed Dec 25 09:12:13 2002
+++ Zope3/src/persistence/cache.py	Fri Mar  7 15:26:46 2003
@@ -21,8 +21,6 @@
 
     __implements__ = ICache
 
-    __iter=None
-
     def __init__(self, size=500, inactive=300):
         self.__ghosts = {}
         self.__gget = self.__ghosts.get
@@ -42,22 +40,23 @@
             return o
 
     def get(self, oid, default=None):
-        o = self.__gget(oid, self)
-        if o is self:
-            o = self.__active.get(oid, self)
-            if o is self: return default
-        o=o()
+        o = self.__gget(oid, None)
+        if o is None:
+            o = self.__active.get(oid, None)
+            if o is None:
+                return default
+        o = o()
         if o is None:
             return default
         else:
             return o
 
-    def __setitem__(self, oid, object):
-        if object._p_changed is None:
+    def __setitem__(self, oid, obj):
+        if obj._p_changed is None:
             # ghost
-            self.__ghosts[oid] = ref(object, _dictdel(oid, self.__ghosts))
+            self.__ghosts[oid] = ref(obj, _dictdel(oid, self.__ghosts))
         else:
-            self.__active[oid] = ref(object, _dictdel(oid, self.__active))
+            self.__active[oid] = ref(obj, _dictdel(oid, self.__active))
 
     def __delitem__(self, oid):
         # XXX is there any way to know which dict the key is in?
@@ -71,7 +70,7 @@
             pass
 
     def __len__(self):
-        return len(self.__ghosts)+len(self.__active)
+        return len(self.__ghosts) + len(self.__active)
 
     def setstate(self, oid, object):
         try:
@@ -80,103 +79,46 @@
             pass
         self.__active[oid] = ref(object, _dictdel(oid, self.__active))
 
-    def incrgc(self, multiple=1):
-        na=len(self.__active)
-        if na < 1: return
-
-        # how many objects do we scan?
-        n=min(multiple * max((na-self._size)/10, 3), na)
-
-        # how long can objects be inactive?
-        inactive = self._inactive * (
-            0.2 + 0.1 * (min(100, 8 * self._size/na))
-            )
-
-        active=self.__active
-        aget=active.get
-        ghosts=self.__ghosts
-        doomed=[]
-
-        now=int(time()%86400)
-
-        i=self.__iter
-        if i is None:
-            i=iter(self.__active)
-
-        while n:
-            n-=1
-            try: oid = i.next()
-            except StopIteration:
-                del self.__iter
-                return
-
-            ob=aget(oid, self)
-            if ob is self: continue
-            ob=ob()
-            state = ob._p_changed
-
-            if state==0 and abs(ob._p_atime-now) > inactive:
-                doomed.append(oid)
-                continue
-            if state is None:
-                doomed.append(oid)
-
-        for oid in doomed:
-            ob=aget(oid, self)
-            if ob is self: continue
-            ob=ob()
-            ob._p_deactivate()
-            state = ob._p_changed
-            if state is None:
-                del active[oid]
-                ghosts[oid] = ref(ob, _dictdel(oid, ghosts))
-
-    def full_sweep(self):
-        now=int(time()%86400)
-        active=self.__active
-        ghosts=self.__ghosts
-        na=len(active)
-
-        # how long can objects be inactive?
-        inactive = self._inactive * (
-            0.2 + 0.1 * (min(100, 8 * self._size/na))
-            )
-
-        doomed=[]
-
-        for oid in active:
-            ob=active[oid]
-            ob=ob()
-            state = ob._p_changed
-            if state==0 and abs(ob._p_atime-now) > inactive:
-                doomed.append(oid)
-                continue
-            if state is None:
-                doomed.append(oid)
-
-        for oid in doomed:
-            ob._p_deactivate()
-            state = ob._p_changed
-            if state is None:
-                del active[oid]
-                ghosts[oid] = ref(ob, _dictdel(oid, ghosts))
-
-    def minimize(self):
-        active=self.__active
-        aget=active.get
-        ghosts=self.__ghosts
-
-        # Grump: I cant use an iterator because the size will change
-        # during iteration. :(
-        for oid in active.keys():
-            ob=aget(oid, self)
-            if ob is self: continue
-            ob=ob()
-            ob._p_deactivate()
-            if ob._p_changed is None:
-                del active[oid]
-                ghosts[oid] = ref(ob, _dictdel(oid, ghosts))
-        self.__iter=None
+    def incrgc(self):
+        na = len(self.__active)
+        if na < 1:
+            return
+
+        now = int(time() % 86400)
+
+        # Implement a trivial LRU cache by sorting the items by
+        # access time and trundling over the last until we've reached
+        # out target.  The number of objects in the cache should
+        # be relatively small (thousands) so the memory for the
+        # list is pretty minimal.
+        L = []
+        for oid, ob in self.__active.iteritems():
+            if ob is not None:
+                ob = ob()
+            L.append((ob._p_atime, oid, ob))
+        L.sort()
+
+        if na > self._size:
+            # If the cache is full, ghostify everything up to the cache
+            # limit.
+            n = na - self._size
+            must_go = L[:n]
+            L = L[n:]
+            for atime, oid, ob in L:
+                self._ghostify(oid, ob)
+
+        # ghostify old objects regardless of cache size
+        stop_at = now - self._inactive
+        for atime, oid, ob in L:
+            if atime > stop_at:
+                break
+            self._ghostify(oid, ob)
+
+    def _ghostify(self, oid, ob):
+        ob._p_deactivate()
+        if ob._p_changed == None:
+            del self.__active[oid]
+            self.__ghosts[oid] = ref(ob, _dictdel(oid, self.__ghosts))
 
     def invalidate(self, oid):
         ob = self.__aget(oid)