]> jfr.im git - erebus.git/blame - erebus.py
Preparing to develop autotests
[erebus.git] / erebus.py
CommitLineData
f4f6442e
JR
1#!/usr/bin/python
2
db75daab
JR
3# Erebus IRC bot - Author: John Runyon
4# main startup code
5
f4f6442e
JR
6#TODO: tons
7
82696467 8import os, sys, select, MySQLdb, MySQLdb.cursors
d1ea05b0 9import bot, config, ctlmod
f4f6442e
JR
10
11class Erebus(object):
9c935294
JR
12 bots = {}
13 fds = {}
14 mods = {}
15 msghandlers = {}
1743224e
JR
16 users = {}
17 chans = {}
9c935294
JR
18
19 class User(object):
20 chans = []
ab7c0c0b 21
9c935294
JR
22 def __init__(self, nick, auth=None):
23 self.nick = nick
1743224e 24 self.auth = auth
7d69441a 25 self.checklevel()
ab7c0c0b 26
1743224e
JR
27 def isauthed(self):
28 return self.auth is not None
29
9c935294 30 def authed(self, auth):
1743224e 31 if auth == '0': auth = None
9c935294
JR
32 self.auth = auth
33 self.checklevel()
ab7c0c0b 34
7d69441a
JR
35 def checklevel(self):
36 if self.auth is None:
67e04ca5 37 self.glevel = -1
7d69441a
JR
38 else:
39 c = main.db.cursor()
40 c.execute("SELECT level FROM users WHERE auth = %s", (self.auth,))
41 row = c.fetchone()
42 if row is not None:
67e04ca5 43 self.glevel = row['level']
7d69441a 44 else:
67e04ca5
JR
45 self.glevel = 0
46 return self.glevel
f9767f79 47
9c935294 48 def __str__(self): return self.nick
67e04ca5 49 def __repr__(self): return "<User %r (%d)>" % (self.nick,self.glevel)
f9767f79 50
9c935294
JR
51 class Channel(object):
52 users = []
53 voices = []
54 ops = []
ab7c0c0b 55
9c935294
JR
56 def __init__(self, name):
57 self.name = name
ab7c0c0b 58
9c935294
JR
59 def userjoin(self, user, level=None):
60 if user not in self.users: self.users.append(user)
61 if level == 'op' and user not in self.ops: self.ops.append(user)
62 if level == 'voice' and user not in self.voices: self.voices.append(user)
63 def userpart(self, user):
64 if user in self.ops: self.ops.remove(user)
65 if user in self.voices: self.voices.remove(user)
66 if user in self.users: self.users.remove(user)
ab7c0c0b 67
9c935294
JR
68 def userop(self, user):
69 if user in self.users and user not in self.ops: self.ops.append(user)
70 def uservoice(self, user):
71 if user in self.users and user not in self.voices: self.voices.append(user)
72 def userdeop(self, user):
73 if user in self.ops: self.ops.remove(user)
74 def userdevoice(self, user):
75 if user in self.voices: self.voices.remove(user)
76
77 def __str__(self): return self.name
78 def __repr__(self): return "<Channel %r>" % (self.name)
79
1743224e
JR
80 def __init__(self, trigger):
81 self.trigger = trigger
c5f1f47b
JR
82 if os.name == "posix":
83 self.potype = "poll"
84 self.po = select.poll()
85 else: # f.e. os.name == "nt" (Windows)
86 self.potype = "select"
87 self.fdlist = []
9c935294
JR
88
89 def newbot(self, nick, user, bind, server, port, realname, chans):
90 if bind is None: bind = ''
91 obj = bot.Bot(self, nick, user, bind, server, port, realname, chans)
92 self.bots[nick.lower()] = obj
ab7c0c0b 93
9c935294 94 def newfd(self, obj, fileno):
9c935294 95 self.fds[fileno] = obj
c5f1f47b
JR
96 if self.potype == "poll":
97 self.po.register(fileno, select.POLLIN)
98 elif self.potype == "select":
99 self.fdlist.append(fileno)
ab7c0c0b 100
f9767f79 101 def bot(self, name): #get Bot() by name (nick)
9c935294 102 return self.bots[name.lower()]
f9767f79 103 def fd(self, fileno): #get Bot() by fd/fileno
9c935294
JR
104 return self.fds[fileno]
105
1743224e
JR
106 def user(self, nick):
107 nick = nick.lower()
108 if nick in self.users:
109 return self.users[nick]
110 else:
111 user = self.User(nick)
112 self.users[nick] = user
113 return user
f9767f79 114 def channel(self, name): #TODO #get Channel() by name
9c935294
JR
115 return self.Channel(name.lower())
116
117 def poll(self):
c5f1f47b
JR
118 if self.potype == "poll":
119 return [fd for (fd, ev) in self.po.poll()]
120 elif self.potype == "select":
121 return select.select(self.fdlist, [], [])[0]
9c935294
JR
122
123 def connectall(self):
124 for bot in self.bots.itervalues():
125 if bot.conn.state == 0:
126 bot.connect()
127
9c935294 128 #bind functions
d1ea05b0 129 def hook(self, word, handler):
d1ea05b0
JR
130 self.msghandlers[word] = handler
131 def unhook(self, word):
132 del self.msghandlers[word]
133 def hashook(self, word):
134 return word in self.msghandlers
135 def gethook(self, word):
136 return self.msghandlers[word]
f4f6442e
JR
137
138def setup():
d1ea05b0
JR
139 global cfg, main
140
141 cfg = config.Config('bot.config')
1743224e 142 main = Erebus(cfg.trigger)
d1ea05b0
JR
143
144 autoloads = [mod for mod, yes in cfg.items('autoloads') if int(yes) == 1]
145 for mod in autoloads:
146 ctlmod.load(main, mod)
147
82696467
JR
148 main.db = MySQLdb.connect(host=cfg.dbhost, user=cfg.dbuser, passwd=cfg.dbpass, db=cfg.dbname, cursorclass=MySQLdb.cursors.DictCursor)
149 c = main.db.cursor()
150 c.execute("SELECT nick, user, bind FROM bots WHERE active = 1")
151 rows = c.fetchall()
152 c.close()
153 for row in rows:
154 c2 = main.db.cursor()
155 c2.execute("SELECT chname FROM chans WHERE bot = %s AND active = 1", (row['nick'],))
156 chans = [chdic['chname'] for chdic in c2.fetchall()]
157 c2.close()
158 main.newbot(row['nick'], row['user'], row['bind'], cfg.host, cfg.port, cfg.realname, chans)
159 main.connectall()
f4f6442e
JR
160
161def loop():
9c935294 162 poready = main.poll()
c5f1f47b 163 for fileno in poready:
711444b7
JR
164 for line in main.fd(fileno).getdata():
165 main.fd(fileno).parse(line)
f4f6442e
JR
166
167if __name__ == '__main__':
168 setup()
9c935294 169 while True: loop()