[Zope-Checkins] CVS: Zope3/src/ZODB/tests - testmvcc.py:1.2
test_storage.py:1.3
Jeremy Hylton
jeremy at zope.com
Thu Mar 11 17:20:03 EST 2004
Update of /cvs-repository/Zope3/src/ZODB/tests
In directory cvs.zope.org:/tmp/cvs-serv6919/tests
Modified Files:
testmvcc.py test_storage.py
Log Message:
Begin a framework for testing invalidation-during-load scenario.
=== Zope3/src/ZODB/tests/testmvcc.py 1.1 => 1.2 ===
--- Zope3/src/ZODB/tests/testmvcc.py:1.1 Thu Mar 11 15:11:28 2004
+++ Zope3/src/ZODB/tests/testmvcc.py Thu Mar 11 17:20:02 2004
@@ -57,6 +57,9 @@
>>> cn2 = db.open()
>>> txn2 = cn2.setLocalTransaction()
+Connection high-water mark
+--------------------------
+
The ZODB Connection tracks a transaction high-water mark, which
represents the latest transaction id that can be read by the current
transaction and still present a consistent view of the database. When
@@ -86,13 +89,15 @@
>>> cn.sync()
>>> cn._txn_time
+Basic functionality
+-------------------
+
The next bit of code includes a simple MVCC test. One transaction
will begin and modify "a." The other transaction will then modify "b"
and commit.
>>> r1 = cn1.root()
>>> r1["a"].value = 2
-
>>> cn1.getTransaction().commit()
>>> txn = db.lastTransaction()
@@ -132,6 +137,161 @@
Traceback (most recent call last):
...
ConflictError: database conflict error (oid 0000000000000001, class ZODB.tests.MinPO.MinPO)
+
+The failed commit aborted the current transaction, so we can try
+again. This example will demonstrate that we can commit a transaction
+if we don't modify the object that isn't current.
+
+>>> cn2._txn_time
+
+>>> r1 = cn1.root()
+>>> r1["a"].value = 3
+>>> cn1.getTransaction().commit()
+>>> txn = db.lastTransaction()
+>>> cn2._txn_time == txn
+True
+
+>>> r2["b"].value = r2["a"].value + 1
+>>> r2["b"].value
+3
+>>> txn2.commit()
+>>> cn2._txn_time
+
+Object cache
+------------
+
+A Connection keeps objects in its cache so that multiple database
+references will always point to the same Python object. At
+transaction boundaries, objects modified by other transactions are
+ghostified so that the next transaction doesn't see stale state. We
+need to be sure the non-current objects loaded by MVCC are always
+ghosted. It should be trivial, because MVCC is only used when an
+invalidation has been received for an object.
+
+First get the database back in an initial state.
+
+>>> cn1.sync()
+>>> r1["a"].value = 0
+>>> r1["b"].value = 0
+>>> cn1.getTransaction().commit()
+
+>>> cn2.sync()
+>>> r2["a"].value
+0
+>>> r2["b"].value = 1
+>>> cn2.getTransaction().commit()
+
+>>> r1["b"].value
+0
+>>> cn1.sync()
+>>> r1["b"]._p_state
+-1
+
+Closing the connection and commit a transaction should have the same effect.
+
+>>> def testit():
+... cn1.sync()
+... r1["a"].value = 0
+... r1["b"].value = 0
+... cn1.getTransaction().commit()
+... cn2.sync()
+... r2["b"].value = 1
+... cn2.getTransaction().commit()
+
+>>> testit()
+>>> r1["a"].value = 1
+>>> cn1.getTransaction().commit()
+>>> r1["b"]._p_state
+-1
+
+When a connection is closed, it is saved by the database. It will be
+reused by the next open() call (along with its object cache).
+
+>>> testit()
+>>> r1["a"].value = 1
+>>> cn1.close()
+>>> cn3 = db.open()
+>>> cn1 is cn3
+True
+>>> cn1 = cn3
+>>> r1 = cn1.root()
+
+It's not just that every object is a ghost. The root was in the
+cache, so our first reference to it doesn't return a ghost.
+
+>>> r1._p_state
+0
+>>> r1["b"]._p_state
+-1
+
+Late invalidation
+-----------------
+
+The combination of ZEO and MVCC adds more complexity. Since
+invalidations are delivered asynchronously by ZEO, it is possible for
+an invalidation to arrive just after a request to load the invalidated
+object is sent. The connection can't use the just-loaded data,
+because the invalidation arrived first. The complexity for MVCC is
+that it must check for invalidated objects after it has loaded them,
+just in case.
+
+Rather than add all the complexity of ZEO to these tests, the
+MinimalMemoryStorage has a hook. We'll write a subclass that will
+deliver an invalidation when it loads an object. The hook allows us
+to test the Connection code.
+
+>>> class TestStorage(MinimalMemoryStorage):
+... def __init__(self):
+... self.hooked = {}
+... self.count = 0
+... super(TestStorage, self).__init__()
+... def registerDB(self, db, limit):
+... self.db = db
+... def hook(self, oid, tid, version):
+... if oid in self.hooked:
+... self.db.invalidate(tid, {oid:1})
+... self.count += 1
+
+Now we'll repeat all the setup that was done earlier.
+
+>>> ts = TestStorage()
+>>> db = DB(ts)
+>>> cn1 = db.open()
+>>> txn1 = cn1.setLocalTransaction()
+>>> r1 = cn1.root()
+>>> r1["a"] = MinPO(0)
+>>> r1["b"] = MinPO(0)
+>>> cn1.getTransaction().commit()
+>>> cn1.cacheMinimize()
+
+>>> oid = r1["b"]._p_oid
+>>> ts.hooked[oid] = 1
+
+This test isn't quite rght yet, because it gets a ReadConflictError
+instead of getting a non-current revision. Stil, it demonstrates that
+the basic mechanism for sending an invalidation during a load works.
+
+>>> oid in cn1._invalidated
+False
+>>> r1["b"]._p_state
+-1
+>>> r1["b"]._p_activate()
+Traceback (most recent call last):
+ ...
+ReadConflictError: database read conflict error (oid 0000000000000002, class ZODB.tests.MinPO.MinPO)
+>>> oid in cn1._invalidated
+True
+>>> ts.count
+1
+
+_p_independent() still has the desired effect.
+
+We still get a non-current version if the invalidation occurs while we
+are loading the current revision. Can that happen without ZEO?
+
+Error cases:
+- storage doesn't have an earlier revision
+- MVCC returns current revision
"""
=== Zope3/src/ZODB/tests/test_storage.py 1.2 => 1.3 ===
--- Zope3/src/ZODB/tests/test_storage.py:1.2 Thu Mar 11 15:16:06 2004
+++ Zope3/src/ZODB/tests/test_storage.py Thu Mar 11 17:20:02 2004
@@ -11,6 +11,14 @@
# FOR A PARTICULAR PURPOSE
#
##############################################################################
+"""A storage used for unittests.
+
+The primary purpose of this module is to have a minimal multi-version
+storage to use for unit tests. MappingStorage isn't sufficient.
+Since even a minimal storage has some complexity, we run standard
+storage tests against the test storage.
+"""
+
import bisect
import threading
import unittest
@@ -41,8 +49,9 @@
"""Simple in-memory storage that supports revisions.
This storage is needed to test multi-version concurrency control.
- It is similar to MappingStorage, but keeps multiple revisions.
- It does not support versions.
+ It is similar to MappingStorage, but keeps multiple revisions. It
+ does not support versions. It doesn't implement operations like
+ pack(), because they aren't necessary for testing.
"""
def __init__(self):
@@ -55,6 +64,10 @@
def isCurrent(self, oid, serial):
return serial == self._cur[oid]
+ def hook(self, oid, tid, version):
+ # A hook for testing
+ pass
+
def __len__(self):
return len(self._index)
@@ -66,6 +79,7 @@
try:
assert not version
tid = self._cur[oid]
+ self.hook(oid, tid, version)
return self._index[(oid, tid)], tid, ""
finally:
self._lock_release()
More information about the Zope-Checkins
mailing list