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
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)
+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):
if parent is not None:
self.gotParent(parent, pointvote)
def gotParent(self, parent, pointvote=False):
- self.parent = parent
- self.questionfile = self.parent.cfg.get('trivia', 'jsonpath', default="./modules/trivia.json")
- self.db = json.load(open(self.questionfile, "r"))
- self.chan = self.db['chan']
- self.curq = None
- self.nextq = None
- self.nextquestiontimer = None
- self.steptimer = None
- self.hintstr = None
- self.hintanswer = None
- self.hintsgiven = 0
+ self.parent = parent
+ self.questionfile = self.parent.cfg.get('trivia', 'jsonpath', default="./modules/trivia.json")
+ self.db = json.load(open(self.questionfile, "r"))
+ self.chan = self.db['chan']
+ self.curq = None
+ self.nextq = None
+ self.nextquestiontimer = None
+ self.steptimer = None
+ self.hintstr = None
+ self.hintanswer = None
+ self.hintsgiven = 0
self.revealpossibilities = None
- self.gameover = False
- self.missedquestions = 0
- self.curqid = None
- self.lastqid = None
+ self.gameover = False
+ self.missedquestions = 0
+ self.curqid = None
+ self.lastqid = None
if 'starttime' not in self.db or self.db['starttime'] is None:
self.db['starttime'] = time.time()
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):
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()
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: "
- if nextqid is not None:
- qtext += "(%d) " % (nextqid)
+ 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()
try:
qid = int(args[0])
state.nextq = state.db['questions'][qid]
- bot.msg(user, "Done. Next question is: %s" % (state.nextq[0]))
+ if user.glevel >= lib.STAFF:
+ respstr = "Done. Next question is: %s" % (state.nextq[0])
+ else:
+ respstr = "Done."
+ bot.msg(user, respstr)
except Exception as e:
bot.msg(user, "Error: %s" % (e))
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:
@lib.hook(glevel=1, needchan=False)
def findq(bot, user, chan, realtarget, *args):
+ if len(args) == 0:
+ bot.msg(user, "You need to specify a search string.")
+ return
+
searcher = re.compile(' '.join(args))
matches = [str(i) for i in range(len(state.db['questions'])) if searcher.search(state.db['questions'][i][0]) is not None]
- if len(matches) > 1:
+ if len(matches) > 25:
+ bot.msg(user, "Too many matches! (>25)")
+ elif len(matches) > 1:
bot.msg(user, "Multiple matches: %s" % (', '.join(matches)))
elif len(matches) == 1:
bot.msg(user, "One match: %s" % (matches[0]))
bot.msg(user, "TOP10")
bot.msg(user, "POINTS [<user>]")
bot.msg(user, "RANK [<user>]")
- bot.msg(user, "BADQ <id> <reason>")
+ bot.msg(user, "BADQ <reason> (include info to identify question)")
if user.glevel >= 1:
bot.msg(user, "SKIP (>=KNOWN)")
bot.msg(user, "STOP (>=KNOWN)")
"%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: