2014-10-27 16:29:04 -06:00
|
|
|
import threading
|
2015-01-18 17:19:39 -06:00
|
|
|
import logging
|
|
|
|
log = logging.getLogger("mysc.repeating_timer")
|
2014-10-27 16:29:04 -06:00
|
|
|
|
|
|
|
class RepeatingTimer(threading.Thread):
|
|
|
|
"""Call a function after a specified number of seconds, it will then repeat again after the specified number of seconds
|
|
|
|
Note: If the function provided takes time to execute, this time is NOT taken from the next wait period
|
|
|
|
|
|
|
|
t = RepeatingTimer(30.0, f, args=[], kwargs={})
|
|
|
|
t.start()
|
|
|
|
t.cancel() # stop the timer's actions
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, interval, function, daemon=True, *args, **kwargs):
|
|
|
|
threading.Thread.__init__(self)
|
|
|
|
self.daemon = daemon
|
|
|
|
self.interval = float(interval)
|
|
|
|
self.function = function
|
|
|
|
self.args = args
|
|
|
|
self.kwargs = kwargs
|
|
|
|
self.finished = threading.Event()
|
|
|
|
|
|
|
|
def cancel(self):
|
|
|
|
"""Stop the timer if it hasn't finished yet"""
|
2015-01-18 17:19:39 -06:00
|
|
|
log.debug("Stopping repeater for %s" % (self.function,))
|
2014-10-27 16:29:04 -06:00
|
|
|
self.finished.set()
|
|
|
|
stop = cancel
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
while not self.finished.is_set():
|
|
|
|
self.finished.wait(self.interval)
|
|
|
|
if not self.finished.is_set(): #In case someone has canceled while waiting
|
2015-01-18 17:19:39 -06:00
|
|
|
try:
|
|
|
|
self.function(*self.args, **self.kwargs)
|
|
|
|
except:
|
|
|
|
log.exception("Execution failed. Function: %r args: %r and kwargs: %r" % (self.function, self.args, self.kwargs))
|