[Zope-Checkins] CVS: Zope/lib/python/ZODB - Connection.py:1.83 DB.py:1.47 ExportImport.py:1.16 ZApplication.py:1.12
Shane Hathaway
shane@zope.com
Fri, 17 Jan 2003 12:23:48 -0500
Update of /cvs-repository/Zope/lib/python/ZODB
In directory cvs.zope.org:/tmp/cvs-serv22844
Modified Files:
Connection.py DB.py ExportImport.py ZApplication.py
Log Message:
Merged shane-local-transactions-branch.
This change adds a new method, setLocalTransaction(), to the
Connection class. ZODB applications can call this method to bind
transactions to connections rather than threads. This is especially
useful for GUI applications, which often have only one thread but
multiple independent activities within that thread (generally one per
window). Thanks to Christian Reis for championing this feature.
Applications that take advantage of this feature should not use the
get_transaction() function. Until now, ZODB itself sometimes assumed
get_transaction() was the only way to get the transaction. Minor
corrections have been added. The ZODB test suite, on the other hand,
can continue to use get_transaction(), since it is free to assume that
transactions are bound to threads.
=== Zope/lib/python/ZODB/Connection.py 1.82 => 1.83 ===
--- Zope/lib/python/ZODB/Connection.py:1.82 Wed Jan 15 16:29:26 2003
+++ Zope/lib/python/ZODB/Connection.py Fri Jan 17 12:23:14 2003
@@ -22,6 +22,7 @@
from zLOG import LOG, ERROR, BLATHER, WARNING
from coptimizations import new_persistent_id
from ConflictResolution import ResolvedSerial
+from Transaction import Transaction, get_transaction
from cPickle import Unpickler, Pickler
from cStringIO import StringIO
@@ -55,6 +56,7 @@
_debug_info=()
_opened=None
_code_timestamp = 0
+ _transaction = None
# Experimental. Other connections can register to be closed
# when we close by putting something here.
@@ -80,6 +82,19 @@
self._load_count = 0 # Number of objects unghosted
self._store_count = 0 # Number of objects stored
+ def getTransaction(self):
+ t = self._transaction
+ if t is None:
+ # Fall back to thread-bound transactions
+ t = get_transaction()
+ return t
+
+ def setLocalTransaction(self):
+ """Use a transaction bound to the connection rather than the thread"""
+ if self._transaction is None:
+ self._transaction = Transaction()
+ return self._transaction
+
def _cache_items(self):
# find all items on the lru list
items = self._cache.lru_items()
@@ -269,7 +284,7 @@
if self.__onCommitActions is None:
self.__onCommitActions = []
self.__onCommitActions.append((method_name, args, kw))
- get_transaction().register(self)
+ self.getTransaction().register(self)
def commit(self, object, transaction):
if object is self:
@@ -484,7 +499,7 @@
assert object._p_jar is self
# XXX Figure out why this assert causes test failures
# assert object._p_oid is not None
- get_transaction().register(object)
+ self.getTransaction().register(object)
def root(self):
return self['\0\0\0\0\0\0\0\0']
@@ -516,7 +531,7 @@
# XXX Need unit tests for _p_independent.
if self._invalid(oid):
if not hasattr(object.__class__, '_p_independent'):
- get_transaction().register(self)
+ self.getTransaction().register(self)
raise ReadConflictError(object=object)
invalid = 1
else:
@@ -544,7 +559,7 @@
except KeyError:
pass
else:
- get_transaction().register(self)
+ self.getTransaction().register(self)
raise ConflictError(object=object)
except ConflictError:
@@ -695,7 +710,7 @@
self._db.finish_invalidation()
def sync(self):
- get_transaction().abort()
+ self.getTransaction().abort()
sync=getattr(self._storage, 'sync', 0)
if sync != 0: sync()
self._cache.invalidate(self._invalidated)
@@ -726,7 +741,7 @@
new._p_oid=oid
new._p_jar=self
new._p_changed=1
- get_transaction().register(new)
+ self.getTransaction().register(new)
self._cache[oid]=new
class tConnection(Connection):
=== Zope/lib/python/ZODB/DB.py 1.46 => 1.47 ===
--- Zope/lib/python/ZODB/DB.py:1.46 Tue Dec 3 12:40:56 2002
+++ Zope/lib/python/ZODB/DB.py Fri Jan 17 12:23:14 2003
@@ -19,7 +19,7 @@
import cPickle, cStringIO, sys, POSException, UndoLogCompatible
from Connection import Connection
from bpthread import allocate_lock
-from Transaction import Transaction
+from Transaction import Transaction, get_transaction
from referencesf import referencesf
from time import time, ctime
from zLOG import LOG, ERROR
@@ -153,8 +153,10 @@
self._temps=t
finally: self._r()
- def abortVersion(self, version):
- AbortVersion(self, version)
+ def abortVersion(self, version, transaction=None):
+ if transaction is None:
+ transaction = get_transaction()
+ transaction.register(AbortVersion(self, version))
def cacheDetail(self):
"""Return information on objects in the various caches
@@ -248,8 +250,10 @@
def close(self):
self._storage.close()
- def commitVersion(self, source, destination=''):
- CommitVersion(self, source, destination)
+ def commitVersion(self, source, destination='', transaction=None):
+ if transaction is None:
+ transaction = get_transaction()
+ transaction.register(CommitVersion(self, source, destination))
def exportFile(self, oid, file=None):
raise 'Not yet implemented'
@@ -542,7 +546,7 @@
def cacheStatistics(self): return () # :(
- def undo(self, id):
+ def undo(self, id, transaction=None):
storage=self._storage
try: supportsTransactionalUndo = storage.supportsTransactionalUndo
except AttributeError:
@@ -552,7 +556,9 @@
if supportsTransactionalUndo:
# new style undo
- TransactionalUndo(self, id)
+ if transaction is None:
+ transaction = get_transaction()
+ transaction.register(TransactionalUndo(self, id))
else:
# fall back to old undo
for oid in storage.undo(id):
@@ -576,7 +582,6 @@
self.tpc_vote=s.tpc_vote
self.tpc_finish=s.tpc_finish
self._sortKey=s.sortKey
- get_transaction().register(self)
def sortKey(self):
return "%s:%s" % (self._sortKey(), id(self))
@@ -613,9 +618,9 @@
in cooperation with a transaction manager.
"""
- # I'm lazy. I'm reusing __init__ and abort and reusing the
- # version attr for the transavtion id. There's such a strong
- # similarity of rythm, that I think it's justified.
+ # I (Jim) am lazy. I'm reusing __init__ and abort and reusing the
+ # version attr for the transaction id. There's such a strong
+ # similarity of rhythm that I think it's justified.
def commit(self, reallyme, t):
db=self._db
=== Zope/lib/python/ZODB/ExportImport.py 1.15 => 1.16 ===
--- Zope/lib/python/ZODB/ExportImport.py:1.15 Mon Dec 2 17:04:37 2002
+++ Zope/lib/python/ZODB/ExportImport.py Fri Jan 17 12:23:14 2003
@@ -76,7 +76,7 @@
return customImporters[magic](self, file, clue)
raise POSException.ExportError, 'Invalid export header'
- t = get_transaction()
+ t = self.getTransaction()
if clue: t.note(clue)
return_oid_list = []
=== Zope/lib/python/ZODB/ZApplication.py 1.11 => 1.12 ===
--- Zope/lib/python/ZODB/ZApplication.py:1.11 Wed Aug 14 18:07:09 2002
+++ Zope/lib/python/ZODB/ZApplication.py Fri Jan 17 12:23:14 2003
@@ -31,7 +31,7 @@
root=conn.root()
if not root.has_key(name):
root[name]=klass()
- get_transaction().commit()
+ conn.getTransaction().commit()
conn.close()
self._klass=klass