[Zope-Checkins] CVS: Zope3/lib/python/ZODB - Connection.py:1.60.6.3

Jeremy Hylton jeremy@zope.com
Thu, 21 Feb 2002 18:25:55 -0500


Update of /cvs-repository/Zope3/lib/python/ZODB
In directory cvs.zope.org:/tmp/cvs-serv15451

Modified Files:
      Tag: Zope-3x-branch
	Connection.py 
Log Message:
Lots of refactoring plus fixes for BTrees.

Fix new_persistent_id() to work with new-style types.  A BTree object,
for example, has an _p_oid attribute.  The BTree _type_ also has a
_p_oid attribute -- the descriptor for the BTree objects.  So the
simple hasattr() test in the pre-2.2 version of the code was
insufficient.  Unfortunately, it's not obvious how to ask if something
is a descriptor, so I extended the test in a different way.

new_persistent_id() does not consider an object to be a persistent
object unless it has an _p_oid attribute *and* that attribute is a
string.  we may need to modify the test in the future, but it works
for now.

Refactoring:
    Replace apply() with extended call syntax.
    Add whitespace when needed.
    Refactor commit() by adding helper method commit_store().



=== Zope3/lib/python/ZODB/Connection.py 1.60.6.2 => 1.60.6.3 ===
 from zLOG import LOG, ERROR, BLATHER
 from ConflictResolution import ResolvedSerial
+from ZODB.utils import U64
 
 from cPickle import Unpickler, Pickler
 from cStringIO import StringIO
 import sys
 from time import time
-from types import StringType, ClassType
+from types import StringType, ClassType, TupleType
 
 from Transaction import get_transaction
 
@@ -78,13 +79,12 @@
         try: del self.cacheGC
         except: pass
 
-    def __getitem__(self, oid,
-                    tt=type(())):
-        cache=self._cache
-        object=cache.get(oid, self)
-        if object is not self: return object
+    def __getitem__(self, oid):
+        object = self._cache.get(oid, self)
+        if object is not self:
+            return object
 
-        __traceback_info__ = (oid)
+        __traceback_info__ = (oid,)
         p, serial = self._storage.load(oid, self._version)
         __traceback_info__ = (oid, p)
         file=StringIO(p)
@@ -99,7 +99,7 @@
 
         klass, args = object
 
-        if type(klass) is tt:
+        if isinstance(klass, TupleType):
             module, name = klass
             klass=self._db._classFactory(self, module, name)
         
@@ -110,17 +110,18 @@
             # The contacts are a bit unclear.
             object=__new__(klass)
         else:
-            object=apply(klass,args)
-            if klass is not ExtensionKlass:
+            object = klass(*args)
+            if klass is not type:
                 object.__dict__.clear()
 
-        object._p_oid=oid
-        object._p_jar=self
-        object._p_changed=None
-        object._p_serial=serial
-
-        cache[oid]=object
-        if oid=='\0\0\0\0\0\0\0\0': self._root_=object # keep a ref
+        object._p_oid = oid
+        object._p_jar = self
+        object._p_changed = None
+        object._p_serial = serial
+
+        self._cache[oid] = object
+        if oid == '\0\0\0\0\0\0\0\0':
+            self._root_ = object # keep a ref
         return object
 
     ######################################################################
@@ -313,7 +314,7 @@
             # We registered ourself.  Execute a commit action, if any.
             if self.__onCommitActions is not None:
                 method_name, args, kw = self.__onCommitActions.pop(0)
-                apply(getattr(self, method_name), (transaction,) + args, kw)
+                getattr(self, method_name)(transaction, *args, **kw)
             return
         oid=object._p_oid
         invalid=self._invalid
@@ -337,80 +338,82 @@
             # Nothing to do
             return
 
-        stack=[object]
+        stack = [object]
         
-        file=StringIO()
-        seek=file.seek
-        pickler=Pickler(file,1)
-        pickler.persistent_id=new_persistent_id(self, stack.append)
-        dbstore=self._storage.store
-        file=file.getvalue
-        cache=self._cache
-        get=cache.get
-        dump=pickler.dump
-        clear_memo=pickler.clear_memo
+        file = StringIO()
+        pickler = Pickler(file, 1)
+        pickler.persistent_id = new_persistent_id(self, stack)
 
