import json, random, threading, re, time, datetime
try:
- import twitter
+ import twitter
except: pass # doesn't matter if we don't have twitter, updating the status just will fall through the try-except if so...
def findnth(haystack, needle, n): #http://stackoverflow.com/a/1884151
return -1
return len(haystack)-len(parts[-1])-len(needle)
-def person(num): return state.db['users'][state.db['ranks'][num]]['realnick']
-def pts(num): return str(state.db['users'][state.db['ranks'][num]]['points'])
-def country(num, default="??"): return lib.mod('userinfo')._get(person(num), 'country', default=default)
+def person(num, throwindexerror=False):
+ try:
+ return state.db['users'][state.db['ranks'][num]]['realnick']
+ except IndexError:
+ if throwindexerror:
+ raise
+ else:
+ return ''
+
+def pts(num):
+ try:
+ return str(state.db['users'][state.db['ranks'][num]]['points'])
+ except IndexError:
+ return 0
+def country(num, default="??"):
+ return lib.mod('userinfo')._get(person(num), 'country', default=default)
+
+class MyTimer(threading._Timer):
+ def __init__(self, *args, **kwargs):
+ threading._Timer.__init__(self, *args, **kwargs)
+ self.daemon = True
class TriviaState(object):
def __init__(self, parent=None, pointvote=False):
self.curqid = None
self.lastqid = None
- if 'starttime' not in self.db or self.db['starttime'] is None:
- self.db['starttime'] = time.time()
+ if 'lastwon' not in self.db or self.db['lastwon'] is None:
+ self.db['lastwon'] = time.time()
if pointvote:
self.getchan().msg("Vote for the next round target points! Options: %s. Vote using !vote <choice>" % (', '.join([str(x) for x in self.db['targetoptions']])))
self.getchan().msg("You have %s seconds." % (self.db['votetimer']))
self.voteamounts = dict([(x, 0) for x in self.db['targetoptions']]) # make a dict {pointsoptionA: 0, pointsoptionB: 0, ...}
- self.pointvote = threading.Timer(self.db['votetimer'], self.endPointVote)
+ self.pointvote = MyTimer(self.db['votetimer'], self.endPointVote)
self.pointvote.start()
else:
self.pointvote = None
revealloc = findnth(''.join(self.hintstr), '*', revealcount)
self.revealpossibilities.remove(revealcount)
self.hintstr[revealloc] = answer[revealloc]
- self.parent.channel(self.chan).bot.msg(self.chan, "\00304,01Here's a hint: %s" % (''.join(self.hintstr)))
+ self.parent.channel(self.chan).bot.fastmsg(self.chan, "\00304,01Here's a hint: %s" % (''.join(self.hintstr)))
self.hintsgiven += 1
if hintnum < self.db['hintnum']:
- self.steptimer = threading.Timer(self.db['hinttimer'], self.nexthint, args=[hintnum+1])
+ self.steptimer = MyTimer(self.db['hinttimer'], self.nexthint, args=[hintnum+1])
self.steptimer.start()
else:
- self.steptimer = threading.Timer(self.db['hinttimer'], self.nextquestion, args=[True])
+ self.steptimer = MyTimer(self.db['hinttimer'], self.nextquestion, args=[True])
self.steptimer.start()
def doGameOver(self):
winner = person(0)
try:
msg("\00312THE GAME IS OVER!!!")
- msg("THE WINNER IS: %s (%s)" % (person(0), pts(0)))
- msg("2ND PLACE: %s (%s)" % (person(1), pts(1)))
- msg("3RD PLACE: %s (%s)" % (person(2), pts(2)))
- [msg("%dth place: %s (%s)" % (i+1, person(i), pts(i))) for i in range(3,10)]
+ msg("THE WINNER IS: %s (%s)" % (person(0, True), pts(0)))
+ msg("2ND PLACE: %s (%s)" % (person(1, True), pts(1)))
+ msg("3RD PLACE: %s (%s)" % (person(2, True), pts(2)))
+ [msg("%dth place: %s (%s)" % (i+1, person(i, True), pts(i))) for i in range(3,10)]
except IndexError: pass
except Exception as e: msg("DERP! %r" % (e))
+ self.db['lastwinner'] = winner
+ self.db['lastwon'] = time.time()
+
if self.db['hofpath'] is not None and self.db['hofpath'] != '':
self.writeHof()
f.seek(insertpos)
f.write((self.db['hofformat']+"\n") % {
'date': time.strftime("%F", time.gmtime()),
- 'duration': str(datetime.timedelta(seconds=time.time()-self.db['starttime'])),
+ 'duration': str(datetime.timedelta(seconds=time.time()-self.db['lastwon'])),
'targetscore': self.db['target'],
'firstperson': person(0),
'firstscore': pts(0),
if self.gameover == True:
return self.doGameOver()
if qskipped:
- self.getchan().msg("\00304Fail! The correct answer was: %s" % (self.hintanswer))
+ self.getchan().fastmsg("\00304Fail! The correct answer was: %s" % (self.hintanswer))
self.missedquestions += 1
else:
self.missedquestions = 0
if skipwait:
self._nextquestion(iteration)
else:
- self.nextquestiontimer = threading.Timer(self.db['questionpause'], self._nextquestion, args=[iteration])
+ self.nextquestiontimer = MyTimer(self.db['questionpause'], self._nextquestion, args=[iteration])
self.nextquestiontimer.start()
def _nextquestion(self, iteration):
nextq[1] = nextq[1].lower()
- qtext = "\00304,01Next up: "
+ qtext = "\00312,01Next up: "
qtext += "(%5d)" % (random.randint(0,99999))
qary = nextq[0].split(None)
qtext += " "
for qword in qary:
qtext += "\00304,01"+qword+"\00301,01"+chr(random.randrange(0x61,0x7A)) #a-z
- self.getbot().msg(self.chan, qtext)
+ self.getbot().fastmsg(self.chan, qtext)
self.curq = nextq
self.curqid = nextqid
if isinstance(self.curq[1], basestring): self.hintanswer = self.curq[1]
else: self.hintanswer = random.choice(self.curq[1])
- self.steptimer = threading.Timer(self.db['hinttimer'], self.nexthint, args=[1])
+ self.steptimer = MyTimer(self.db['hinttimer'], self.nexthint, args=[1])
self.steptimer.start()
def checkanswer(self, answer):
line = ' '.join([str(arg) for arg in args])
if state.checkanswer(line):
state.curq = None
- bot.msg(chan, "\00312%s\003 has it! The answer was \00312%s\003. New score: %d. Rank: %d. Target: %s (%s)." % (user, line, state.addpoint(user), state.rank(user), state.targetuser(user), state.targetpoints(user)))
+ bot.fastmsg(chan, "\00312%s\003 has it! The answer was \00312%s\003. New score: %d. Rank: %d. Target: %s (%s)." % (user, line, state.addpoint(user), state.rank(user), state.targetuser(user), state.targetpoints(user)))
if state.hintsgiven == 0:
bot.msg(chan, "\00312%s\003 got an extra point for getting it before the hints! New score: %d." % (user, state.addpoint(user)))
state.nextquestion()
bot.msg(state.db['chan'], "%s has started the game!" % (user))
state.nextquestion(skipwait=True)
elif state.pointvote is not None:
- bot.msg(replyto, "There's a vote in progress!")
+ bot.msg(user, "There's a vote in progress!")
else:
- bot.msg(replyto, "Game is already started!")
+ bot.msg(user, "Game is already started!")
@lib.hook('stop', glevel=1, needchan=False)
def cmd_stop(bot, user, chan, realtarget, *args):
else:
bot.msg(user, "Game isn't running.")
+@lib.hook('exception')
+def cmd_exception(*args, **kwargs):
+ raise Exception()
+
def stop():
- try:
- if state.curq is not None or state.nextq is not None:
- state.curq = None
- state.nextq = None
- try:
- state.steptimer.cancel()
- except Exception as e:
- print "!!! steptimer.cancel(): %s %r" % (e,e)
- try:
- state.nextquestiontimer.cancel()
- state.nextquestiontimer = None
- except Exception as e:
- print "!!! nextquestiontimer.cancel(): %s %r" % (e,e)
- return True
- else:
- return False
- except NameError:
- pass
+ state.curq = None
+ state.nextq = None
+ try: state.steptimer.cancel()
+ except Exception as e: print "!!! steptimer.cancel(): %s %r" % (e,e)
+ state.steptimer = None
+ try: state.nextquestiontimer.cancel()
+ except Exception as e: print "!!! nextquestiontimer.cancel(): %s %r" % (e,e)
+ state.nextquestiontimer = None
+ return True
@lib.hook(needchan=False)
@lib.argsGE(1)
if max > 10:
max = 10
replylist = ', '.join(["%s (%s) %s" % (person(x), country(x, "unknown"), pts(x)) for x in range(max)])
- bot.msg(state.db['chan'], "Top %d: %s" % (max, replylist))
+ bot.msg(state.db['chan'], "Top 10: %s" % (replylist))
@lib.hook(glevel=lib.ADMIN, needchan=False)
def settarget(bot, user, chan, realtarget, *args):
@lib.hooknum(417)
def num_417(bot, textline):
- bot.msg(state.db['chan'], "Whoops, it looks like that question didn't quite go through! (E:417). Let's try another...")
+ bot.fastmsg(state.db['chan'], "Whoops, it looks like that question didn't quite go through! (E:417). Let's try another...")
state.nextquestion(qskipped=False, skipwait=True)
@lib.hooknum(332)
"%s (%s, %s)" % (person(x), pts(x), country(x))
for x in range(10) if x < len(state.db['ranks'])
]),
+ 'lastwinner': state.db['lastwinner'],
+ 'lastwon': time.strftime("%b %d", time.gmtime(state.db['lastwon'])),
'target': state.db['target'],
}
if gottopic != formatted:
def spellout(num):
return [
- "zero", "one", "two", "three", "four", "five", "six", "seven", "eight",
+ "zero", "one", "two", "three", "four", "five", "six", "seven", "eight",
"nine", "ten", "eleven", "twelve", "thirteen", "fourteen", "fifteen",
"sixteen", "seventeen", "eighteen", "nineteen", "twenty"
][num]