[Zope-Checkins] CVS: ZODB3/BDBStorage/tests - test_autopack.py:1.13
Jeremy Hylton
jeremy at zope.com
Mon Sep 15 12:29:53 EDT 2003
Update of /cvs-repository/ZODB3/BDBStorage/tests
In directory cvs.zope.org:/tmp/cvs-serv29167/BDBStorage/tests
Modified Files:
test_autopack.py
Log Message:
Merge changes from ZODB3-3_2-branch to Zope-2_7-branch.
Please make all future changes on the Zope-2_7-branch instead.
=== ZODB3/BDBStorage/tests/test_autopack.py 1.12 => 1.13 ===
--- ZODB3/BDBStorage/tests/test_autopack.py:1.12 Thu Sep 11 13:01:30 2003
+++ ZODB3/BDBStorage/tests/test_autopack.py Mon Sep 15 12:29:22 2003
@@ -56,7 +56,7 @@
# Autopack every 1 second, 2 seconds into the past, no classic packs
config.frequency = 1
config.packtime = 2
- config.classicpack = 0
+ config.gcpack = 0
return config
def _wait_for_next_autopack(self):
@@ -119,7 +119,7 @@
# every time.
config.frequency = 1
config.packtime = 2
- config.classicpack = 1
+ config.gcpack = 1
return config
def testAutomaticClassicPack(self):
@@ -271,277 +271,15 @@
-class RaceConditionBase(BerkeleyTestBase):
- def setUp(self):
- BerkeleyTestBase.setUp(self)
- self._cv = threading.Condition()
- self._storage.cv = self._cv
-
- def tearDown(self):
- # clean up any outstanding transactions
- get_transaction().abort()
- BerkeleyTestBase.tearDown(self)
-
- def _getPackThread(self, storage):
- raise NotImplementedError
-
- def testRaceCondition(self):
- unless = self.failUnless
- storage = self._storage
- db = DB(storage)
- conn = db.open()
- root = conn.root()
- # Start by storing a root reachable object.
- obj1 = C()
- obj1.value = 888
- root.obj1 = obj1
- txn = get_transaction()
- txn.note('root -> obj1')
- txn.commit()
- # Now, start a transaction, store an object, but don't yet complete
- # the transaction. This will ensure that the second object has a tid
- # < packtime, but it won't be root reachable yet.
- obj2 = C()
- t = Transaction()
- storage.tpc_begin(t)
- obj2sn = storage.store('\0'*7 + '\2', ZERO, zodb_pickle(obj2), '', t)
- # Now, acquire the condvar lock and start a thread that will do a
- # pack, up to the _sweep call. Wait for the _mark() call to
- # complete.
- now = time.time()
- while now == time.time():
- time.sleep(0.1)
- self._cv.acquire()
- packthread = self._getPackThread(storage)
- packthread.start()
- self._cv.wait()
- # Now that the _mark() has finished, complete the transaction, which
- # links the object to root.
- root.obj2 = obj2
- rootsn = storage.getSerial(ZERO)
- rootsn = storage.store(ZERO, rootsn, zodb_pickle(root), '', t)
- storage.tpc_vote(t)
- storage.tpc_finish(t)
- # And notify the pack thread that it can do the sweep and collect
- self._cv.notify()
- self._cv.wait()
- # We're done with the condvar and the thread
- self._cv.release()
- packthread.join()
- # Now make sure that all the interesting objects are still available
- rootsn = storage.getSerial(ZERO)
- obj1sn = storage.getSerial('\0'*7 + '\1')
- obj2sn = storage.getSerial('\0'*7 + '\2')
- # obj1 revision was written before the second revision of the root
- unless(obj1sn < rootsn)
- unless(rootsn == obj2sn)
- unless(obj1sn < obj2sn)
-
- def testEarlierRaceCondition(self):
- unless = self.failUnless
- storage = self._storage
- db = DB(storage)
- conn = db.open()
- root = conn.root()
- # Start by storing a root reachable object.
- obj1 = C()
- obj1.value = 888
- root.obj1 = obj1
- txn = get_transaction()
- txn.note('root -> obj1')
- txn.commit()
- # Now, start a transaction, store an object, but don't yet complete
- # the transaction. This will ensure that the second object has a tid
- # < packtime, but it won't be root reachable yet.
- obj2 = C()
- t = Transaction()
- storage.tpc_begin(t)
- # Now, acquire the condvar lock and start a thread that will do a
- # pack, up to the _sweep call. Wait for the _mark() call to
- # complete.
- now = time.time()
- while now == time.time():
- time.sleep(0.1)
- self._cv.acquire()
- packthread = self._getPackThread(storage)
- packthread.start()
- self._cv.wait()
- obj2sn = storage.store('\0'*7 + '\2', ZERO, zodb_pickle(obj2), '', t)
- # Now that the _mark() has finished, complete the transaction, which
- # links the object to root.
- root.obj2 = obj2
- rootsn = storage.getSerial(ZERO)
- rootsn = storage.store(ZERO, rootsn, zodb_pickle(root), '', t)
- storage.tpc_vote(t)
- storage.tpc_finish(t)
- # And notify the pack thread that it can do the sweep and collect
- self._cv.notify()
- self._cv.wait()
- # We're done with the condvar and the thread
- self._cv.release()
- packthread.join()
- # Now make sure that all the interesting objects are still available
- rootsn = storage.getSerial(ZERO)
- obj1sn = storage.getSerial('\0'*7 + '\1')
- obj2sn = storage.getSerial('\0'*7 + '\2')
- # obj1 revision was written before the second revision of the root
- unless(obj1sn < rootsn)
- unless(rootsn == obj2sn)
- unless(obj1sn < obj2sn)
-
-
-
-# Subclass which does ugly things to _dopack so we can actually test the race
-# condition. We need to store a new object in the database between the
-# _mark() call and the _sweep() call.
-class SynchronizedFullStorage(BDBFullStorage):
- # XXX Cut and paste copy from BDBFullStorage, except where indicated
- def _dopack(self, t, gc=True):
- # t is a TimeTime, or time float, convert this to a TimeStamp object,
- # using an algorithm similar to what's used in FileStorage. We know
- # that our transaction ids, a.k.a. revision ids, are timestamps.
- #
- # BAW: should a pack time in the future be a ValueError? We'd have to
- # worry about clock skew, so for now, we just set the pack time to the
- # minimum of t and now.
- packtime = min(t, time.time())
- t0 = TimeStamp(*(time.gmtime(packtime)[:5] + (packtime % 60,)))
- packtid = `t0`
- # Collect all revisions of all objects earlier than the pack time.
- self._lock_acquire()
- try:
- self._withtxn(self._collect_revs, packtid)
- finally:
- self._lock_release()
- # Collect any objects with refcount zero.
- self._lock_acquire()
- try:
- self._withtxn(self._collect_objs)
- finally:
- self._lock_release()
- # If we're not doing a classic pack, we're done.
- if not gc:
- return
- # Do a mark and sweep for garbage collection. Calculate the set of
- # objects reachable from the root. Anything else is a candidate for
- # having all their revisions packed away. The set of reachable
- # objects lives in the _packmark table.
- self._lock_acquire()
- try:
- self._withtxn(self._mark, packtid)
- finally:
- self._lock_release()
- # XXX thread coordination code start
- self.cv.acquire()
- self.cv.notify()
- self.cv.wait()
- # XXX thread coordination code stop
- #
- # Now perform a sweep, using oidqueue to hold all object ids for
- # objects which are not root reachable as of the pack time.
- self._lock_acquire()
- try:
- self._withtxn(self._sweep, packtid)
- finally:
- self._lock_release()
- # Once again, collect any objects with refcount zero due to the mark
- # and sweep garbage collection pass.
- self._lock_acquire()
- try:
- self._withtxn(self._collect_objs)
- finally:
- self._lock_release()
- # XXX thread coordination code start
- self.cv.notify()
- self.cv.release()
- # XXX thread coordination code stop
-
-
-class FullPackThread(threading.Thread):
- def __init__(self, storage):
- threading.Thread.__init__(self)
- self._storage = storage
-
- def run(self):
- self._storage.autopack(time.time(), gc=True)
-
-
-class TestFullClassicPackRaceCondition(RaceConditionBase):
- ConcreteStorage = SynchronizedFullStorage
-
- def _getPackThread(self, storage):
- return FullPackThread(storage)
-
-
-
-# Subclass which does ugly things to _dopack so we can actually test the race
-# condition. We need to storage a new object in the database between the
-# _mark() call and the _sweep() call.
-class SynchronizedMinimalStorage(BDBMinimalStorage):
- # XXX Cut and paste copy from BDBMinimalStorage, except where indicated
- def _dopack(self):
- # Do a mark and sweep for garbage collection. Calculate the set of
- # objects reachable from the root. Anything else is a candidate for
- # having all their revisions packed away. The set of reachable
- # objects lives in the _packmark table.
- self._lock_acquire()
- try:
- self._withtxn(self._mark)
- finally:
- self._lock_release()
- # XXX thread coordination code start
- self.cv.acquire()
- self.cv.notify()
- self.cv.wait()
- # XXX thread coordination code stop
- #
- # Now perform a sweep, using oidqueue to hold all object ids for
- # objects which are not root reachable as of the pack time.
- self._lock_acquire()
- try:
- self._withtxn(self._sweep)
- finally:
- self._lock_release()
- # Once again, collect any objects with refcount zero due to the mark
- # and sweep garbage collection pass.
- self._lock_acquire()
- try:
- self._withtxn(self._collect_objs)
- finally:
- self._lock_release()
- # XXX thread coordination code start
- self.cv.notify()
- self.cv.release()
- # XXX thread coordination code stop
-
-
-class MinimalPackThread(threading.Thread):
- def __init__(self, storage):
- threading.Thread.__init__(self)
- self._storage = storage
-
- def run(self):
- self._storage.pack(time.time(), referencesf)
-
-
-class TestMinimalClassicPackRaceCondition(RaceConditionBase):
- ConcreteStorage = SynchronizedMinimalStorage
-
- def _getPackThread(self, storage):
- return MinimalPackThread(storage)
-
-
-
def test_suite():
- return BDBStorage.tests.BerkeleyTestBase.makeSuite(
- TestAutopack,
- TestAutomaticClassicPack,
- TestMinimalPack,
- TestFullClassicPackRaceCondition,
- TestMinimalClassicPackRaceCondition,
- prefix='test',
- level=2
- )
+ suite = unittest.TestSuite()
+ suite.level = 2
+ if BDBStorage.is_available:
+ suite.addTest(unittest.makeSuite(TestAutopack))
+ suite.addTest(unittest.makeSuite(TestAutomaticClassicPack))
+ suite.addTest(unittest.makeSuite(TestMinimalPack))
+ return suite
+
if __name__ == '__main__':
More information about the Zope-Checkins
mailing list