-
-        version=self._version
+        version = self._version
         
         while stack:
-            object=stack[-1]
-            del stack[-1]
-            oid=object._p_oid
-            serial=getattr(object, '_p_serial', '\0\0\0\0\0\0\0\0')
-            if serial == '\0\0\0\0\0\0\0\0':
-                # new object
-                self._creating.append(oid)
+            self.commit_store(stack.pop(), file, pickler, transaction)
+
+    def commit_store(self, pobject, file, pickler, transaction):
+        # XXX the file and pickler get reset each time through...
+        # Maybe just create new ones each time?  Except I'm not sure
+        # how that interacts with the persistent_id attribute.
+        oid = pobject._p_oid
+        serial = getattr(pobject, '_p_serial', '\0\0\0\0\0\0\0\0')
+        if serial == '\0\0\0\0\0\0\0\0':
+            # new object
+            self._creating.append(oid)
+        else:
+            #XXX We should never get here
+            #jer: Don't understand previous comment.
+            if ((self._invalidated.has_key(oid) and
+                 not hasattr(pobject, '_p_resolveConflict'))
+                or self._invalidated.has_key(None)):
+                raise ConflictError, `oid`
+            self._invalidating.append(oid)
+
+
+        klass = pobject.__class__
+
+        # It's unclear what exactly this conditional is testing and
+        # why it takes one branch or the other.  It looks like a
+        # Persistent object will never has it's __getstate__() called.
+        if pobject is type:
+            dict = {}
+            dict.update(pobject.__dict__)
+            # XXX we delete the _p_jar, but not, say, _p_oid?
+            del dict['_p_jar']
+            args = pobject.__name__, pobject.__bases__, dict
+            state = None
+        else:
+            # XXX For what kind of object is this path taken?
+            if hasattr(klass, '__getinitargs__'):
+                args = pobject.__getinitargs__()
+                len(args) # XXX Assert it's a sequence
             else:
-                #XXX We should never get here
-                if (
-                    (invalid(oid) and
-                     not hasattr(object, '_p_resolveConflict'))
-                    or
-                    invalid(None)
-                    ):
-                    raise ConflictError, `oid`
-                self._invalidating.append(oid)
-                
-            klass = object.__class__
-        
-            if klass is ExtensionKlass:
-                # Yee Ha!
-                dict={}
-                dict.update(object.__dict__)
-                del dict['_p_jar']
-                args=object.__name__, object.__bases__, dict
-                state=None
+                args = None # New no-constructor protocol!
+
+            module = getattr(klass,'__module__','')
+            if module:
+                klass = module, klass.__name__
+            __traceback_info__ = klass, oid, self._version
+            state = pobject.__getstate__()
+
+        file.seek(0)
+        pickler.clear_memo()
+        pickler.dump((klass, args))
+        pickler.dump(state)
+        p = file.getvalue(1)
+        s = self._storage.store(oid, serial, p, self._version, transaction)
+        # Put the object in the cache before handling the
+        # response, just in case the response contains the
+        # serial number for a newly created object
+        try:
+            self._cache[oid] = pobject
+        except:
+            # Dang, I bet its wrapped:
+            # jer: Why does this mean?
+            if hasattr(pobject, 'aq_base'):
+                self._cache[oid] = pobject.aq_base
             else:
-                if hasattr(klass, '__getinitargs__'):
-                    args = object.__getinitargs__()
-                    len(args) # XXX Assert it's a sequence
-                else:
-                    args = None # New no-constructor protocol!
+                raise
+        self._handle_serial(s, oid)
         
-                module=getattr(klass,'__module__','')
-                if module: klass=module, klass.__name__
-                __traceback_info__=klass, oid, self._version
-                state=object.__getstate__()
-        
-            seek(0)
-            clear_memo()
-            dump((klass,args))
-            dump(state)
-            p=file(1)
-            s=dbstore(oid,serial,p,version,transaction)
-            # Put the object in the cache before handling the
-            # response, just in case the response contains the
-            # serial number for a newly created object
-            try: cache[oid]=object
-            except:
-                # Dang, I bet its wrapped:
-                if hasattr(object, 'aq_base'):
-                    cache[oid]=object.aq_base
-                else:
-                    raise
-
-            self._handle_serial(s, oid)
 
     def commit_sub(self, t):
         """Commit all work done in subtransactions"""
