[Zope-Checkins] CVS: ZODB3/ZEO/tests - ConnectionTests.py:1.4.2.5.4.1

Barry Warsaw barry@zope.com
Fri, 1 Aug 2003 16:10:29 -0400


Update of /cvs-repository/ZODB3/ZEO/tests
In directory cvs.zope.org:/tmp/cvs-serv3990

Modified Files:
      Tag: ZODB3-3_1-boston-branch
	ConnectionTests.py 
Log Message:
checkTimeoutAfterVote(): A new test which tickles a bug discovered via
code inspection (and other means <wink>) in ClientStorage's
tpc_finish().  If a storage times out between the vote and the finish,
the zeo cache would have gotten populated with objects that didn't
make it to the storage server.

A fix for this is coming up next.


=== ZODB3/ZEO/tests/ConnectionTests.py 1.4.2.5 => 1.4.2.5.4.1 ===
--- ZODB3/ZEO/tests/ConnectionTests.py:1.4.2.5	Wed Jun 11 19:11:32 2003
+++ ZODB3/ZEO/tests/ConnectionTests.py	Fri Aug  1 16:10:23 2003
@@ -26,6 +26,7 @@
 from ZEO.ClientStorage import ClientStorage, ClientDisconnected
 from ZEO.Exceptions import Disconnected
 from ZEO.zrpc.marshal import Marshaller
+from ZEO.zrpc.error import DisconnectedError
 from ZEO.tests.InvalidationTests import InvalidationTests
 
 from ZODB.Transaction import get_transaction, Transaction
@@ -35,7 +36,7 @@
 from ZODB.tests.StorageTestBase import zodb_pickle, zodb_unpickle
 
 class DummyDB:
-    def invalidate(self, *args):
+    def invalidate(self, *args, **kws):
         pass
 
 class ConnectionTests(StorageTestBase.StorageTestBase,
@@ -577,3 +578,33 @@
         storage.tpc_begin(txn)
         storage.tpc_abort(txn)
 
+    def checkTimeoutAfterVote(self):
+        raises = self.assertRaises
+        unless = self.failUnless
+        self.shutdownServer()
+        self.startServer(transaction_timeout=1)
+        self._storage = storage = self.openClientStorage()
+        # Assert that the zeo cache is empty
+        unless(not storage._cache._index)
+        # Create the object
+        oid = storage.new_oid()
+        obj = MinPO(7)
+        ZERO = '\0'*8
+        # Now do a store, sleeping before the finish so as to cause a timeout
+        t = Transaction()
+        storage.tpc_begin(t)
+        revid1 = storage.store(oid, ZERO, zodb_pickle(obj), '', t)
+        storage.tpc_vote(t)
+        # Now sleep long enough for the storage to time out
+        time.sleep(3)
+        storage.sync()
+        self.assert_(not storage.is_connected())
+        storage._wait()
+        self.assert_(storage.is_connected())
+        # This will just return
+        raises(ClientDisconnected, storage.tpc_finish, t)
+        # The cache should still be empty
+        unless(not storage._cache._index)
+        # Load should fail since the object should not be in either the cache
+        # or the server.
+        raises(KeyError, storage.load, oid, '')