[Zodb-checkins] SVN: ZODB/branches/3.4/ Fix obscure bug.

Tim Peters tim.one at comcast.net
Fri May 6 11:04:41 EDT 2005


Log message for revision 30279:
  Fix obscure bug.
  
  If a threaded transaction manager ever passed None to
  the Transaction constructor's `synchronizers` argument, then
  synchronizers registered later in the same transaction
  were invisible to the transaction, and so their afterCompletion()
  methods wouldn't get called when the transaction ended.
  

Changed:
  U   ZODB/branches/3.4/NEWS.txt
  U   ZODB/branches/3.4/src/ZODB/tests/synchronizers.txt
  U   ZODB/branches/3.4/src/transaction/_manager.py

-=-
Modified: ZODB/branches/3.4/NEWS.txt
===================================================================
--- ZODB/branches/3.4/NEWS.txt	2005-05-06 14:53:54 UTC (rev 30278)
+++ ZODB/branches/3.4/NEWS.txt	2005-05-06 15:04:41 UTC (rev 30279)
@@ -1,3 +1,20 @@
+What's new in ZODB3 3.4a7?
+==========================
+Release date: 06-May-2005
+
+This was an internal release, to fix an obscure older bug discovered while
+testing the ``ISynchronizer`` enhancements.
+
+transaction
+-----------
+
+- If the first activity seen by a new ``ThreadTransactionManager`` was
+  an explicit ``begin()`` call, then synchronizers registered after that
+  (but still during the first transaction) were not communicated to the
+  transaction object.  As a result, the ``afterCompletion()`` methods of
+  registered synchronizers weren't called when the first transaction ended.
+
+
 What's new in ZODB3 3.4a6?
 ==========================
 Release date: 05-May-2005

Modified: ZODB/branches/3.4/src/ZODB/tests/synchronizers.txt
===================================================================
--- ZODB/branches/3.4/src/ZODB/tests/synchronizers.txt	2005-05-06 14:53:54 UTC (rev 30278)
+++ ZODB/branches/3.4/src/ZODB/tests/synchronizers.txt	2005-05-06 15:04:41 UTC (rev 30279)
@@ -50,7 +50,6 @@
     >>> st.sync_called  # False before 3.4
     True
 
-
 Clean up.  Closing db isn't enough -- closing a DB doesn't close its
 Connections.  Leaving our Connection open here can cause the
 SimpleStorage.sync() method to get called later, during another test, and
@@ -58,4 +57,46 @@
 a weird traceback then ;-)
 
     >>> cn.close()
+
+One more, very obscure.  It was the case that if the first action a new
+threaded transaction manager saw was a begin() call, then synchronizers
+registered after that in the same transaction weren't communicated to
+the Transaction object, and so the storage's afterCompletion() hook wasn't
+called when the transaction commited.  None of the test suites (ZODB's,
+Zope 2.8's, or Zope3's) caught that, but apparently Zope3 takes this path
+at some point when serving pages.
+
+    >>> tm = transaction.ThreadTransactionManager()
+    >>> st.sync_called = False
+    >>> dummy = tm.begin()  # we're doing this _before_ opening a connection
+    >>> cn = db.open(txn_mgr=tm)
+    >>> rt = cn.root()      # make a change
+    >>> rt['c'] = 3
+    >>> st.sync_called
+    False
+
+Now ensure that st.afterCompletion() gets called by commit despite that the
+Connection registered after the transaction began:
+
+    >>> tm.commit()
+    >>> st.sync_called
+    True
+
+And try the same thing with a non-threaded TM:
+
+    >>> cn.close()
+    >>> tm = transaction.TransactionManager()
+    >>> st.sync_called = False
+    >>> dummy = tm.begin()  # we're doing this _before_ opening a connection
+    >>> cn = db.open(txn_mgr=tm)
+    >>> rt = cn.root()      # make a change
+    >>> rt['d'] = 4
+    >>> st.sync_called
+    False
+    >>> tm.commit()
+    >>> st.sync_called
+    True
+
+    >>> cn.close()
     >>> db.close()
+

Modified: ZODB/branches/3.4/src/transaction/_manager.py
===================================================================
--- ZODB/branches/3.4/src/transaction/_manager.py	2005-05-06 14:53:54 UTC (rev 30278)
+++ ZODB/branches/3.4/src/transaction/_manager.py	2005-05-06 15:04:41 UTC (rev 30279)
@@ -33,7 +33,7 @@
 # at top level here.
 
 # Call the ISynchronizer newTransaction() method on every element of
-# WeakSet synchs (or skip it if synchs is None).
+# WeakSet synchs.
 # A transaction manager needs to do this whenever begin() is called.
 # Since it would be good if tm.get() returned the new transaction while
 # newTransaction() is running, calling this has to be delayed until after
@@ -43,6 +43,13 @@
     if synchs:
         synchs.map(lambda s: s.newTransaction(txn))
 
+# Important:  we must always pass a WeakSet (even if empty) to the Transaction
+# constructor:  synchronizers are registered with the TM, but the
+# ISynchronizer xyzCompletion() methods are called by Transactions without
+# consulting the TM, so we need to pass a mutable collection of synchronizers
+# so that Transactions "see" synchronizers that get registered after the
+# Transaction object is constructed.
+
 class TransactionManager(object):
 
     def __init__(self):
@@ -91,6 +98,7 @@
     def __init__(self):
         # _threads maps thread ids to transactions
         self._txns = {}
+
         # _synchs maps a thread id to a WeakSet of registered synchronizers.
         # The WeakSet is passed to the Transaction constructor, because the
         # latter needs to call the synchronizers when it commits.
@@ -101,7 +109,12 @@
         txn = self._txns.get(tid)
         if txn is not None:
             txn.abort()
+
         synchs = self._synchs.get(tid)
+        if synchs is None:
+            from ZODB.utils import WeakSet
+            synchs = self._synchs[tid] = WeakSet()
+
         txn = self._txns[tid] = Transaction(synchs, self)
         _new_transaction(txn, synchs)
         return txn
@@ -111,6 +124,9 @@
         txn = self._txns.get(tid)
         if txn is None:
             synchs = self._synchs.get(tid)
+            if synchs is None:
+                from ZODB.utils import WeakSet
+                synchs = self._synchs[tid] = WeakSet()
             txn = self._txns[tid] = Transaction(synchs, self)
         return txn
 
@@ -120,11 +136,10 @@
         del self._txns[tid]
 
     def registerSynch(self, synch):
-        from ZODB.utils import WeakSet
-
         tid = thread.get_ident()
         ws = self._synchs.get(tid)
         if ws is None:
+            from ZODB.utils import WeakSet
             ws = self._synchs[tid] = WeakSet()
         ws.add(synch)
 



More information about the Zodb-checkins mailing list