"""DistributedObject module: contains the DistributedObject class""" from panda3d.core import * from panda3d.direct import * from direct.directnotify.DirectNotifyGlobal import directNotify from direct.distributed.DistributedObjectBase import DistributedObjectBase #from PyDatagram import PyDatagram #from PyDatagramIterator import PyDatagramIterator # Values for DistributedObject.activeState ESNew = 1 ESDeleted = 2 ESDisabling = 3 ESDisabled = 4 # values here and lower are considered "disabled" ESGenerating = 5 # values here and greater are considered "generated" ESGenerated = 6 # update this table if the values above change ESNum2Str = { ESNew: 'ESNew', ESDeleted: 'ESDeleted', ESDisabling: 'ESDisabling', ESDisabled: 'ESDisabled', ESGenerating: 'ESGenerating', ESGenerated: 'ESGenerated', } class DistributedObject(DistributedObjectBase): """ The Distributed Object class is the base class for all network based (i.e. distributed) objects. These will usually (always?) have a dclass entry in a *.dc file. """ notify = directNotify.newCategory("DistributedObject") # A few objects will set neverDisable to 1... Examples are # localToon, and anything that lives in the UberZone. This # keeps them from being disabled when you change zones, # even to the quiet zone. neverDisable = 0 def __init__(self, cr): assert self.notify.debugStateCall(self) try: self.DistributedObject_initialized except: self.DistributedObject_initialized = 1 DistributedObjectBase.__init__(self, cr) # Most DistributedObjects are simple and require no real # effort to load. Some, particularly actors, may take # some significant time to load; these we can optimize by # caching them when they go away instead of necessarily # deleting them. The object should set cacheable to 1 if # it needs to be optimized in this way. self.setCacheable(0) # this is for Toontown only, see toontown.distributed.DelayDeletable self._token2delayDeleteName = {} self._delayDeleteForceAllow = False self._delayDeleted = 0 # Keep track of our state as a distributed object. This # is only trustworthy if the inheriting class properly # calls up the chain for disable() and generate(). self.activeState = ESNew # These are used by getCallbackContext() and doCallbackContext(). self.__nextContext = 0 self.__callbacks = {} # This is used by doneBarrier(). self.__barrierContext = None if __debug__: def status(self, indent=0): """ print out "doId(parentId, zoneId) className and conditionally show generated, disabled, neverDisable, or cachable" """ spaces = ' ' * (indent + 2) try: print("%s%s:" % (' ' * indent, self.__class__.__name__)) flags = [] if self.activeState == ESGenerated: flags.append("generated") if self.activeState < ESGenerating: flags.append("disabled") if self.neverDisable: flags.append("neverDisable") if self.cacheable: flags.append("cacheable") flagStr = "" if len(flags): flagStr = " (%s)" % (" ".join(flags)) print("%sfrom DistributedObject doId:%s, parent:%s, zone:%s%s" % ( spaces, self.doId, self.parentId, self.zoneId, flagStr)) except Exception as e: print("%serror printing status %s" % (spaces, e)) def getAutoInterests(self): # returns the sub-zones under this object that are automatically # opened for us by the server. # have we already cached it? def _getAutoInterests(cls): # returns set of auto-interests for this class and all derived # have we already computed this class's autoInterests? if 'autoInterests' in cls.__dict__: autoInterests = cls.autoInterests else: autoInterests = set() # grab autoInterests from base classes for base in cls.__bases__: autoInterests.update(_getAutoInterests(base)) # grab autoInterests from this class if cls.__name__ in self.cr.dclassesByName: dclass = self.cr.dclassesByName[cls.__name__] field = dclass.getFieldByName('AutoInterest') if field is not None: p = DCPacker() p.setUnpackData(field.getDefaultValue()) len = p.rawUnpackUint16()/4 for i in range(len): zone = int(p.rawUnpackUint32()) autoInterests.add(zone) autoInterests.update(autoInterests) cls.autoInterests = autoInterests return set(autoInterests) autoInterests = _getAutoInterests(self.__class__) # if the server starts supporting multiple auto-interest per class, this check # should be removed if len(autoInterests) > 1: self.notify.error( 'only one auto-interest allowed per DC class, %s has %s autoInterests (%s)' % (self.dclass.getName(), len(autoInterests), list(autoInterests))) _getAutoInterests = None return list(autoInterests) def setNeverDisable(self, bool): assert bool == 1 or bool == 0 self.neverDisable = bool def getNeverDisable(self): return self.neverDisable def _retrieveCachedData(self): # once we know our doId, grab any data that might be stored in the data cache # from the last time we were on the client if self.cr.doDataCache.hasCachedData(self.doId): self._cachedData = self.cr.doDataCache.popCachedData(self.doId) def setCachedData(self, name, data): assert type(name) == type('') # ownership of the data passes to the repository data cache self.cr.doDataCache.setCachedData(self.doId, name, data) def hasCachedData(self, name): assert type(name) == type('') if not hasattr(self, '_cachedData'): return False return name in self._cachedData def getCachedData(self, name): assert type(name) == type('') # ownership of the data passes to the caller of this method data = self._cachedData[name] del self._cachedData[name] return data def flushCachedData(self, name): assert type(name) == type('') # call this to throw out cached data from a previous instantiation self._cachedData[name].flush() def setCacheable(self, bool): assert bool == 1 or bool == 0 self.cacheable = bool def getCacheable(self): return self.cacheable def deleteOrDelay(self): if len(self._token2delayDeleteName) > 0: if not self._delayDeleted: self._delayDeleted = 1 # Object is delayDeleted. Clean up DistributedObject state, # remove from repository tables, so that we won't crash if # another instance of the same object gets generated while # this instance is still delayDeleted. messenger.send(self.getDelayDeleteEvent()) if len(self._token2delayDeleteName) > 0: self.delayDelete() if len(self._token2delayDeleteName) > 0: self._deactivateDO() else: self.disableAnnounceAndDelete() def disableAnnounceAndDelete(self): self.disableAndAnnounce() self.delete() self._destroyDO() def getDelayDeleteCount(self): return len(self._token2delayDeleteName) def getDelayDeleteEvent(self): return self.uniqueName("delayDelete") def getDisableEvent(self): return self.uniqueName("disable") def disableAndAnnounce(self): """ Inheritors should *not* redefine this function. """ # We must send the disable announce message *before* we # actually disable the object. That way, the various cleanup # tasks can run first and take care of restoring the object to # a normal, nondisabled state; and *then* the disable function # can properly disable it (for instance, by parenting it to # hidden). if self.activeState != ESDisabled: self.activeState = ESDisabling messenger.send(self.getDisableEvent()) self.disable() self.activeState = ESDisabled if not self._delayDeleted: # if the object is DelayDeleted, _deactivateDO has # already been called self._deactivateDO() def announceGenerate(self): """ Sends a message to the world after the object has been generated and all of its required fields filled in. """ assert self.notify.debug('announceGenerate(): %s' % (self.doId)) def _deactivateDO(self): # after this is called, the object is no longer an active DistributedObject # and it may be placed in the cache if not self.cr: # we are going to crash, output the destroyDo stacktrace self.notify.warning('self.cr is none in _deactivateDO %d' % self.doId) if hasattr(self, 'destroyDoStackTrace'): print(self.destroyDoStackTrace) self.__callbacks = {} self.cr.closeAutoInterests(self) self.setLocation(0,0) self.cr.deleteObjectLocation(self, self.parentId, self.zoneId) def _destroyDO(self): # after this is called, the object is no longer a DistributedObject # but may still be used as a DelayDeleted object if __debug__: # StackTrace is omitted in packed versions from direct.showbase.PythonUtil import StackTrace self.destroyDoStackTrace = StackTrace() # check for leftover cached data that was not retrieved or flushed by this object # this will catch typos in the data name in calls to get/setCachedData if hasattr(self, '_cachedData'): for name, cachedData in self._cachedData.items(): self.notify.warning('flushing unretrieved cached data: %s' % name) cachedData.flush() del self._cachedData self.cr = None self.dclass = None def disable(self): """ Inheritors should redefine this to take appropriate action on disable """ assert self.notify.debug('disable(): %s' % (self.doId)) pass def isDisabled(self): """ Returns true if the object has been disabled and/or deleted, or if it is brand new and hasn't yet been generated. """ return (self.activeState < ESGenerating) def isGenerated(self): """ Returns true if the object has been fully generated by now, and not yet disabled. """ assert self.notify.debugStateCall(self) return (self.activeState == ESGenerated) def delete(self): """ Inheritors should redefine this to take appropriate action on delete """ assert self.notify.debug('delete(): %s' % (self.doId)) try: self.DistributedObject_deleted except: self.DistributedObject_deleted = 1 def generate(self): """ Inheritors should redefine this to take appropriate action on generate """ assert self.notify.debugStateCall(self) self.activeState = ESGenerating # this has already been set at this point #self.cr.storeObjectLocation(self, self.parentId, self.zoneId) # semi-hack: we seem to be calling generate() more than once for objects that multiply-inherit if not hasattr(self, '_autoInterestHandle'): self.cr.openAutoInterests(self) def generateInit(self): """ This method is called when the DistributedObject is first introduced to the world... Not when it is pulled from the cache. """ self.activeState = ESGenerating def getDoId(self): """ Return the distributed object id """ return self.doId #This message was moved out of announce generate #to avoid ordering issues. def postGenerateMessage(self): if self.activeState != ESGenerated: self.activeState = ESGenerated messenger.send(self.uniqueName("generate"), [self]) def updateRequiredFields(self, dclass, di): dclass.receiveUpdateBroadcastRequired(self, di) self.announceGenerate() self.postGenerateMessage() def updateAllRequiredFields(self, dclass, di): dclass.receiveUpdateAllRequired(self, di) self.announceGenerate() self.postGenerateMessage() def updateRequiredOtherFields(self, dclass, di): # First, update the required fields dclass.receiveUpdateBroadcastRequired(self, di) # Announce generate after updating all the required fields, # but before we update the non-required fields. self.announceGenerate() self.postGenerateMessage() dclass.receiveUpdateOther(self, di) def sendUpdate(self, fieldName, args = [], sendToId = None): if self.cr: dg = self.dclass.clientFormatUpdate( fieldName, sendToId or self.doId, args) self.cr.send(dg) else: assert self.notify.error("sendUpdate failed, because self.cr is not set") def sendDisableMsg(self): self.cr.sendDisableMsg(self.doId) def sendDeleteMsg(self): self.cr.sendDeleteMsg(self.doId) def taskName(self, taskString): return ("%s-%s" % (taskString, self.doId)) def uniqueName(self, idString): return ("%s-%s" % (idString, self.doId)) def getCallbackContext(self, callback, extraArgs = []): # Some objects implement a back-and-forth handshake operation # with the AI via an arbitrary context number. This method # (coupled with doCallbackContext(), below) maps a Python # callback onto that context number so that client code may # easily call the method and wait for a callback, rather than # having to negotiate context numbers. # This method generates a new context number and stores the # callback so that it may later be called when the response is # returned. # This is intended to be called within derivations of # DistributedObject, not directly by other objects. context = self.__nextContext self.__callbacks[context] = (callback, extraArgs) # We assume the context number is passed as a uint16. self.__nextContext = (self.__nextContext + 1) & 0xffff return context def getCurrentContexts(self): # Returns a list of the currently outstanding contexts created # by getCallbackContext(). return list(self.__callbacks.keys()) def getCallback(self, context): # Returns the callback that was passed in to the previous # call to getCallbackContext. return self.__callbacks[context][0] def getCallbackArgs(self, context): # Returns the extraArgs that were passed in to the previous # call to getCallbackContext. return self.__callbacks[context][1] def doCallbackContext(self, context, args): # This is called after the AI has responded to the message # sent via getCallbackContext(), above. The context number is # looked up in the table and the associated callback is # issued. # This is intended to be called within derivations of # DistributedObject, not directly by other objects. tuple = self.__callbacks.get(context) if tuple: callback, extraArgs = tuple completeArgs = args + extraArgs if callback != None: callback(*completeArgs) del self.__callbacks[context] else: self.notify.warning("Got unexpected context from AI: %s" % (context)) def setBarrierData(self, data): # This message is sent by the AI to tell us the barriers and # avIds for which the AI is currently waiting. The client # needs to look up its pending context in the table (and # ignore the other contexts). When the client is done # handling whatever it should handle in its current state, it # should call doneBarrier(), which will send the context # number back to the AI. for context, name, avIds in data: for avId in avIds: if self.cr.isLocalId(avId): # We found the local avatar's id; stop here. self.__barrierContext = (context, name) assert self.notify.debug('setBarrierData(%s, %s)' % (context, name)) return # This barrier didn't involve this client; ignore it. assert self.notify.debug('setBarrierData(%s)' % (None)) self.__barrierContext = None def getBarrierData(self): # Return a trivially-empty (context, name, avIds) value. return ((0, '', []),) def doneBarrier(self, name = None): # Tells the AI we have finished handling our task. If the # optional name parameter is specified, it must match the # barrier name specified on the AI, or the barrier is ignored. # This is used to ensure we are not clearing the wrong # barrier. # If this is None, it either means we have called # doneBarrier() twice, or we have not received a barrier # context from the AI. I think in either case it's ok to # silently ignore the error. if self.__barrierContext != None: context, aiName = self.__barrierContext if name == None or name == aiName: assert self.notify.debug('doneBarrier(%s, %s)' % (context, aiName)) self.sendUpdate("setBarrierReady", [context]) self.__barrierContext = None else: assert self.notify.debug('doneBarrier(%s) ignored; current barrier is %s' % (name, aiName)) else: assert self.notify.debug('doneBarrier(%s) ignored; no active barrier.' % (name)) def addInterest(self, zoneId, note="", event=None): return self.cr.addInterest(self.getDoId(), zoneId, note, event) def removeInterest(self, handle, event=None): return self.cr.removeInterest(handle, event) def b_setLocation(self, parentId, zoneId): self.d_setLocation(parentId, zoneId) self.setLocation(parentId, zoneId) def d_setLocation(self, parentId, zoneId): self.cr.sendSetLocation(self.doId, parentId, zoneId) def setLocation(self, parentId, zoneId): self.cr.storeObjectLocation(self, parentId, zoneId) def getLocation(self): try: if self.parentId == 0 and self.zoneId == 0: return None # This is a -1 stuffed into a uint32 if self.parentId == 0xffffffff and self.zoneId == 0xffffffff: return None return (self.parentId, self.zoneId) except AttributeError: return None def getParentObj(self): if self.parentId is None: return None return self.cr.doId2do.get(self.parentId) def isLocal(self): # This returns true if the distributed object is "local," # which means the client created it instead of the AI, and it # gets some other special handling. Normally, only the local # avatar class overrides this to return true. return self.cr and self.cr.isLocalId(self.doId) def isGridParent(self): # If this distributed object is a DistributedGrid return 1. 0 by default return 0 def execCommand(self, string, mwMgrId, avId, zoneId): pass