from direct.showbase.DirectObject import DirectObject if __debug__: from panda3d.core import PStatCollector class Job(DirectObject): # Base class for cpu-intensive or non-time-critical operations that # are run through the JobManager. # values to yield from your run() generator method Done = object() Continue = None # 'yield None' is acceptable in place of 'yield Job.Continue' Sleep = object() # yield any remaining time for this job until next frame # These priorities determine how many timeslices a job gets relative to other # jobs. A job with priority of 1000 will run 10 times more often than a job # with priority of 100. Priorities = ScratchPad(Min=1, Low=100, Normal=1000, High=10000) _SerialGen = SerialNumGen() def __init__(self, name): self._name = name self._generator = None self._id = Job._SerialGen.next() self._printing = False self._priority = Job.Priorities.Normal self._finished = False if __debug__: self._pstats = PStatCollector("App:Show code:jobManager:%s" % self._name) def destroy(self): del self._name del self._generator del self._printing def getFinishedEvent(self): return 'job-finished-%s' % self._id def run(self): # this is a generator # override and do your processing # yield Job.Continue when possible/reasonable # try not to run longer than the JobManager's timeslice between yields # # when done, yield Job.Done # raise NotImplementedError("don't call down") def getPriority(self): return self._priority def setPriority(self, priority): self._priority = priority def printingBegin(self): self._printing = True def printingEnd(self): self._printing = False def resume(self): # called every time JobManager is going to start running this job """ if self._printing: # we may be suspended/resumed multiple times per frame, that gets spammy # if we need to pick out the output of a job, put a prefix onto each line # of the output print 'JOB:%s:RESUME' % self._name """ pass def suspend(self): # called when JobManager is going to stop running this job for a while """ if self._printing: #print 'JOB:%s:SUSPEND' % self._name pass """ pass def _setFinished(self): self._finished = True self.finished() def isFinished(self): return self._finished def finished(self): # called when the job finishes and has been removed from the JobManager pass def getJobName(self): return self._name def _getJobId(self): return self._id def _getGenerator(self): if self._generator is None: self._generator = self.run() return self._generator def _cleanupGenerator(self): if self._generator is not None: self._generator = None if __debug__: # __dev__ not yet available at this point from direct.showbase.Job import Job class TestJob(Job): def __init__(self): Job.__init__(self, 'TestJob') self._counter = 0 self._accum = 0 self._finished = False def run(self): self.printingBegin() while True: while self._accum < 100: self._accum += 1 print('counter = %s, accum = %s' % (self._counter, self._accum)) yield None self._accum = 0 self._counter += 1 if self._counter >= 100: print('Job.Done') self.printingEnd() yield Job.Done else: yield None def addTestJob(): jobMgr.add(TestJob())