[Zodb-checkins] CVS: Zope3/src/transaction/tests -
test_SampleDataManager.py:1.1.2.2
Jim Fulton
jim at zope.com
Thu Jan 22 14:44:27 EST 2004
Update of /cvs-repository/Zope3/src/transaction/tests
In directory cvs.zope.org:/tmp/cvs-serv30783
Modified Files:
Tag: zope3-zodb3-devel-branch
test_SampleDataManager.py
Log Message:
This test module exists for two reasons:
- To provide details of the ZODB 4 transaction protocol, as I
understand them.
- To provide a test stub that can be used to test the zodb3
data-manager adapter.
=== Zope3/src/transaction/tests/test_SampleDataManager.py 1.1.2.1 => 1.1.2.2 ===
--- Zope3/src/transaction/tests/test_SampleDataManager.py:1.1.2.1 Thu Jan 22 11:32:20 2004
+++ Zope3/src/transaction/tests/test_SampleDataManager.py Thu Jan 22 14:44:26 2004
@@ -19,72 +19,72 @@
class DataManager(object):
"""Sample data manager
- This class provides a trivial data-manager implementation and doc
- strings to illustrate the the protocol and to provide a tool for
- writing tests.
+ This class provides a trivial data-manager implementation and doc
+ strings to illustrate the the protocol and to provide a tool for
+ writing tests.
- Our sample data manager has state that is updated through an inc
- method and through transaction operations.
+ Our sample data manager has state that is updated through an inc
+ method and through transaction operations.
- When we create a sample data manager:
+ When we create a sample data manager:
- >>> dm = DataManager()
+ >>> dm = DataManager()
- It has two bits of state, state:
+ It has two bits of state, state:
- >>> dm.state
- 0
+ >>> dm.state
+ 0
- and delta:
+ and delta:
- >>> dm.delta
- 0
+ >>> dm.delta
+ 0
- Both of which are initialized to 0. state is meant to model
- committed state, while delta represents tentative changes within a
- transaction. We change the state by calling inc:
+ Both of which are initialized to 0. state is meant to model
+ committed state, while delta represents tentative changes within a
+ transaction. We change the state by calling inc:
- >>> dm.inc()
+ >>> dm.inc()
- which updates delta:
+ which updates delta:
- >>> dm.delta
- 1
+ >>> dm.delta
+ 1
- but state isn't changed until we commit the transaction:
+ but state isn't changed until we commit the transaction:
- >>> dm.state
- 0
+ >>> dm.state
+ 0
- To commit the changes, we use 2-phase commit. We execute the first
- stage by calling prepare. We need to pass a transation. Our
- sample data managers don't really use the transactions for much,
- so we'll be lazy and use integers for transactions:
+ To commit the changes, we use 2-phase commit. We execute the first
+ stage by calling prepare. We need to pass a transation. Our
+ sample data managers don't really use the transactions for much,
+ so we'll be lazy and use strings for transactions:
- >>> t1 = 1
- >>> dm.prepare(t1)
+ >>> t1 = '1'
+ >>> dm.prepare(t1)
- The sample data manager updates the state when we call prepare:
+ The sample data manager updates the state when we call prepare:
- >>> dm.state
- 1
- >>> dm.delta
- 1
+ >>> dm.state
+ 1
+ >>> dm.delta
+ 1
- This is mainly so we can detect some affect of calling the methods.
+ This is mainly so we can detect some affect of calling the methods.
- Now if we call commit:
+ Now if we call commit:
- >>> dm.commit(t1)
+ >>> dm.commit(t1)
- Our changes are"permanent". The state reflects the changes and the
- delta has been reset to 0.
+ Our changes are"permanent". The state reflects the changes and the
+ delta has been reset to 0.
- >>> dm.state
- 1
- >>> dm.delta
- 0
- """
+ >>> dm.state
+ 1
+ >>> dm.delta
+ 0
+ """
def __init__(self):
self.state = 0
@@ -101,35 +101,44 @@
>>> dm = DataManager()
>>> dm.inc()
- >>> dm.prepare(1)
- >>> dm.commit(1)
+ >>> t1 = '1'
+ >>> dm.prepare(t1)
+ >>> dm.commit(t1)
>>> dm.state
1
>>> dm.inc()
- >>> dm.prepare(2)
- >>> dm.abort(2)
+ >>> t2 = '2'
+ >>> dm.prepare(t2)
+ >>> dm.abort(t2)
>>> dm.state
1
It is en error to call prepare more than once without an intervening
commit or abort:
- >>> dm.prepare(1)
-Traceback (most recent call last):
+ >>> dm.prepare(t1)
-TypeError: Already prepared
+ >>> dm.prepare(t1)
+ Traceback (most recent call last):
+ ...
+ TypeError: Already prepared
+
+ >>> dm.prepare(t2)
+ Traceback (most recent call last):
+ ...
+ TypeError: Already prepared
- >>> dm.prepare(1)
- >>> dm.prepare(2)
-
- >>> dm.abort(1)
+ >>> dm.abort(t1)
If there was a preceeding savepoint, the transaction must match:
- >>> rollback = dm.savepoint(1)
- >>> rollback.prepare(2)
+ >>> rollback = dm.savepoint(t1)
+ >>> dm.prepare(t2)
+ Traceback (most recent call last):
+ ,,,
+ TypeError: ('Transaction missmatch', '2', '1')
- >>> rollback.prepare(1)
+ >>> dm.prepare(t1)
"""
if self.prepared:
@@ -148,15 +157,121 @@
def abort(self, transaction):
"""Abort a transaction
+ The abort method can be called before two-phase commit to
+ throw away work done in the transaction:
+
+ >>> dm = DataManager()
+ >>> dm.inc()
+ >>> dm.state, dm.delta
+ (0, 1)
+ >>> t1 = '1'
+ >>> dm.abort(t1)
+ >>> dm.state, dm.delta
+ (0, 0)
+
+ The abort method also throws away work done in savepoints:
+
+ >>> dm.inc()
+ >>> r = dm.savepoint(t1)
+ >>> dm.inc()
+ >>> r = dm.savepoint(t1)
+ >>> dm.state, dm.delta
+ (0, 2)
+ >>> dm.abort(t1)
+ >>> dm.state, dm.delta
+ (0, 0)
+
+ If savepoints are used, abort must be passed the same
+ transaction:
+
+ >>> dm.inc()
+ >>> r = dm.savepoint(t1)
+ >>> t2 = '2'
+ >>> dm.abort(t2)
+ Traceback (most recent call last):
+ ...
+ TypeError: ('Transaction missmatch', '2', '1')
+
+ >>> dm.abort(t1)
+
+ The abort method is also used to abort a two-phase commit:
+
+ >>> dm.inc()
+ >>> dm.state, dm.delta
+ (0, 1)
+ >>> dm.prepare(t1)
+ >>> dm.state, dm.delta
+ (1, 1)
+ >>> dm.abort(t1)
+ >>> dm.state, dm.delta
+ (0, 0)
+
+ Of course, the transactions passed to prepare and abort must
+ match:
+
+ >>> dm.prepare(t1)
+ >>> dm.abort(t2)
+ Traceback (most recent call last):
+ ...
+ TypeError: ('Transaction missmatch', '2', '1')
+
+ >>> dm.abort(t1)
+
+
"""
self._checkTransaction(transaction)
if self.transaction is not None:
- self.state -= self.delta
self.transaction = None
+
+ if self.prepared:
+ self.state -= self.delta
+ self.prepared = False
+
self.delta = 0
- self.prepared = False
def commit(self, transaction):
+ """Complete two-phase commit
+
+ >>> dm = DataManager()
+ >>> dm.state
+ 0
+ >>> dm.inc()
+
+ We start two-phase commit by calling prepare:
+
+ >>> t1 = '1'
+ >>> dm.prepare(t1)
+
+ We complete it by calling commit:
+
+ >>> dm.commit(t1)
+ >>> dm.state
+ 1
+
+ It is an error ro call commit without calling prepare first:
+
+ >>> dm.inc()
+ >>> t2 = '2'
+ >>> dm.commit(t2)
+ Traceback (most recent call last):
+ ...
+ TypeError: Not prepared to commit
+
+ >>> dm.prepare(t2)
+ >>> dm.commit(t2)
+
+ If course, the transactions given to prepare and commit must
+ be the same:
+
+ >>> dm.inc()
+ >>> t3 = '3'
+ >>> dm.prepare(t3)
+ >>> dm.commit(t2)
+ Traceback (most recent call last):
+ ...
+ TypeError: ('Transaction missmatch', '2', '3')
+
+ """
if not self.prepared:
raise TypeError('Not prepared to commit')
self._checkTransaction(transaction)
@@ -165,6 +280,105 @@
self.prepared = False
def savepoint(self, transaction):
+ """Provide the ability to rollback transaction state
+
+ Savepoints provide a way to:
+
+ - Save partial transaction work. For some data managers, this
+ could allow resources to be used more efficiently.
+
+ - Provide the ability to revert state to a point in a
+ transaction without aborting the entire transaction. In
+ other words, savepoints support partial aborts.
+
+ Savepoints don't use two-phase commit. If there are errors in
+ setting or rolling back to savepoints, the application should
+ abort the containing transaction. This is *not* the
+ responsibility of the data manager.
+
+ Savepoints are always associated with a transaction. Any work
+ done in a savepoint's transaction is tentative until the
+ transaction is committed using two-phase commit.
+
+ >>> dm = DataManager()
+ >>> dm.inc()
+ >>> t1 = '1'
+ >>> r = dm.savepoint(t1)
+ >>> dm.state, dm.delta
+ (0, 1)
+ >>> dm.inc()
+ >>> dm.state, dm.delta
+ (0, 2)
+ >>> r.rollback()
+ >>> dm.state, dm.delta
+ (0, 1)
+ >>> dm.prepare(t1)
+ >>> dm.commit(t1)
+ >>> dm.state, dm.delta
+ (1, 0)
+
+ Savepoints must have the same transaction:
+
+ >>> r1 = dm.savepoint(t1)
+ >>> dm.state, dm.delta
+ (1, 0)
+ >>> dm.inc()
+ >>> dm.state, dm.delta
+ (1, 1)
+ >>> t2 = '2'
+ >>> r2 = dm.savepoint(t2)
+ Traceback (most recent call last):
+ ...
+ TypeError: ('Transaction missmatch', '2', '1')
+
+ >>> r2 = dm.savepoint(t1)
+ >>> dm.inc()
+ >>> dm.state, dm.delta
+ (1, 2)
+
+ If we rollback to an earlier savepoint, we discard all work
+ done later:
+
+ >>> r1.rollback()
+ >>> dm.state, dm.delta
+ (1, 0)
+
+ and we can no longer rollback to the later savepoint:
+
+ >>> r2.rollback()
+ Traceback (most recent call last):
+ ...
+ TypeError: ('Attempt to roll back to invalid save point', 3, 2)
+
+ We can roll back to a savepoint as often as we like:
+
+ >>> r1.rollback()
+ >>> r1.rollback()
+ >>> r1.rollback()
+ >>> dm.state, dm.delta
+ (1, 0)
+
+ >>> dm.inc()
+ >>> dm.inc()
+ >>> dm.inc()
+ >>> dm.state, dm.delta
+ (1, 3)
+ >>> r1.rollback()
+ >>> dm.state, dm.delta
+ (1, 0)
+
+ But we can't rollback to a savepoint after it has been
+ committed:
+
+ >>> dm.prepare(t1)
+ >>> dm.commit(t1)
+
+ >>> r1.rollback()
+ Traceback (most recent call last):
+ ...
+ TypeError: Attempt to rollback stale rollback
+
+ """
if self.prepared:
raise TypeError("Can't get savepoint during two-phase commit")
self._checkTransaction(transaction)
@@ -177,18 +391,17 @@
def __init__(self, dm):
self.dm = dm
self.sp = dm.sp
- self.state = dm.state
+ self.delta = dm.delta
self.transaction = dm.transaction
def rollback(self):
- if self.transaction is not self.sp.transaction:
+ if self.transaction is not self.dm.transaction:
raise TypeError("Attempt to rollback stale rollback")
if self.dm.sp < self.sp:
raise TypeError("Attempt to roll back to invalid save point",
self.sp, self.dm.sp)
self.dm.sp = self.sp
- self.dm.state = self.state
- self.dm.delta = 0
+ self.dm.delta = self.delta
def test_suite():
More information about the Zodb-checkins
mailing list