Scheduler: Add --delay option, to add an offset before the first run.

Closes GH-397.
This commit is contained in:
Valentin Lorentz 2020-05-02 20:02:06 +02:00
parent ad05468257
commit 482658529b
2 changed files with 24 additions and 13 deletions

View File

@ -188,15 +188,9 @@ class Scheduler(callbacks.Plugin):
irc.error(_('Invalid event id.')) irc.error(_('Invalid event id.'))
remove = wrap(remove, ['lowered']) remove = wrap(remove, ['lowered'])
def _repeat(self, irc, msg, name, seconds, command, first_run=None, next_run_in=None): def _repeat(self, irc, msg, name, seconds, command, first_run, next_run_in):
f = self._makeCommandFunction(irc, msg, command, remove=False) f = self._makeCommandFunction(irc, msg, command, remove=False)
f_wrapper = schedule.schedule.makePeriodicWrapper(f, seconds, name) f_wrapper = schedule.schedule.makePeriodicWrapper(f, seconds, name)
if next_run_in is None:
assert first_run is None
# run immediately
id = f_wrapper()
first_run = time.time()
else:
assert first_run is not None assert first_run is not None
id = schedule.addEvent(f_wrapper, time.time() + next_run_in, name) id = schedule.addEvent(f_wrapper, time.time() + next_run_in, name)
assert id == name assert id == name
@ -208,23 +202,29 @@ class Scheduler(callbacks.Plugin):
} }
@internationalizeDocstring @internationalizeDocstring
def repeat(self, irc, msg, args, name, seconds, command): def repeat(self, irc, msg, args, optlist, name, seconds, command):
"""<name> <seconds> <command> """[--delay <delay>] <name> <seconds> <command>
Schedules the command <command> to run every <seconds> seconds, Schedules the command <command> to run every <seconds> seconds,
starting now (i.e., the command runs now, and every <seconds> seconds starting now (i.e., the command runs now, and every <seconds> seconds
thereafter). <name> is a name by which the command can be thereafter). <name> is a name by which the command can be
unscheduled. unscheduled.
If --delay is given, starts in <delay> seconds instead of now.
""" """
opts = dict(optlist)
name = name.lower() name = name.lower()
if name in self.events: if name in self.events:
irc.error(_('There is already an event with that name, please ' irc.error(_('There is already an event with that name, please '
'choose another name.'), Raise=True) 'choose another name.'), Raise=True)
self._repeat(irc, msg, name, seconds, command) next_run_in = opts.get('delay', 0)
first_run = time.time() + next_run_in
self._repeat(irc, msg, name, seconds, command, first_run, next_run_in)
# We don't reply because the command runs immediately. # We don't reply because the command runs immediately.
# But should we? What if the command doesn't have visible output? # But should we? What if the command doesn't have visible output?
# irc.replySuccess() # irc.replySuccess()
repeat = wrap(repeat, ['nonInt', 'positiveInt', 'text']) repeat = wrap(repeat, [
getopts({'delay': 'positiveInt'}),
'nonInt', 'positiveInt', 'text'])
@internationalizeDocstring @internationalizeDocstring
def list(self, irc, msg, args): def list(self, irc, msg, args):

View File

@ -87,6 +87,17 @@ class SchedulerTestCase(ChannelPluginTestCase):
timeFastForward(5) timeFastForward(5)
self.assertNoResponse(' ', timeout=1) self.assertNoResponse(' ', timeout=1)
def testRepeatDelay(self):
self.assertNoResponse(
'scheduler repeat --delay 5 repeater 20 echo testRepeat',
timeout=1)
timeFastForward(5)
self.assertResponse(' ', 'testRepeat', timeout=1)
timeFastForward(17)
self.assertNoResponse(' ', timeout=1)
timeFastForward(5)
self.assertResponse(' ', 'testRepeat', timeout=1)
def testRepeatWorksWithNestedCommands(self): def testRepeatWorksWithNestedCommands(self):
self.assertRegexp('scheduler repeat foo 5 "echo foo [echo nested]"', self.assertRegexp('scheduler repeat foo 5 "echo foo [echo nested]"',
'foo nested') 'foo nested')