This commit is contained in:
Ryan Hitchman 2009-11-07 11:41:51 -07:00
commit 0602629f98
4 changed files with 32 additions and 123 deletions

10
bot.py
View File

@ -1,8 +1,8 @@
#!/usr/bin/python
network = "irc.synirc.net"
network = "localhost"
nick = "skybot"
channel = "#cobol"
channel = "#test"
import sys
import os
@ -12,25 +12,25 @@ sys.path += ['plugins'] # so 'import hook' works without duplication
sys.path += ['lib']
os.chdir(sys.path[0]) # do stuff relative to the installation directory
import irc
class Bot(object):
pass
bot = Bot()
print 'Loading plugins'
# bootstrap the reloader
eval(compile(open('core/reload.py', 'U').read(), 'core/reload.py', 'exec'))
reload(init=True)
print 'Connecting to IRC'
bot.nick = nick
bot.channel = channel
bot.network = network
bot.irc = irc.irc(network, nick)
bot.irc = irc(network, nick)
bot.irc.join(channel)
bot.persist_dir = os.path.abspath('persist')

View File

@ -8,11 +8,15 @@ if 'mtimes' not in globals():
if 'lastfiles' not in globals():
lastfiles = set()
def reload():
init = False
if not hasattr(bot, 'plugs'):
def format_plug(plug, lpad=0, width=40):
out = ' ' * lpad + '%s:%s:%s' % (plug[0])
if len(plug) == 3 and 'hook' in plug[2]:
out += '%s%s' % (' ' * (width - len(out)), plug[2]['hook'])
return out
def reload(init=False):
if init:
bot.plugs = collections.defaultdict(lambda: [])
init = True
for filename in glob.glob("core/*.py"):
mtime = os.stat(filename).st_mtime
@ -26,7 +30,7 @@ def reload():
continue
if filename == 'core/reload.py':
reload()
reload(init=init)
return
fileset = set(glob.glob("plugins/*py"))
@ -54,15 +58,20 @@ def reload():
for type, data in obj._skybot_hook:
bot.plugs[type] += [data]
if not init:
print '### new plugin (type: %s) loaded:' % type, format_plug(data)
if type == 'init': # run-once functions
try:
obj(bot) # not thread-safe!
except Exception:
traceback.print_exc(Exception)
if init:
print ' plugin listing:'
for type, plugs in sorted(bot.plugs.iteritems()):
print ' %s:' % type
for plug in plugs:
out = ' %s:%s:%s' % (plug[0])
print out,
if len(plug) == 3 and 'hook' in plug[2]:
print '%s%s' % (' ' * (40 - len(out)), plug[2]['hook'])
else:
print
print format_plug(plug, lpad=6)
print

108
irc.py
View File

@ -1,108 +0,0 @@
import sys
import re
import socket
import thread
import asyncore
import asynchat
import Queue
def decode(txt):
for codec in ('utf-8', 'iso-8859-1', 'shift_jis', 'cp1252'):
try:
return txt.decode(codec)
except UnicodeDecodeError:
continue
return txt.decode('utf-8', 'ignore')
class crlf_tcp(asynchat.async_chat):
"Handles tcp connections that consist of utf-8 lines ending with crlf"
def __init__(self, host, port):
asynchat.async_chat.__init__(self)
self.set_terminator('\r\n')
self.buffer = ""
self.obuffer = ""
self.oqueue = Queue.Queue() #where we stick things that need to be sent
self.iqueue = Queue.Queue() #where we stick things that were received
self.create_socket(socket.AF_INET, socket.SOCK_STREAM)
self.socket.setsockopt(socket.SOL_TCP, socket.TCP_NODELAY, 0)
self.host = host
self.port = port
def run(self):
self.connect((self.host, self.port))
asyncore.loop()
def handle_connect(self):
thread.start_new_thread(self.queue_read_loop, ())
def queue_read_loop(self):
while True:
line = self.oqueue.get().splitlines()[0][:500]
print ">>> %r" % line
self.push(line.encode('utf-8', 'replace') + '\r\n')
def collect_incoming_data(self, data):
self.buffer += data
def found_terminator(self):
line = self.buffer
self.iqueue.put(decode(line))
self.buffer = ''
irc_prefix_rem = re.compile(r'(.*?) (.*?) (.*)').match
irc_noprefix_rem = re.compile(r'()(.*?) (.*)').match
irc_netmask_rem = re.compile(r':?([^!@]*)!?([^@]*)@?(.*)').match
irc_param_ref = re.compile(r'(?:^|(?<= ))(:.*|[^ ]+)').findall
class irc(object):
"handles the IRC protocol"
#see the docs/ folder for more information on the protocol
def __init__(self, network, nick, port=6667):
self.conn = crlf_tcp(network, port)
thread.start_new_thread(self.conn.run, ())
self.out = Queue.Queue() #responses from the server are placed here
# format: [rawline, prefix, command, params,
# nick, user, host, paramlist, msg]
self.nick(nick)
self.cmd("USER", ["skybot v0.01", "0", "bot"])
thread.start_new_thread(self.parse_loop, ())
def parse_loop(self):
while True:
msg = self.conn.iqueue.get()
if msg.startswith(":"): #has a prefix
prefix, command, params = irc_prefix_rem(msg).groups()
else:
prefix, command, params = irc_noprefix_rem(msg).groups()
nick, user, host = irc_netmask_rem(prefix).groups()
paramlist = irc_param_ref(params)
lastparam = ""
if paramlist and paramlist[-1].startswith(':'):
lastparam = paramlist[-1][1:]
self.out.put([msg, prefix, command, params, nick, user, host,
paramlist, lastparam])
if command == "PING":
self.cmd("PONG", [params])
def nick(self, nick):
self.cmd("NICK", [nick])
def join(self, channel):
self.cmd("JOIN", [":"+channel])
def msg(self, target, text):
self.cmd("PRIVMSG", [target, ":"+text])
def cmd(self, command, params=None):
if params:
self.send(command+' '+' '.join(params))
else:
self.send(command)
def send(self, str):
self.conn.oqueue.put(str)

View File

@ -30,6 +30,14 @@ def sieve(func):
return func
def init(func):
if func.func_code.co_argcount != 1:
raise ValueError(
'initializers must take 1 argument: bot')
_hook_add(func, ['init', (_make_sig(func), func)])
return func
def command(func=None, hook=None, **kwargs):
args = {}