#!/usr/bin/python # -*- coding:utf-8 -*- ### # Copyright (c) 2002, Stéphan Kochen # 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. ### """ A module for managing and voting on polls. """ __revision__ = "$Id$" import plugins import os import time import conf import utils import ircdb import ircutils import privmsgs import callbacks try: import sqlite except ImportError: raise callbacks.Error, 'You need to have PySQLite installed to use this ' \ 'plugin. Download it at ' def configure(onStart, afterConnect, advanced): # This will be called by setup.py to configure this module. onStart and # afterConnect are both lists. Append to onStart the commands you would # like to be run when the bot is started; append to afterConnect the # commands you would like to be run when the bot has finished connecting. from questions import expect, anything, something, yn onStart.append('load Poll') class Poll(callbacks.Privmsg, plugins.ChannelDBHandler): def __init__(self): callbacks.Privmsg.__init__(self) plugins.ChannelDBHandler.__init__(self) def makeDb(self, filename): if os.path.exists(filename): db = sqlite.connect(filename) else: db = sqlite.connect(filename) cursor = db.cursor() cursor.execute("""CREATE TABLE polls ( id INTEGER PRIMARY KEY, question TEXT UNIQUE ON CONFLICT IGNORE, started_by INTEGER, open INTEGER)""") cursor.execute("""CREATE TABLE options ( id INTEGER, poll_id INTEGER, option TEXT, UNIQUE (poll_id, id) ON CONFLICT IGNORE)""") cursor.execute("""CREATE TABLE votes ( user_id INTEGER, poll_id INTEGER, option_id INTEGER, UNIQUE (user_id, poll_id) ON CONFLICT IGNORE)""") db.commit() return db def poll(self, irc, msg, args): """ Displays the poll question and options for the given poll id. """ channel = privmsgs.getChannel(msg, args) poll_id = privmsgs.getArgs(args) db = self.getDb(channel) cursor = db.cursor() cursor.execute("""SELECT id, question, started_by, open FROM polls WHERE id=%s""", poll_id) if cursor.rowcount == 0: irc.error('There is no poll with id %s' % poll_id) return _, question, started_by, open = cursor.fetchone() starter = ircdb.users.getUser(started_by).name if open: statusstr = 'open' else: statusstr = 'closed' cursor.execute("""SELECT id, option FROM options WHERE poll_id=%s""", poll_id) if cursor.rowcount == 0: optionstr = 'This poll has no options yet' else: options = cursor.fetchall() optionstr = 'Options:' optionstr += ''.join([' %s: %r' % (id, option) for id, option in options]) pollstr = 'Poll #%s: %r started by %s. %s. Poll is %s.' % \ (poll_id, question, starter, optionstr, statusstr) irc.reply(pollstr) def open(self, irc, msg, args): """[] Creates a new poll with the given question. """ channel = privmsgs.getChannel(msg, args) question = privmsgs.getArgs(args) # Must be registered to create a poll try: userId = ircdb.users.getUserId(msg.prefix) except KeyError: irc.errorNotRegistered() return db = self.getDb(channel) cursor = db.cursor() cursor.execute("""INSERT INTO polls VALUES (NULL, %s, %s, 1)""", question, userId) db.commit() cursor.execute("""SELECT id FROM polls WHERE question=%s""", question) id = cursor.fetchone()[0] irc.replySuccess('(poll #%s)' % id) def close(self, irc, msg, args): """[] Closes the poll with the given ; further votes will not be allowed. """ channel = privmsgs.getChannel(msg, args) id = privmsgs.getArgs(args) try: id = int(id) except ValueError: irc.error('The id must be an integer.') return db = self.getDb(channel) cursor = db.cursor() # Check to make sure that the poll exists cursor.execute("""SELECT id FROM polls WHERE id=%s""", id) if cursor.rowcount == 0: irc.error('Id #%s is not an existing poll.') return cursor.execute("""UPDATE polls SET open=0 WHERE id=%s""", id) irc.replySuccess() def add(self, irc, msg, args): """[]