@@ -429,7 +432,6 @@
         load=src.load
         store=tmp.store
         dest=self._version
-        get=self._cache.get
         oids=src._index.keys()
 
         # Copy invalidating and creating info from temporary storage:
@@ -485,11 +487,13 @@
         self._invalidated[oid]=1
 
     def modifiedInVersion(self, oid):
-        try: return self._db.modifiedInVersion(oid)
+        try:
+            return self._db.modifiedInVersion(oid)
         except KeyError:
             return self._version
 
-    def root(self): return self['\0\0\0\0\0\0\0\0']
+    def root(self):
+        return self['\0\0\0\0\0\0\0\0']
 
     def oldstate(self, object, serial):
         """Return the state of an object as it existed in some time in the past
@@ -520,20 +524,20 @@
     
             klass, args = copy
     
-            if klass is not ExtensionKlass:
+            if klass is not type:
                 LOG('ZODB',ERROR,
-                    "Unexpected klass when setting class state on %s"
+                    "Unexpected class when setting class state on %s"
                     % getattr(object,'__name__','(?)'))
                 return
             
-            copy=apply(klass,args)
+            copy = klass(*args)
             object.__dict__.clear()
             object.__dict__.update(copy.__dict__)
     
-            object._p_oid=oid
-            object._p_jar=self
-            object._p_changed=0
-            object._p_serial=serial
+            object._p_oid = oid
+            object._p_jar = self
+            object._p_changed = 0
+            object._p_serial = serial
         except:
             LOG('ZODB',ERROR, 'setklassstate failed', error=sys.exc_info())
             raise
@@ -542,9 +546,8 @@
         if self.__onCommitActions is not None:
             del self.__onCommitActions
         self._storage.tpc_abort(transaction)
-        cache=self._cache
-        cache.invalidateMany(self._invalidated)
-        cache.invalidateMany(self._invalidating)
+        self._cache.invalidateMany(self._invalidated)
+        self._cache.invalidateMany(self._invalidating)
         self._invalidate_creating()
 
     def tpc_begin(self, transaction, sub=None):
@@ -671,34 +674,53 @@
     def close(self):
         self._breakcr()
 
-def new_persistent_id(self, stackup):
-
-    # Create a special persistent_id that passes T and the subobject
-    # stack along:
+def new_persistent_id(self, stack):
+    # XXX need a doc string.  not sure if the one for persistent_id()
+    # below is correct.
+
+    # Create a special persistent_id that captures T and the subobject
+    # stack in a closure.
+
+    def persistent_id(object):
+        """Test if an object is persistent, returning an oid if it is.
+
+        This function is used by the pickler to test whether an object
+        is persistent.  If it isn't, the function returns None and the
+        object is included in the pickle for the current persistent
+        object.
 
-    def persistent_id(object, self=self, stackup=stackup):
-
-        new_oid = self.new_oid
+        If it is persistent, it returns the oid and sometimes a tuple
+        with other stuff.
+        """
         
         if (not hasattr(object, '_p_oid') or
-            type(object) is ClassType): return None
-
-        oid=object._p_oid
+            isinstance(object, ClassType)):
+            return None
+        
+        oid = object._p_oid
 
+        # I'd like to write something like this --
+        # if isinstance(oid, types.MemberDescriptor):
+        # -- but I can't because the type doesn't have a canonical name.
+        # Instead, we'll assert that an oid must always be a string
+        if not (isinstance(oid, StringType) or oid is None):
+            return None
+        
         if oid is None or object._p_jar is not self:
             oid = self.new_oid()
-            object._p_jar=self
-            object._p_oid=oid
-            stackup(object)
-
-        klass=object.__class__
-
-        if klass is ExtensionKlass: return oid
-
-        if hasattr(klass, '__getinitargs__'): return oid
-
-        module=getattr(klass,'__module__','')
-        if module: klass=module, klass.__name__
+            object._p_jar = self
+            object._p_oid = oid
+            stack.append(object)
+
+        klass = object.__class__
+
+        if klass is type:
+            return oid
+        if hasattr(klass, '__getinitargs__'):
+            return oid
+        module = getattr(klass,'__module__', '')
+        if module:
+            klass = module, klass.__name__
 
         return oid, klass