[Zope-Checkins] CVS: Packages/ZODB - Connection.py:1.98.4.7 POSException.py:1.20.4.3

Tim Peters tim.one at comcast.net
Fri Aug 27 15:04:13 EDT 2004


Update of /cvs-repository/Packages/ZODB
In directory cvs.zope.org:/tmp/cvs-serv16182/lib/python/ZODB

Modified Files:
      Tag: Zope-2_7-branch
	Connection.py POSException.py 
Log Message:
New ConnectionStateError raised if an attempt to close a Connection occurs
while a transaction involving that Connection has pending changes.
Collector #789 is a starting point for rationale (short course:
POSKeyErrors, both temporary and permanent, can result otherwise).
The ZODB3 NEWS file will contain more info.


=== Packages/ZODB/Connection.py 1.98.4.6 => 1.98.4.7 ===
--- Packages/ZODB/Connection.py:1.98.4.6	Fri May 21 12:14:04 2004
+++ Packages/ZODB/Connection.py	Fri Aug 27 15:03:42 2004
@@ -15,8 +15,16 @@
 
 $Id$"""
 
+from cPickle import Unpickler, Pickler
+from cStringIO import StringIO
+import sys
+import threading
+from time import time
+from types import StringType, ClassType
+
 from cPickleCache import PickleCache
 from POSException import ConflictError, ReadConflictError, TransactionError
+from POSException import ConnectionStateError
 from ExtensionClass import Base
 import ExportImport, TmpStore
 from zLOG import LOG, ERROR, BLATHER, WARNING
@@ -25,13 +33,6 @@
 from Transaction import Transaction, get_transaction
 from ZODB.utils import oid_repr
 
-from cPickle import Unpickler, Pickler
-from cStringIO import StringIO
-import sys
-import threading
-from time import time
-from types import StringType, ClassType
-
 global_code_timestamp = 0
 
 def updateCodeTimestamp():
@@ -53,9 +54,9 @@
 
     The Connection manages movement of objects in and out of object storage.
     """
-    _tmp=None
-    _debug_info=()
-    _opened=None
+    _tmp = None
+    _debug_info = ()
+    _opened = None
     _code_timestamp = 0
     _transaction = None
 
@@ -98,6 +99,18 @@
         self._invalid = d.has_key
         self._conflicts = {}
 
+        # We want to raise ConnectionStateError if an attempt to close is
+        # made while modifications are pending.  This is easier in ZODB
+        # 3.3.  While it's not obvious, self._tmp is non-None iff a
+        # committed subtransaction is pending.  For "top level" pending
+        # changes, we set _object_registered to True in register(), and
+        # back to False in abort() and _flush_invalidations().  The latter
+        # is an odd choice, but is common to most ways of clearing pending 
+        # top-level change (tpc_finish, tpc_abort, sync, _setDB).  abort()
+        # is also a bit peculiar, but it's the only Connection method
+        # Transaction.abort() calls, so there's not much choice.
+        self._object_registered = False
+
     def getTransaction(self):
         t = self._transaction
         if t is None:
@@ -258,6 +271,7 @@
         else:
             assert object._p_oid is not None
             self._cache.invalidate(object._p_oid)
+            self._object_registered = False
 
     def cacheFullSweep(self, dt=0):
         self._cache.full_sweep(dt)
@@ -274,6 +288,19 @@
         self.__onCloseCallbacks.append(f)
 
     def close(self):
+        if self._object_registered:
+            # We're currently joined to a transaction.  This spelling
+            # doesn't make much sense here, but I want to keep this
+            # exception the same as in ZODB 3.3 (where this spelling
+            # does make sense).
+            raise ConnectionStateError("Cannot close a connection joined to "
+                                       "a transaction")
+
+        if self._tmp is not None:
+            # There are no direct modifications pending, but a subtransaction
+            # is pending.
+            raise ConnectionStateError("Cannot close a connection with a "
+                                       "pending subtransaction")
         if self._incrgc is not None:
             self._incrgc() # This is a good time to do some GC
 
@@ -519,6 +546,7 @@
             self._inv_lock.release()
         # Now is a good time to collect some garbage
         self._cache.incrgc()
+        self._object_registered = False
 
     def modifiedInVersion(self, oid):
         try: return self._db.modifiedInVersion(oid)
@@ -535,6 +563,7 @@
         # XXX Figure out why this assert causes test failures
         # assert object._p_oid is not None
         self.getTransaction().register(object)
+        self._object_registered = True
 
     def root(self):
         return self['\0\0\0\0\0\0\0\0']


=== Packages/ZODB/POSException.py 1.20.4.2 => 1.20.4.3 ===
--- Packages/ZODB/POSException.py:1.20.4.2	Thu Oct 23 20:45:55 2003
+++ Packages/ZODB/POSException.py	Fri Aug 27 15:03:42 2004
@@ -219,3 +219,14 @@
 
     XXX The exception ought to have a member that is the invalid object.
     """
+
+class ConnectionStateError(POSError):
+    """A Connection isn't in the required state for an operation.
+
+    o An operation such as a load is attempted on a closed connection.
+
+    o An attempt to close a connection is made while the connection is
+      still joined to a transaction (for example, a transaction is in
+      progress, with uncommitted modifications in the connection).
+    """
+ 
\ No newline at end of file



More information about the Zope-Checkins mailing list