[Zope-Checkins] CVS: Zope/lib/python/ZODB/tests - PackableStorage.py:1.13.6.2
Jeremy Hylton
jeremy@zope.com
Tue, 20 May 2003 08:58:22 -0400
Update of /cvs-repository/Zope/lib/python/ZODB/tests
In directory cvs.zope.org:/tmp/cvs-serv25467/ZODB/tests
Modified Files:
Tag: Zope-2_6-branch
PackableStorage.py
Log Message:
Backport fix for concurrent packing and writing.
=== Zope/lib/python/ZODB/tests/PackableStorage.py 1.13.6.1 => 1.13.6.2 ===
--- Zope/lib/python/ZODB/tests/PackableStorage.py:1.13.6.1 Tue May 13 15:03:53 2003
+++ Zope/lib/python/ZODB/tests/PackableStorage.py Tue May 20 08:58:21 2003
@@ -25,11 +25,15 @@
except ImportError:
from StringIO import StringIO
+import threading
import time
+
from ZODB import DB
from Persistence import Persistent
from ZODB.referencesf import referencesf
-
+from ZODB.tests.MinPO import MinPO
+from ZODB.tests.StorageTestBase import snooze
+from ZODB.POSException import ConflictError
ZERO = '\0'*8
@@ -376,3 +380,58 @@
conn.sync()
eq(root['obj'].value, 7)
+
+ def checkPackWhileWriting(self):
+ # A storage should allow some reading and writing during
+ # a pack. This test attempts to exercise locking code
+ # in the storage to test that it is safe. It generates
+ # a lot of revisions, so that pack takes a long time.
+
+ db = DB(self._storage)
+ conn = db.open()
+ root = conn.root()
+
+ for i in range(10):
+ root[i] = MinPO(i)
+ get_transaction().commit()
+
+ snooze()
+ packt = time.time()
+
+ for j in range(10):
+ for i in range(10):
+ root[i].value = MinPO(i)
+ get_transaction().commit()
+
+ threads = [ClientThread(db) for i in range(4)]
+ for t in threads:
+ t.start()
+ db.pack(packt)
+ for t in threads:
+ t.join(30)
+ for t in threads:
+ t.join(1)
+ self.assert_(not t.isAlive())
+
+ # iterator over the storage to make sure it's sane
+ if not hasattr(self._storage, "iterator"):
+ return
+ iter = self._storage.iterator()
+ for txn in iter:
+ for data in txn:
+ pass
+ iter.close()
+
+class ClientThread(threading.Thread):
+
+ def __init__(self, db):
+ threading.Thread.__init__(self)
+ self.root = db.open().root()
+
+ def run(self):
+ for j in range(50):
+ try:
+ self.root[j % 10].value = MinPO(j)
+ get_transaction().commit()
+ except ConflictError:
+ get_transaction().abort()