# Written by Scaevolus 2010 from util import hook, http import string import re re_lineends = re.compile(r'[\r\n]*') # some simple "shortcodes" for formatting purposes shortcodes = { '[b]': '\x02', '[/b]': '\x02', '[u]': '\x1F', '[/u]': '\x1F', '[i]': '\x16', '[/i]': '\x16'} def db_init(db): db.execute("create table if not exists mem(word, data, nick," " primary key(word))") db.commit() def get_memory(db, word): row = db.execute("select data from mem where word=lower(?)", [word]).fetchone() if row: return row[0] else: return None def multiwordReplace(text, wordDic): """ take a text and replace words that match a key in a dictionary with the associated value, return the changed text """ rc = re.compile('|'.join(map(re.escape, wordDic))) def translate(match): return wordDic[match.group(0)] return rc.sub(translate, text) @hook.command("r", adminonly=True) @hook.command(adminonly=True) def remember(inp, nick='', db=None, say=None, input=None, notice=None): "remember [+] -- Remembers with . Add +" " to to append." db_init(db) append = False try: head, tail = inp.split(None, 1) except ValueError: return remember.__doc__ data = get_memory(db, head) if tail[0] == '+': append = True # ignore + symbol new = tail[1:] _head, _tail = data.split(None, 1) # data is stored with the input so ignore it when re-adding it if len(tail) > 1 and tail[1] in (string.punctuation + ' '): tail = _tail + new else: tail = _tail + ' ' + new db.execute("replace into mem(word, data, nick) values" " (lower(?),?,?)", (head, tail, nick)) db.commit() if data: if append: notice("Appending %s to %s" % (new, data.replace('"', "''"))) else: notice('Forgetting existing data (%s), remembering this instead!' % data.replace('"', "''")) return else: notice('Remembered!') return @hook.command("f", adminonly=True) @hook.command(adminonly=True) def forget(inp, db=None, input=None, notice=None): "forget -- Forgets a remembered ." db_init(db) data = get_memory(db, inp) if data: db.execute("delete from mem where word=lower(?)", [inp]) db.commit() notice('"%s" has been forgotten.' % data.replace('`', "'")) return else: notice("I don't know about that.") return @hook.command() def info(inp, notice=None, db=None): "info -- Shows the source of a factoid." db_init(db) # attempt to get the factoid from the database data = get_memory(db, inp.strip()) if data: notice(data) else: notice("Unknown Factoid") @hook.regex(r'^\? ?(.+)') def factoid(inp, say=None, db=None, bot=None, me=None, conn=None, input=None): "? -- Shows what data is associated with ." try: prefix_on = bot.config["plugins"]["factoids"].get("prefix", False) except KeyError: prefix_on = False db_init(db) # split up the input split = inp.group(1).strip().split(" ") factoid_id = split[0] if len(split) >= 1: arguments = " ".join(split[1:]) else: arguments = None # attempt to get the factoid from the database data = get_memory(db, factoid_id) if data: # dynamic variables for factoids data = data.replace("$nick", input.nick) data = data.replace("$chan", input.chan) data = data.replace("$botnick", conn.nick) # if factoid needs input, do that if "$inp" in data: if arguments: data = data.replace("$inp", arguments) else: return "This factoid requires input. You can provide this" \ " with ?%s " % factoid_id # if , execute python code if data.startswith(""): data = data[4:].strip() res = http.get("http://eval.appspot.com/eval", statement=data).splitlines() if len(res) == 0: return res[0] = re_lineends.split(res[0])[0] if not res[0] == 'Traceback (most recent call last):': result = res[0].decode('utf8', 'ignore') else: result = res[-1].decode('utf8', 'ignore') else: result = data # process formatting "shortcodes" result = multiwordReplace(result, shortcodes) # return final dataput if result.startswith(""): result = result[5:].strip() me(result) else: if prefix_on: say("\x02[%s]:\x02 %s" % (factoid_id, result)) else: say(result)