[Zope-Checkins] CVS: ZODB3/ZODB - utils.py:1.14 POSException.py:1.15 FileStorage.py:1.119 Connection.py:1.80 ConflictResolution.py:1.17
Jeremy Hylton
jeremy@zope.com
Tue, 3 Dec 2002 13:36:30 -0500
Update of /cvs-repository/ZODB3/ZODB
In directory cvs.zope.org:/tmp/cvs-serv21022/ZODB
Modified Files:
utils.py POSException.py FileStorage.py Connection.py
ConflictResolution.py
Log Message:
Finish synchronizing FileStorage and POSException between ZODB3 and ZODB4.
There are few differences between the two files, except for necessary
differences because of API changes. This change involves backporting
the new UndoError style from ZODB4.
=== ZODB3/ZODB/utils.py 1.13 => 1.14 ===
--- ZODB3/ZODB/utils.py:1.13 Tue Nov 5 16:50:50 2002
+++ ZODB3/ZODB/utils.py Tue Dec 3 13:36:29 2002
@@ -17,6 +17,8 @@
from struct import pack, unpack
+z64 = '\0'*8
+
if sys.version >= (2, 2):
# Note that the distinction between ints and longs is blurred in
=== ZODB3/ZODB/POSException.py 1.14 => 1.15 ===
--- ZODB3/ZODB/POSException.py:1.14 Thu Sep 5 06:19:40 2002
+++ ZODB3/ZODB/POSException.py Tue Dec 3 13:36:29 2002
@@ -11,23 +11,28 @@
# FOR A PARTICULAR PURPOSE
#
##############################################################################
-"""BoboPOS-defined exceptions
+"""ZODB-defined exceptions
$Id$"""
-__version__ = '$Revision$'.split()[-2:][0]
-from string import join
from types import StringType, DictType
-from ZODB import utils
+import ZODB.utils
-class POSError(Exception):
+def _fmt_oid(oid):
+ return "%016x" % ZODB.utils.u64(oid)
+
+def _fmt_undo(oid, reason):
+ s = reason and (": %s" % reason) or ""
+ return "Undo error %s%s" % (_fmt_oid(oid), s)
+
+class POSError(StandardError):
"""Persistent object system error."""
class POSKeyError(KeyError, POSError):
"""Key not found in database."""
def __str__(self):
- return "%016x" % utils.U64(self.args[0])
+ return _fmt_oid(self.args[0])
class TransactionError(POSError):
"""An error occured due to normal transaction processing."""
@@ -78,12 +83,12 @@
def __str__(self):
extras = []
if self.oid:
- extras.append("oid %016x" % utils.U64(self.oid))
+ extras.append("oid %s" % _fmt_oid(self.oid))
if self.class_name:
extras.append("class %s" % self.class_name)
if self.serials:
- extras.append("serial was %016x, now %016x" %
- tuple(map(utils.U64, self.serials)))
+ extras.append("serial was %s, now %s" %
+ tuple(map(_fmt_oid, self.serials)))
if extras:
return "%s (%s)" % (self.message, ", ".join(extras))
else:
@@ -104,27 +109,8 @@
def get_serials(self):
return self.serials
-class DanglingReferenceError(TransactionError):
- """The transaction stored an object A containing a reference to another
- object B, but B does not exist
-
- Instance attributes:
-
- Aoid: oid of the object being written
-
- Boid: referenced oid that does not have a corresponding object
- """
-
- def __init__(self,Aoid,Boid):
- self.Aoid = Aoid
- self.Boid = Boid
-
- def __str__(self):
- return "from %r to %r" % (self.Aoid,self.Boid)
-
-
class ReadConflictError(ConflictError):
- """A conflict was detected at read time.
+ """Conflict detected when object was loaded.
An attempt was made to read an object that has changed in another
transaction (eg. another thread or process).
@@ -144,6 +130,27 @@
ConflictError.__init__(self, message="BTrees conflict error")
self.btree = btree_args
+class DanglingReferenceError(TransactionError):
+ """An object has a persistent reference to a missing object.
+
+ If an object is stored and it has a reference to another object
+ that does not exist (for example, it was deleted by pack), this
+ exception may be raised. Whether a storage supports this feature,
+ it a quality of implementation issue.
+
+ Instance attributes:
+ referer: oid of the object being written
+ missing: referenced oid that does not have a corresponding object
+ """
+
+ def __init__(self, Aoid, Boid):
+ self.referer = Aoid
+ self.missing = Boid
+
+ def __str__(self):
+ return "from %s to %s" % (_fmt_oid(self.referer),
+ _fmt_oid(self.missing))
+
class VersionError(POSError):
"""An error in handling versions occurred."""
@@ -159,26 +166,24 @@
class UndoError(POSError):
"""An attempt was made to undo a non-undoable transaction."""
- def __init__(self, *reason):
- if len(reason) == 1: reason=reason[0]
- self.__reason=reason
-
- def __repr__(self):
- reason=self.__reason
- if type(reason) is not DictType:
- if reason: return str(reason)
- return "non-undoable transaction"
- r=[]
- for oid, reason in reason.items():
- if reason:
- r.append("Couldn't undo change to %s because %s"
- % (`oid`, reason))
- else:
- r.append("Couldn't undo change to %s" % (`oid`))
- return join(r,'\n')
+ def __init__(self, oid, reason=None):
+ self._oid = oid
+ self._reason = reason
+
+ def __str__(self):
+ return _fmt_undo(self._oid, self._reason)
+
+class MultipleUndoErrors(UndoError):
+ """Several undo errors occured during a single transaction."""
+
+ def __init__(self, errs):
+ # provide an oid and reason for clients that only look at that
+ UndoError.__init__(self, *errs[0])
+ self._errs = errs
- __str__=__repr__
+ def __str__(self):
+ return "\n".join([_fmt_undo(*pair) for pair in self._errs])
class StorageError(POSError):
"""Base class for storage based exceptions."""
@@ -200,9 +205,6 @@
class ExportError(POSError):
"""An export file doesn't have the right format."""
-
-class Unimplemented(POSError):
- """An unimplemented feature was used."""
class Unsupported(POSError):
"""An feature that is unsupported bt the storage was used."""
=== ZODB3/ZODB/FileStorage.py 1.118 => 1.119 === (575/675 lines abridged)
--- ZODB3/ZODB/FileStorage.py:1.118 Mon Dec 2 17:45:49 2002
+++ ZODB3/ZODB/FileStorage.py Tue Dec 3 13:36:29 2002
@@ -133,10 +133,10 @@
fsync = None
from ZODB import BaseStorage, ConflictResolution, POSException
-from ZODB.POSException import UndoError, POSKeyError
+from ZODB.POSException import UndoError, POSKeyError, MultipleUndoErrors
from ZODB.TimeStamp import TimeStamp
from ZODB.lock_file import lock_file
-from ZODB.utils import t32, p64, U64, cp
+from ZODB.utils import p64, u64, cp, z64
try:
from ZODB.fsIndex import fsIndex
@@ -144,9 +144,9 @@
def fsIndex():
return {}
-from zLOG import LOG, BLATHER, WARNING, ERROR, PANIC, register_subsystem
+from zLOG import LOG, BLATHER, WARNING, ERROR, PANIC
-z64='\0'*8
+t32 = 1L << 32
# the struct formats for the headers
TRANS_HDR = ">8s8scHHH"
DATA_HDR = ">8s8s8s8sH8s"
@@ -173,7 +173,7 @@
def panic(message, *data):
message = message % data
LOG('ZODB FS', PANIC, "%s ERROR: %s\n" % (packed_version, message))
- raise CorruptedTransactionError, message
+ raise CorruptedTransactionError(message)
class FileStorageError(POSException.StorageError):
pass
@@ -191,8 +191,11 @@
POSException.StorageSystemError):
"""Corrupted file storage."""
-class CorruptedTransactionError(CorruptedFileStorageError): pass
-class CorruptedDataError(CorruptedFileStorageError): pass
+class CorruptedTransactionError(CorruptedFileStorageError):
+ pass
+
+class CorruptedDataError(CorruptedFileStorageError):
+ pass
class FileStorageQuotaError(FileStorageError,
POSException.StorageSystemError):
[-=- -=- -=- 575 lines omitted -=- -=- -=-]
# Hm, the data were truncated or the checkpoint flag wasn't
@@ -2288,7 +2291,7 @@
# the end and read what should be the transaction
# length of the last transaction.
seek(-8, 2)
- rtl=U64(read(8))
+ rtl=u64(read(8))
# Now check to see if the redundant transaction length is
# reasonable:
if self._file_size - rtl < pos or rtl < TRANS_HDR_LEN:
@@ -2346,8 +2349,8 @@
raise IndexError, index
class RecordIterator(Iterator, BaseStorage.TransactionRecord):
- """Iterate over the transactions in a FileStorage file.
- """
+ """Iterate over the transactions in a FileStorage file."""
+
def __init__(self, tid, status, user, desc, ext, pos, tend, file, tpos):
self.tid = tid
self.status = status
@@ -2366,15 +2369,15 @@
self._file.seek(pos)
h = self._file.read(DATA_HDR_LEN)
oid, serial, sprev, stloc, vlen, splen = unpack(DATA_HDR, h)
- prev = U64(sprev)
- tloc = U64(stloc)
- plen = U64(splen)
+ prev = u64(sprev)
+ tloc = u64(stloc)
+ plen = u64(splen)
dlen = DATA_HDR_LEN + (plen or 8)
if vlen:
dlen += (16 + vlen)
tmp = self._file.read(16)
- pv = U64(tmp[8:16])
+ pv = u64(tmp[8:16])
version = self._file.read(vlen)
else:
version = ''
@@ -2447,7 +2450,7 @@
def _readnext(self):
"""Read the next record from the storage."""
self.file.seek(self.pos - 8)
- self.pos -= U64(self.file.read(8)) + 8
+ self.pos -= u64(self.file.read(8)) + 8
self.file.seek(self.pos)
h = self.file.read(TRANS_HDR_LEN)
tid, tl, status, ul, dl, el = struct.unpack(TRANS_HDR, h)
=== ZODB3/ZODB/Connection.py 1.79 => 1.80 ===
--- ZODB3/ZODB/Connection.py:1.79 Mon Nov 18 18:17:40 2002
+++ ZODB3/ZODB/Connection.py Tue Dec 3 13:36:29 2002
@@ -40,7 +40,7 @@
global global_code_timestamp
global_code_timestamp = time()
-ExtensionKlass=Base.__class__
+ExtensionKlass = Base.__class__
class Connection(ExportImport.ExportImport):
"""Object managers for individual object space.
=== ZODB3/ZODB/ConflictResolution.py 1.16 => 1.17 ===
--- ZODB3/ZODB/ConflictResolution.py:1.16 Mon Nov 18 18:17:40 2002
+++ ZODB3/ZODB/ConflictResolution.py Tue Dec 3 13:36:29 2002
@@ -92,7 +92,7 @@
unpickler.persistent_load = prfactory.persistent_load
class_tuple = unpickler.load()[0]
if bad_class(class_tuple):
- return 0
+ return None
newstate = unpickler.load()
klass = load_class(class_tuple)
inst = klass.__basicnew__()
@@ -101,7 +101,7 @@
resolve = inst._p_resolveConflict
except AttributeError:
bad_classes[class_tuple] = 1
- return 0
+ return None
old = state(self, oid, oldSerial, prfactory)
committed = state(self, oid, committedSerial, prfactory, committedData)
@@ -115,7 +115,7 @@
pickler.dump(resolved)
return file.getvalue(1)
except ConflictError:
- return 0
+ return None
except:
# If anything else went wrong, catch it here and avoid passing an
# arbitrary exception back to the client. The error here will mask
@@ -124,7 +124,7 @@
# the error so that any problems can be fixed.
zLOG.LOG("Conflict Resolution", zLOG.ERROR,
"Unexpected error", error=sys.exc_info())
- return 0
+ return None
class ConflictResolvingStorage:
"Mix-in class that provides conflict resolution handling for storages"