608 lines
24 KiB
Python
608 lines
24 KiB
Python
from panda3d.core import *
|
|
from panda3d.direct import *
|
|
from .MsgTypes import *
|
|
from direct.task import Task
|
|
from direct.directnotify import DirectNotifyGlobal
|
|
from . import CRCache
|
|
from direct.distributed.CRDataCache import CRDataCache
|
|
from direct.distributed.ConnectionRepository import ConnectionRepository
|
|
from direct.showbase import PythonUtil
|
|
from . import ParentMgr
|
|
from . import RelatedObjectMgr
|
|
import time
|
|
from .ClockDelta import *
|
|
|
|
|
|
class ClientRepositoryBase(ConnectionRepository):
|
|
"""
|
|
This maintains a client-side connection with a Panda server.
|
|
|
|
This base class exists to collect the common code between
|
|
ClientRepository, which is the CMU-provided, open-source version
|
|
of the client repository code, and OTPClientRepository, which is
|
|
the VR Studio's implementation of the same.
|
|
"""
|
|
notify = DirectNotifyGlobal.directNotify.newCategory("ClientRepositoryBase")
|
|
|
|
def __init__(self, dcFileNames = None, dcSuffix = '',
|
|
connectMethod = None, threadedNet = None):
|
|
if connectMethod is None:
|
|
connectMethod = self.CM_HTTP
|
|
ConnectionRepository.__init__(self, connectMethod, base.config, hasOwnerView = True, threadedNet = threadedNet)
|
|
self.dcSuffix = dcSuffix
|
|
if hasattr(self, 'setVerbose'):
|
|
if self.config.GetBool('verbose-clientrepository'):
|
|
self.setVerbose(1)
|
|
|
|
self.context=100000
|
|
self.setClientDatagram(1)
|
|
|
|
self.deferredGenerates = []
|
|
self.deferredDoIds = {}
|
|
self.lastGenerate = 0
|
|
self.setDeferInterval(base.config.GetDouble('deferred-generate-interval', 0.2))
|
|
self.noDefer = False # Set this True to temporarily disable deferring.
|
|
|
|
self.recorder = base.recorder
|
|
|
|
self.readDCFile(dcFileNames)
|
|
self.cache=CRCache.CRCache()
|
|
self.doDataCache = CRDataCache()
|
|
self.cacheOwner=CRCache.CRCache()
|
|
self.serverDelta = 0
|
|
|
|
self.bootedIndex = None
|
|
self.bootedText = None
|
|
|
|
# create a parentMgr to handle distributed reparents
|
|
# this used to be 'token2nodePath'
|
|
self.parentMgr = ParentMgr.ParentMgr()
|
|
|
|
# The RelatedObjectMgr helps distributed objects find each
|
|
# other.
|
|
self.relatedObjectMgr = RelatedObjectMgr.RelatedObjectMgr(self)
|
|
|
|
# This will be filled in when a TimeManager is created.
|
|
self.timeManager = None
|
|
|
|
# Keep track of how recently we last sent a heartbeat message.
|
|
# We want to keep these coming at heartbeatInterval seconds.
|
|
self.heartbeatInterval = base.config.GetDouble('heartbeat-interval', 10)
|
|
self.heartbeatStarted = 0
|
|
self.lastHeartbeat = 0
|
|
|
|
self._delayDeletedDOs = {}
|
|
|
|
self.specialNameNumber = 0
|
|
|
|
def setDeferInterval(self, deferInterval):
|
|
"""Specifies the minimum amount of time, in seconds, that must
|
|
elapse before generating any two DistributedObjects whose
|
|
class type is marked "deferrable". Set this to 0 to indicate
|
|
no deferring will occur."""
|
|
|
|
self.deferInterval = deferInterval
|
|
self.setHandleCUpdates(self.deferInterval == 0)
|
|
|
|
if self.deferredGenerates:
|
|
taskMgr.remove('deferredGenerate')
|
|
taskMgr.doMethodLater(self.deferInterval, self.doDeferredGenerate, 'deferredGenerate')
|
|
|
|
## def queryObjectAll(self, doID, context=0):
|
|
## """
|
|
## Get a one-time snapshot look at the object.
|
|
## """
|
|
## assert self.notify.debugStateCall(self)
|
|
## # Create a message
|
|
## datagram = PyDatagram()
|
|
## datagram.addServerHeader(
|
|
## doID, localAvatar.getDoId(), 2020)
|
|
## # A context that can be used to index the response if needed
|
|
## datagram.addUint32(context)
|
|
## self.send(datagram)
|
|
## # Make sure the message gets there.
|
|
## self.flush()
|
|
|
|
def specialName(self, label):
|
|
name = ("SpecialName %s %s" % (self.specialNameNumber, label))
|
|
self.specialNameNumber += 1
|
|
return name
|
|
|
|
def getTables(self, ownerView):
|
|
if ownerView:
|
|
return self.doId2ownerView, self.cacheOwner
|
|
else:
|
|
return self.doId2do, self.cache
|
|
|
|
def _getMsgName(self, msgId):
|
|
# we might get a list of message names, use the first one
|
|
return makeList(MsgId2Names.get(msgId, 'UNKNOWN MESSAGE: %s' % msgId))[0]
|
|
|
|
def allocateContext(self):
|
|
self.context+=1
|
|
return self.context
|
|
|
|
def setServerDelta(self, delta):
|
|
"""
|
|
Indicates the approximate difference in seconds between the
|
|
client's clock and the server's clock, in universal time (not
|
|
including timezone shifts). This is mainly useful for
|
|
reporting synchronization information to the logs; don't
|
|
depend on it for any precise timing requirements.
|
|
|
|
Also see Notify.setServerDelta(), which also accounts for a
|
|
timezone shift.
|
|
"""
|
|
self.serverDelta = delta
|
|
|
|
def getServerDelta(self):
|
|
return self.serverDelta
|
|
|
|
def getServerTimeOfDay(self):
|
|
"""
|
|
Returns the current time of day (seconds elapsed since the
|
|
1972 epoch) according to the server's clock. This is in GMT,
|
|
and hence is irrespective of timezones.
|
|
|
|
The value is computed based on the client's clock and the
|
|
known delta from the server's clock, which is not terribly
|
|
precisely measured and may drift slightly after startup, but
|
|
it should be accurate plus or minus a couple of seconds.
|
|
"""
|
|
return time.time() + self.serverDelta
|
|
|
|
def doGenerate(self, parentId, zoneId, classId, doId, di):
|
|
# Look up the dclass
|
|
assert parentId == self.GameGlobalsId or parentId in self.doId2do
|
|
dclass = self.dclassesByNumber[classId]
|
|
assert(self.notify.debug("performing generate for %s %s" % (dclass.getName(), doId)))
|
|
dclass.startGenerate()
|
|
# Create a new distributed object, and put it in the dictionary
|
|
distObj = self.generateWithRequiredOtherFields(dclass, doId, di, parentId, zoneId)
|
|
dclass.stopGenerate()
|
|
|
|
def flushGenerates(self):
|
|
""" Forces all pending generates to be performed immediately. """
|
|
while self.deferredGenerates:
|
|
msgType, extra = self.deferredGenerates[0]
|
|
del self.deferredGenerates[0]
|
|
self.replayDeferredGenerate(msgType, extra)
|
|
|
|
taskMgr.remove('deferredGenerate')
|
|
|
|
def replayDeferredGenerate(self, msgType, extra):
|
|
""" Override this to do something appropriate with deferred
|
|
"generate" messages when they are replayed().
|
|
"""
|
|
|
|
if msgType == CLIENT_ENTER_OBJECT_REQUIRED_OTHER:
|
|
# It's a generate message.
|
|
doId = extra
|
|
if doId in self.deferredDoIds:
|
|
args, deferrable, dg, updates = self.deferredDoIds[doId]
|
|
del self.deferredDoIds[doId]
|
|
self.doGenerate(*args)
|
|
|
|
if deferrable:
|
|
self.lastGenerate = globalClock.getFrameTime()
|
|
|
|
for dg, di in updates:
|
|
# non-DC updates that need to be played back in-order are
|
|
# stored as (msgType, (dg, di))
|
|
if type(di) is tuple:
|
|
msgType = dg
|
|
dg, di = di
|
|
self.replayDeferredGenerate(msgType, (dg, di))
|
|
else:
|
|
# ovUpdated is set to True since its OV
|
|
# is assumbed to have occured when the
|
|
# deferred update was originally received
|
|
self.__doUpdate(doId, di, True)
|
|
else:
|
|
self.notify.warning("Ignoring deferred message %s" % (msgType))
|
|
|
|
def doDeferredGenerate(self, task):
|
|
""" This is the task that generates an object on the deferred
|
|
queue. """
|
|
|
|
now = globalClock.getFrameTime()
|
|
while self.deferredGenerates:
|
|
if now - self.lastGenerate < self.deferInterval:
|
|
# Come back later.
|
|
return Task.again
|
|
|
|
# Generate the next deferred object.
|
|
msgType, extra = self.deferredGenerates[0]
|
|
del self.deferredGenerates[0]
|
|
self.replayDeferredGenerate(msgType, extra)
|
|
|
|
# All objects are generaetd.
|
|
return Task.done
|
|
|
|
def generateWithRequiredFields(self, dclass, doId, di, parentId, zoneId):
|
|
if doId in self.doId2do:
|
|
# ...it is in our dictionary.
|
|
# Just update it.
|
|
distObj = self.doId2do[doId]
|
|
assert distObj.dclass == dclass
|
|
distObj.generate()
|
|
distObj.setLocation(parentId, zoneId)
|
|
distObj.updateRequiredFields(dclass, di)
|
|
# updateRequiredFields calls announceGenerate
|
|
elif self.cache.contains(doId):
|
|
# ...it is in the cache.
|
|
# Pull it out of the cache:
|
|
distObj = self.cache.retrieve(doId)
|
|
assert distObj.dclass == dclass
|
|
# put it in the dictionary:
|
|
self.doId2do[doId] = distObj
|
|
# and update it.
|
|
distObj.generate()
|
|
# make sure we don't have a stale location
|
|
distObj.parentId = None
|
|
distObj.zoneId = None
|
|
distObj.setLocation(parentId, zoneId)
|
|
distObj.updateRequiredFields(dclass, di)
|
|
# updateRequiredFields calls announceGenerate
|
|
else:
|
|
# ...it is not in the dictionary or the cache.
|
|
# Construct a new one
|
|
classDef = dclass.getClassDef()
|
|
if classDef == None:
|
|
self.notify.error("Could not create an undefined %s object." % (dclass.getName()))
|
|
distObj = classDef(self)
|
|
distObj.dclass = dclass
|
|
# Assign it an Id
|
|
distObj.doId = doId
|
|
# Put the new do in the dictionary
|
|
self.doId2do[doId] = distObj
|
|
# Update the required fields
|
|
distObj.generateInit() # Only called when constructed
|
|
distObj._retrieveCachedData()
|
|
distObj.generate()
|
|
distObj.setLocation(parentId, zoneId)
|
|
distObj.updateRequiredFields(dclass, di)
|
|
# updateRequiredFields calls announceGenerate
|
|
self.notify.debug("New DO:%s, dclass:%s" % (doId, dclass.getName()))
|
|
return distObj
|
|
|
|
def generateWithRequiredOtherFields(self, dclass, doId, di,
|
|
parentId = None, zoneId = None):
|
|
if doId in self.doId2do:
|
|
# ...it is in our dictionary.
|
|
# Just update it.
|
|
distObj = self.doId2do[doId]
|
|
assert distObj.dclass == dclass
|
|
distObj.generate()
|
|
distObj.setLocation(parentId, zoneId)
|
|
distObj.updateRequiredOtherFields(dclass, di)
|
|
# updateRequiredOtherFields calls announceGenerate
|
|
elif self.cache.contains(doId):
|
|
# ...it is in the cache.
|
|
# Pull it out of the cache:
|
|
distObj = self.cache.retrieve(doId)
|
|
assert distObj.dclass == dclass
|
|
# put it in the dictionary:
|
|
self.doId2do[doId] = distObj
|
|
# and update it.
|
|
distObj.generate()
|
|
# make sure we don't have a stale location
|
|
distObj.parentId = None
|
|
distObj.zoneId = None
|
|
distObj.setLocation(parentId, zoneId)
|
|
distObj.updateRequiredOtherFields(dclass, di)
|
|
# updateRequiredOtherFields calls announceGenerate
|
|
else:
|
|
# ...it is not in the dictionary or the cache.
|
|
# Construct a new one
|
|
classDef = dclass.getClassDef()
|
|
if classDef == None:
|
|
self.notify.error("Could not create an undefined %s object." % (dclass.getName()))
|
|
distObj = classDef(self)
|
|
distObj.dclass = dclass
|
|
# Assign it an Id
|
|
distObj.doId = doId
|
|
# Put the new do in the dictionary
|
|
self.doId2do[doId] = distObj
|
|
# Update the required fields
|
|
distObj.generateInit() # Only called when constructed
|
|
distObj._retrieveCachedData()
|
|
distObj.generate()
|
|
distObj.setLocation(parentId, zoneId)
|
|
distObj.updateRequiredOtherFields(dclass, di)
|
|
# updateRequiredOtherFields calls announceGenerate
|
|
return distObj
|
|
|
|
def generateWithRequiredOtherFieldsOwner(self, dclass, doId, di):
|
|
if doId in self.doId2ownerView:
|
|
# ...it is in our dictionary.
|
|
# Just update it.
|
|
self.notify.error('duplicate owner generate for %s (%s)' % (
|
|
doId, dclass.getName()))
|
|
distObj = self.doId2ownerView[doId]
|
|
assert distObj.dclass == dclass
|
|
distObj.generate()
|
|
distObj.updateRequiredOtherFields(dclass, di)
|
|
# updateRequiredOtherFields calls announceGenerate
|
|
elif self.cacheOwner.contains(doId):
|
|
# ...it is in the cache.
|
|
# Pull it out of the cache:
|
|
distObj = self.cacheOwner.retrieve(doId)
|
|
assert distObj.dclass == dclass
|
|
# put it in the dictionary:
|
|
self.doId2ownerView[doId] = distObj
|
|
# and update it.
|
|
distObj.generate()
|
|
distObj.updateRequiredOtherFields(dclass, di)
|
|
# updateRequiredOtherFields calls announceGenerate
|
|
else:
|
|
# ...it is not in the dictionary or the cache.
|
|
# Construct a new one
|
|
classDef = dclass.getOwnerClassDef()
|
|
if classDef == None:
|
|
self.notify.error("Could not create an undefined %s object. Have you created an owner view?" % (dclass.getName()))
|
|
distObj = classDef(self)
|
|
distObj.dclass = dclass
|
|
# Assign it an Id
|
|
distObj.doId = doId
|
|
# Put the new do in the dictionary
|
|
self.doId2ownerView[doId] = distObj
|
|
# Update the required fields
|
|
distObj.generateInit() # Only called when constructed
|
|
distObj.generate()
|
|
distObj.updateRequiredOtherFields(dclass, di)
|
|
# updateRequiredOtherFields calls announceGenerate
|
|
return distObj
|
|
|
|
|
|
def disableDoId(self, doId, ownerView=False):
|
|
table, cache = self.getTables(ownerView)
|
|
# Make sure the object exists
|
|
if doId in table:
|
|
# Look up the object
|
|
distObj = table[doId]
|
|
# remove the object from the dictionary
|
|
del table[doId]
|
|
|
|
# Only cache the object if it is a "cacheable" type
|
|
# object; this way we don't clutter up the caches with
|
|
# trivial objects that don't benefit from caching.
|
|
# also don't try to cache an object that is delayDeleted
|
|
cached = False
|
|
if distObj.getCacheable() and distObj.getDelayDeleteCount() <= 0:
|
|
cached = cache.cache(distObj)
|
|
if not cached:
|
|
distObj.deleteOrDelay()
|
|
if distObj.getDelayDeleteCount() <= 0:
|
|
# make sure we're not leaking
|
|
distObj.detectLeaks()
|
|
|
|
elif doId in self.deferredDoIds:
|
|
# The object had been deferred. Great; we don't even have
|
|
# to generate it now.
|
|
del self.deferredDoIds[doId]
|
|
i = self.deferredGenerates.index((CLIENT_ENTER_OBJECT_REQUIRED_OTHER, doId))
|
|
del self.deferredGenerates[i]
|
|
if len(self.deferredGenerates) == 0:
|
|
taskMgr.remove('deferredGenerate')
|
|
|
|
else:
|
|
self._logFailedDisable(doId, ownerView)
|
|
|
|
def _logFailedDisable(self, doId, ownerView):
|
|
self.notify.warning(
|
|
"Disable failed. DistObj "
|
|
+ str(doId) +
|
|
" is not in dictionary, ownerView=%s" % ownerView)
|
|
|
|
def handleDelete(self, di):
|
|
# overridden by ClientRepository
|
|
assert 0
|
|
|
|
def handleUpdateField(self, di):
|
|
"""
|
|
This method is called when a CLIENT_OBJECT_UPDATE_FIELD
|
|
message is received; it decodes the update, unpacks the
|
|
arguments, and calls the corresponding method on the indicated
|
|
DistributedObject.
|
|
|
|
In fact, this method is exactly duplicated by the C++ method
|
|
cConnectionRepository::handle_update_field(), which was
|
|
written to optimize the message loop by handling all of the
|
|
CLIENT_OBJECT_UPDATE_FIELD messages in C++. That means that
|
|
nowadays, this Python method will probably never be called,
|
|
since UPDATE_FIELD messages will not even be passed to the
|
|
Python message handlers. But this method remains for
|
|
documentation purposes, and also as a "just in case" handler
|
|
in case we ever do come across a situation in the future in
|
|
which python might handle the UPDATE_FIELD message.
|
|
"""
|
|
# Get the DO Id
|
|
doId = di.getUint32()
|
|
|
|
ovUpdated = self.__doUpdateOwner(doId, di)
|
|
|
|
if doId in self.deferredDoIds:
|
|
# This object hasn't really been generated yet. Sit on
|
|
# the update.
|
|
args, deferrable, dg0, updates = self.deferredDoIds[doId]
|
|
|
|
# Keep a copy of the datagram, and move the di to the copy
|
|
dg = Datagram(di.getDatagram())
|
|
di = DatagramIterator(dg, di.getCurrentIndex())
|
|
|
|
updates.append((dg, di))
|
|
else:
|
|
# This object has been fully generated. It's OK to update.
|
|
self.__doUpdate(doId, di, ovUpdated)
|
|
|
|
|
|
def __doUpdate(self, doId, di, ovUpdated):
|
|
# Find the DO
|
|
do = self.doId2do.get(doId)
|
|
if do is not None:
|
|
# Let the dclass finish the job
|
|
do.dclass.receiveUpdate(do, di)
|
|
elif not ovUpdated:
|
|
# this next bit is looking for avatar handles so that if you get an update
|
|
# for an avatar that isn't in your doId2do table but there is a
|
|
# avatar handle for that object then it's messages will be forwarded to that
|
|
# object. We are currently using that for whisper echoing
|
|
# if you need a more general perpose system consider registering proxy objects on
|
|
# a dict and adding the avatar handles to that dict when they are created
|
|
# then change/remove the old method. I didn't do that because I couldn't think
|
|
# of a use for it. -JML
|
|
try :
|
|
handle = self.identifyAvatar(doId)
|
|
if handle:
|
|
dclass = self.dclassesByName[handle.dclassName]
|
|
dclass.receiveUpdate(handle, di)
|
|
|
|
else:
|
|
self.notify.warning(
|
|
"Asked to update non-existent DistObj " + str(doId))
|
|
except:
|
|
self.notify.warning(
|
|
"Asked to update non-existent DistObj " + str(doId) + "and failed to find it")
|
|
|
|
def __doUpdateOwner(self, doId, di):
|
|
ovObj = self.doId2ownerView.get(doId)
|
|
if ovObj:
|
|
odg = Datagram(di.getDatagram())
|
|
odi = DatagramIterator(odg, di.getCurrentIndex())
|
|
ovObj.dclass.receiveUpdate(ovObj, odi)
|
|
return True
|
|
return False
|
|
|
|
def handleGoGetLost(self, di):
|
|
# The server told us it's about to drop the connection on us.
|
|
# Get ready!
|
|
if (di.getRemainingSize() > 0):
|
|
self.bootedIndex = di.getUint16()
|
|
self.bootedText = di.getString()
|
|
|
|
self.notify.warning(
|
|
"Server is booting us out (%d): %s" % (self.bootedIndex, self.bootedText))
|
|
else:
|
|
self.bootedIndex = None
|
|
self.bootedText = None
|
|
self.notify.warning(
|
|
"Server is booting us out with no explanation.")
|
|
|
|
# disconnect now, don't wait for send/recv to fail
|
|
self.stopReaderPollTask()
|
|
self.lostConnection()
|
|
|
|
def handleServerHeartbeat(self, di):
|
|
# Got a heartbeat message from the server.
|
|
if base.config.GetBool('server-heartbeat-info', 1):
|
|
self.notify.info("Server heartbeat.")
|
|
|
|
def handleSystemMessage(self, di):
|
|
# Got a system message from the server.
|
|
message = di.getString()
|
|
self.notify.info('Message from server: %s' % (message))
|
|
return message
|
|
|
|
def handleSystemMessageAknowledge(self, di):
|
|
# Got a system message from the server.
|
|
message = di.getString()
|
|
self.notify.info('Message with aknowledge from server: %s' % (message))
|
|
messenger.send("system message aknowledge", [message])
|
|
return message
|
|
|
|
def getObjectsOfClass(self, objClass):
|
|
""" returns dict of doId:object, containing all objects
|
|
that inherit from 'class'. returned dict is safely mutable. """
|
|
doDict = {}
|
|
for doId, do in self.doId2do.items():
|
|
if isinstance(do, objClass):
|
|
doDict[doId] = do
|
|
return doDict
|
|
|
|
def getObjectsOfExactClass(self, objClass):
|
|
""" returns dict of doId:object, containing all objects that
|
|
are exactly of type 'class' (neglecting inheritance). returned
|
|
dict is safely mutable. """
|
|
doDict = {}
|
|
for doId, do in self.doId2do.items():
|
|
if do.__class__ == objClass:
|
|
doDict[doId] = do
|
|
return doDict
|
|
|
|
|
|
def considerHeartbeat(self):
|
|
"""Send a heartbeat message if we haven't sent one recently."""
|
|
if not self.heartbeatStarted:
|
|
self.notify.debug("Heartbeats not started; not sending.")
|
|
return
|
|
|
|
elapsed = globalClock.getRealTime() - self.lastHeartbeat
|
|
if elapsed < 0 or elapsed > self.heartbeatInterval:
|
|
# It's time to send the heartbeat again (or maybe someone
|
|
# reset the clock back).
|
|
self.notify.info("Sending heartbeat mid-frame.")
|
|
self.startHeartbeat()
|
|
|
|
def stopHeartbeat(self):
|
|
taskMgr.remove("heartBeat")
|
|
self.heartbeatStarted = 0
|
|
|
|
def startHeartbeat(self):
|
|
self.stopHeartbeat()
|
|
self.heartbeatStarted = 1
|
|
self.sendHeartbeat()
|
|
self.waitForNextHeartBeat()
|
|
|
|
def sendHeartbeatTask(self, task):
|
|
self.sendHeartbeat()
|
|
return Task.again
|
|
|
|
def waitForNextHeartBeat(self):
|
|
taskMgr.doMethodLater(self.heartbeatInterval, self.sendHeartbeatTask,
|
|
"heartBeat", taskChain = 'net')
|
|
|
|
def replaceMethod(self, oldMethod, newFunction):
|
|
return 0
|
|
|
|
def getWorld(self, doId):
|
|
# Get the world node for this object
|
|
obj = self.doId2do[doId]
|
|
|
|
worldNP = obj.getParent()
|
|
while 1:
|
|
nextNP = worldNP.getParent()
|
|
if nextNP == render:
|
|
break
|
|
elif worldNP.isEmpty():
|
|
return None
|
|
return worldNP
|
|
|
|
def isLive(self):
|
|
if base.config.GetBool('force-live', 0):
|
|
return True
|
|
return not (__dev__ or launcher.isTestServer())
|
|
|
|
def isLocalId(self, id):
|
|
# By default, no ID's are local. See also
|
|
# ClientRepository.isLocalId().
|
|
return 0
|
|
|
|
# methods for tracking delaydeletes
|
|
def _addDelayDeletedDO(self, do):
|
|
# use the id of the object, it's possible to have multiple DelayDeleted instances
|
|
# with identical doIds if an object gets deleted then re-generated
|
|
key = id(do)
|
|
assert key not in self._delayDeletedDOs
|
|
self._delayDeletedDOs[key] = do
|
|
|
|
def _removeDelayDeletedDO(self, do):
|
|
key = id(do)
|
|
del self._delayDeletedDOs[key]
|
|
|
|
def printDelayDeletes(self):
|
|
print('DelayDeletes:')
|
|
print('=============')
|
|
for obj in self._delayDeletedDOs.values():
|
|
print('%s\t%s (%s)\tdelayDeletes=%s' % (
|
|
obj.doId, safeRepr(obj), itype(obj), obj.getDelayDeleteNames()))
|