Commit 7b6cdb78 authored by Barry Warsaw's avatar Barry Warsaw

_finish(): Fixed bug that was breaking abortVersion(). The lrevid

(a.k.a. pickle pointer) is usually tid but not always!  In fact, it's
specifically not tid when we've aborted a version -- it'll be the
pointer to the pickle of the last non-version revision.  We were
actually setting up lrevid correctly, but writing tid to the metadata
record.  This fix writes lrevid instead.
parent a64b1c71
...@@ -4,7 +4,7 @@ See Minimal.py for an implementation of Berkeley storage that does not support ...@@ -4,7 +4,7 @@ See Minimal.py for an implementation of Berkeley storage that does not support
undo or versioning. undo or versioning.
""" """
# $Revision: 1.8 $ # $Revision: 1.9 $
__version__ = '0.1' __version__ = '0.1'
import struct import struct
...@@ -268,7 +268,8 @@ class Full(BerkeleyBase): ...@@ -268,7 +268,8 @@ class Full(BerkeleyBase):
refcount = utils.p64(utils.U64(refcount) + 1) refcount = utils.p64(utils.U64(refcount) + 1)
self._refcounts.put(roid, refcount, txn=txn) self._refcounts.put(roid, refcount, txn=txn)
# Update the metadata table # Update the metadata table
self._metadata.put(key, vid+nvrevid+tid+prevrevid, txn=txn) self._metadata.put(key, vid+nvrevid+lrevid+prevrevid,
txn=txn)
# If we're in a real version, update this table too. This # If we're in a real version, update this table too. This
# ends up putting multiple copies of the vid/oid records # ends up putting multiple copies of the vid/oid records
# in the table, but it's easier to weed those out later # in the table, but it's easier to weed those out later
...@@ -344,14 +345,18 @@ class Full(BerkeleyBase): ...@@ -344,14 +345,18 @@ class Full(BerkeleyBase):
# information for undo. # information for undo.
while rec: while rec:
oid = rec[1] # ignore the key oid = rec[1] # ignore the key
rec = c.next()
if oids.has_key(oid):
# We've already dealt with this oid...
continue
revid = self._serials[oid] revid = self._serials[oid]
meta = self._metadata[oid+revid] meta = self._metadata[oid+revid]
curvid, nvrevid = struct.unpack('8s8s', meta[:16]) curvid, nvrevid = struct.unpack('8s8s', meta[:16])
# Make sure that the vid in the metadata record is the same as # Make sure that the vid in the metadata record is the same as
# the vid we sucked out of the vids table, otherwise we've got # the vid we sucked out of the vids table.
# an internal database inconsistency.
if curvid <> vid: if curvid <> vid:
raise InternalInconsistencyError raise POSException.VersionError(
'aborting a non-current version')
if nvrevid == zero: if nvrevid == zero:
# This object was created in the version, so we don't need # This object was created in the version, so we don't need
# to do anything about it. # to do anything about it.
...@@ -369,8 +374,6 @@ class Full(BerkeleyBase): ...@@ -369,8 +374,6 @@ class Full(BerkeleyBase):
self._commitlog.write_nonversion_object(oid, lrevid, revid) self._commitlog.write_nonversion_object(oid, lrevid, revid)
# Remember to return the oid... # Remember to return the oid...
oids[oid] = 1 oids[oid] = 1
# ...and get the next record for this vid
rec = c.next()
# We've now processed all the objects on the discarded version, so # We've now processed all the objects on the discarded version, so
# write this to the commit log and return the list of oids to # write this to the commit log and return the list of oids to
# invalidate. # invalidate.
...@@ -598,6 +601,8 @@ class Full(BerkeleyBase): ...@@ -598,6 +601,8 @@ class Full(BerkeleyBase):
return self._serial return self._serial
def transactionalUndo(self, tid, transaction): def transactionalUndo(self, tid, transaction):
# FIXME: what if we undo an abortVersion or commitVersion, don't we
# need to re-populate the currentVersions table?
if transaction is not self._transaction: if transaction is not self._transaction:
raise POSException.StorageTransactionError(self, transaction) raise POSException.StorageTransactionError(self, transaction)
......
...@@ -4,7 +4,7 @@ See Minimal.py for an implementation of Berkeley storage that does not support ...@@ -4,7 +4,7 @@ See Minimal.py for an implementation of Berkeley storage that does not support
undo or versioning. undo or versioning.
""" """
# $Revision: 1.8 $ # $Revision: 1.9 $
__version__ = '0.1' __version__ = '0.1'
import struct import struct
...@@ -268,7 +268,8 @@ class Full(BerkeleyBase): ...@@ -268,7 +268,8 @@ class Full(BerkeleyBase):
refcount = utils.p64(utils.U64(refcount) + 1) refcount = utils.p64(utils.U64(refcount) + 1)
self._refcounts.put(roid, refcount, txn=txn) self._refcounts.put(roid, refcount, txn=txn)
# Update the metadata table # Update the metadata table
self._metadata.put(key, vid+nvrevid+tid+prevrevid, txn=txn) self._metadata.put(key, vid+nvrevid+lrevid+prevrevid,
txn=txn)
# If we're in a real version, update this table too. This # If we're in a real version, update this table too. This
# ends up putting multiple copies of the vid/oid records # ends up putting multiple copies of the vid/oid records
# in the table, but it's easier to weed those out later # in the table, but it's easier to weed those out later
...@@ -344,14 +345,18 @@ class Full(BerkeleyBase): ...@@ -344,14 +345,18 @@ class Full(BerkeleyBase):
# information for undo. # information for undo.
while rec: while rec:
oid = rec[1] # ignore the key oid = rec[1] # ignore the key
rec = c.next()
if oids.has_key(oid):
# We've already dealt with this oid...
continue
revid = self._serials[oid] revid = self._serials[oid]
meta = self._metadata[oid+revid] meta = self._metadata[oid+revid]
curvid, nvrevid = struct.unpack('8s8s', meta[:16]) curvid, nvrevid = struct.unpack('8s8s', meta[:16])
# Make sure that the vid in the metadata record is the same as # Make sure that the vid in the metadata record is the same as
# the vid we sucked out of the vids table, otherwise we've got # the vid we sucked out of the vids table.
# an internal database inconsistency.
if curvid <> vid: if curvid <> vid:
raise InternalInconsistencyError raise POSException.VersionError(
'aborting a non-current version')
if nvrevid == zero: if nvrevid == zero:
# This object was created in the version, so we don't need # This object was created in the version, so we don't need
# to do anything about it. # to do anything about it.
...@@ -369,8 +374,6 @@ class Full(BerkeleyBase): ...@@ -369,8 +374,6 @@ class Full(BerkeleyBase):
self._commitlog.write_nonversion_object(oid, lrevid, revid) self._commitlog.write_nonversion_object(oid, lrevid, revid)
# Remember to return the oid... # Remember to return the oid...
oids[oid] = 1 oids[oid] = 1
# ...and get the next record for this vid
rec = c.next()
# We've now processed all the objects on the discarded version, so # We've now processed all the objects on the discarded version, so
# write this to the commit log and return the list of oids to # write this to the commit log and return the list of oids to
# invalidate. # invalidate.
...@@ -598,6 +601,8 @@ class Full(BerkeleyBase): ...@@ -598,6 +601,8 @@ class Full(BerkeleyBase):
return self._serial return self._serial
def transactionalUndo(self, tid, transaction): def transactionalUndo(self, tid, transaction):
# FIXME: what if we undo an abortVersion or commitVersion, don't we
# need to re-populate the currentVersions table?
if transaction is not self._transaction: if transaction is not self._transaction:
raise POSException.StorageTransactionError(self, transaction) raise POSException.StorageTransactionError(self, transaction)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment