Limnoria/src/schedule.py

153 lines
5.3 KiB
Python
Raw Normal View History

2003-03-12 07:26:59 +01:00
#!/usr/bin/env python
###
# Copyright (c) 2002-2004, Jeremiah Fincher
2003-03-12 07:26:59 +01:00
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:
#
# * Redistributions of source code must retain the above copyright notice,
# this list of conditions, and the following disclaimer.
# * Redistributions in binary form must reproduce the above copyright notice,
# this list of conditions, and the following disclaimer in the
# documentation and/or other materials provided with the distribution.
# * Neither the name of the author of this software nor the name of
# contributors to this software may be used to endorse or promote products
# derived from this software without specific prior written consent.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
# POSSIBILITY OF SUCH DAMAGE.
###
2003-10-04 14:45:44 +02:00
"""
Schedule plugin with a subclass of drivers.IrcDriver in order to be run as a
Supybot driver.
"""
2003-11-25 09:38:19 +01:00
__revision__ = "$Id$"
2004-07-24 07:18:26 +02:00
import supybot.fix as fix
2003-03-12 07:26:59 +01:00
import time
import heapq
2003-03-12 07:26:59 +01:00
2004-07-24 07:18:26 +02:00
import supybot.log as log
import supybot.world as world
2004-07-24 07:18:26 +02:00
import supybot.drivers as drivers
2003-03-12 07:26:59 +01:00
class mytuple(tuple):
def __cmp__(self, other):
return cmp(self[0], other[0])
def __le__(self, other):
return self[0] <= other[0]
def __lt__(self, other):
return self[0] < other[0]
def __gt__(self, other):
return self[0] > other[0]
def __ge__(self, other):
return self[0] >= other[0]
2003-08-20 18:26:23 +02:00
2003-03-12 07:26:59 +01:00
class Schedule(drivers.IrcDriver):
2003-10-04 14:45:44 +02:00
"""An IrcDriver to handling scheduling of events.
Events, in this case, are functions accepting no arguments.
"""
2003-03-12 07:26:59 +01:00
def __init__(self):
drivers.IrcDriver.__init__(self)
self.schedule = []
self.events = {}
self.counter = 0
2004-02-13 05:43:06 +01:00
def reset(self):
self.events.clear()
self.schedule[:] = []
2004-04-13 02:06:57 +02:00
# We don't reset the counter here because if someone has held an id of
# one of the nuked events, we don't want him removing new events with
# his old id.
2004-02-13 05:43:06 +01:00
def name(self):
return 'Schedule'
2003-03-12 07:26:59 +01:00
def addEvent(self, f, t, name=None):
2003-10-04 14:45:44 +02:00
"""Schedules an event f to run at time t.
name must be hashable and not an int.
"""
2003-03-12 07:26:59 +01:00
if name is None:
name = self.counter
self.counter += 1
assert name not in self.events
self.events[name] = f
heapq.heappush(self.schedule, mytuple((t, name)))
2004-01-01 21:00:29 +01:00
return name
2003-03-12 07:26:59 +01:00
def removeEvent(self, name):
2003-10-04 14:45:44 +02:00
"""Removes the event with the given name from the schedule."""
2004-08-19 18:58:42 +02:00
f = self.events.pop(name)
2003-03-12 07:26:59 +01:00
self.schedule = [(t, n) for (t, n) in self.schedule if n != name]
2004-04-13 02:12:23 +02:00
# We must heapify here because the heap property may not be preserved
# by the above list comprehension. We could, conceivably, just mark
# the elements of the heap as removed and ignore them when we heappop,
# but that would only save a constant factor (we're already linear for
# the listcomp) so I'm not worried about it right now.
heapq.heapify(self.schedule)
2004-08-19 18:58:42 +02:00
return f
def rescheduleEvent(self, name, t):
f = self.removeEvent(name)
self.addEvent(f, t, name=name)
2003-03-12 07:26:59 +01:00
def addPeriodicEvent(self, f, t, name=None, now=True):
2003-10-04 14:45:44 +02:00
"""Adds a periodic event that is called every t seconds."""
2003-03-12 07:26:59 +01:00
def wrapper():
2004-02-12 14:27:02 +01:00
try:
f()
finally:
# Even if it raises an exception, let's schedule it.
return self.addEvent(wrapper, time.time() + t, name)
if now:
return wrapper()
else:
return self.addEvent(wrapper, time.time() + t, name)
2003-03-12 07:26:59 +01:00
removePeriodicEvent = removeEvent
def run(self):
if len(drivers._drivers) == 1 and not world.testing:
2004-08-25 06:25:06 +02:00
log.error('Schedule is the only remaining driver, '
'why do we continue to live?')
time.sleep(1) # We're the only driver; let's pause to think.
while self.schedule and self.schedule[0][0] < time.time():
(t, name) = heapq.heappop(self.schedule)
f = self.events[name]
2003-03-12 07:26:59 +01:00
del self.events[name]
2004-02-12 14:27:02 +01:00
try:
f()
except Exception, e:
log.exception('Uncaught exception in scheduled function:')
2003-03-12 07:26:59 +01:00
try:
ignore(schedule)
except NameError:
schedule = Schedule()
addEvent = schedule.addEvent
removeEvent = schedule.removeEvent
2004-08-19 18:58:42 +02:00
rescheduleEvent = schedule.rescheduleEvent
2003-03-12 07:26:59 +01:00
addPeriodicEvent = schedule.addPeriodicEvent
removePeriodicEvent = removeEvent
run = schedule.run
2004-01-02 23:29:03 +01:00
# vim:set shiftwidth=4 tabstop=8 expandtab textwidth=78: