]>
jfr.im git - erebus.git/blob - bot.py
3 # Erebus IRC bot - Author: John Runyon
4 # "Bot" and "BotConnection" classes (handling a specific "arm")
6 import socket
, sys
, time
, threading
7 from collections
import deque
9 #bots = {'erebus': bot.Bot(nick='Erebus', user='erebus', bind='', server='irc.quakenet.org', port=6667, realname='Erebus')}
11 def __init__(self
, parent
, nick
, user
, bind
, authname
, authpass
, server
, port
, realname
):
15 self
.realname
= realname
17 self
.authname
= authname
18 self
.authpass
= authpass
20 curs
= self
.parent
.db
.cursor()
21 if curs
.execute("SELECT chname FROM chans WHERE bot = %s AND active = 1", (self
.nick
,)):
22 chansres
= curs
.fetchall()
24 self
.chans
= [self
.parent
.newchannel(self
, row
['chname']) for row
in chansres
]
26 self
.conn
= BotConnection(self
, bind
, server
, port
)
28 self
.msgqueue
= deque()
29 self
.slowmsgqueue
= deque()
34 if self
.conn
.connect():
35 self
.parent
.newfd(self
, self
.conn
.socket
.fileno())
38 return self
.conn
.read()
40 def parse(self
, line
):
43 if not self
.conn
.registered() and pieces
[0] == "NOTICE":
47 if self
.parent
.hasnumhook(pieces
[1]):
48 hooks
= self
.parent
.getnumhook(pieces
[1])
49 for callback
in hooks
:
52 if pieces
[1] == "001":
53 self
.conn
.registered(True)
54 self
.conn
.send("MODE %s +x" % (pieces
[2]))
55 if self
.authname
is not None and self
.authpass
is not None:
56 self
.conn
.send("AUTH %s %s" % (self
.authname
, self
.authpass
))
60 elif pieces
[1] == "PRIVMSG":
61 nick
= pieces
[0].split('!')[0][1:]
62 user
= self
.parent
.user(nick
)
64 msg
= ' '.join(pieces
[3:])[1:]
65 self
.parsemsg(user
, target
, msg
)
67 elif pieces
[0] == "PING":
68 self
.conn
.send("PONG %s" % (pieces
[1]))
70 elif pieces
[1] == "354": # WHOX
74 self
.parent
.user(nick
).authed(auth
)
76 elif pieces
[1] == "JOIN":
77 nick
= pieces
[0].split('!')[0][1:]
78 chan
= self
.parent
.channel(pieces
[2])
81 self
.conn
.send("WHO %s c%%ant,1" % (chan
))
83 user
= self
.parent
.user(nick
, justjoined
=True)
87 elif pieces
[1] == "PART":
88 nick
= pieces
[0].split('!')[0][1:]
89 chan
= self
.parent
.channel(pieces
[2])
92 self
.parent
.user(nick
).part(chan
)
93 chan
.userpart(self
.parent
.user(nick
))
95 elif pieces
[1] == "QUIT":
96 nick
= pieces
[0].split('!')[0][1:]
98 self
.parent
.user(nick
).quit()
99 del self
.parent
.users
[nick
.lower()]
101 elif pieces
[1] == "NICK":
102 oldnick
= pieces
[0].split('!')[0][1:]
103 newnick
= pieces
[2][1:]
104 if newnick
.lower() != oldnick
.lower():
105 self
.parent
.users
[newnick
.lower()] = self
.parent
.users
[oldnick
.lower()]
106 del self
.parent
.users
[oldnick
.lower()]
107 self
.parent
.users
[newnick
.lower()].nickchange(newnick
)
109 elif pieces
[0] == "ERROR": #TODO handle better
113 elif pieces
[1] == "MODE": #TODO parse for ops/voices (at least)
117 def __debug_cbexception(self
, *args
):
118 if int(self
.parent
.cfg
.get('debug', 'cbexc', default
=0)) == 1:
119 self
.conn
.send("PRIVMSG DimeCadmium :%09.3f ^C4^B!!!^B^C CBEXC" % (time
.time() % 100000))
120 print "%09.3f %s [!] CBEXC %r" % (time
.time() % 100000, self
.nick
, args
)
121 __import__('traceback').print_exc()
124 def parsemsg(self
, user
, target
, msg
):
129 triggerused
= msg
[0] == self
.parent
.trigger
130 if triggerused
: msg
= msg
[1:]
133 if target
== self
.nick
:
134 if msg
[0] == "\001": #ctcp
135 msg
= msg
.strip("\001")
137 self
.msg(user
, "\001VERSION Erebus v%d.%d - http://github.com/zonidjan/erebus" % (self
.parent
.APIVERSION
, self
.parent
.RELEASE
))
141 if chanword
[0] == '#':
142 chan
= self
.parent
.channel(chanword
)
143 if chan
is not None: #if chan is still none, there's no bot on "chanword", and chanword is used as a parameter.
146 else: # message was sent to a channel
147 chan
= self
.parent
.channel(target
)
149 if msg
[0] == '*': # message may be addressed to bot by "*BOTNICK" trigger?
150 if pieces
[0][1:].lower() == self
.nick
.lower():
151 pieces
.pop(0) # command actually starts with next word
152 msg
= ' '.join(pieces
) # command actually starts with next word
153 elif not triggerused
:
154 if self
.parent
.haschanhook(target
.lower()):
155 for callback
in self
.parent
.getchanhook(target
.lower()):
157 cbret
= callback(self
, user
, chan
, *pieces
)
158 if cbret
is NotImplemented: self
.msg(user
, "Command not implemented.")
160 self
.msg(user
, "Command failed. Code: CBEXC%09.3f" % (time
.time() % 100000))
161 self
.__debug
_cbexception
("chanhook", user
, target
, msg
)
162 return # not to bot, don't process!
164 return # "message" is empty
166 cmd
= pieces
[0].lower()
168 if self
.parent
.hashook(cmd
):
169 for callback
in self
.parent
.gethook(cmd
):
170 if chan
is None and callback
.needchan
:
171 self
.msg(user
, "You need to specify a channel for that command.")
172 elif user
.glevel
>= callback
.reqglevel
and (not callback
.needchan
or chan
.levelof(user
.auth
) >= callback
.reqclevel
):
174 cbret
= callback(self
, user
, chan
, target
, *pieces
[1:])
175 if cbret
is NotImplemented: self
.msg(user
, "Command not implemented.")
177 self
.msg(user
, "Command failed. Code: CBEXC%09.3f" % (time
.time() % 100000))
178 self
.__debug
_cbexception
("hook", user
, target
, msg
)
180 def __debug_nomsg(self
, target
, msg
):
181 if int(self
.parent
.cfg
.get('debug', 'nomsg', default
=0)) == 1:
182 self
.conn
.send("PRIVMSG DimeCadmium :%09.3f \ 34\ 2!!!\ 2\ 3 NOMSG %r, %r" % (time
.time() % 100000, target
, msg
))
183 print "%09.3f %s [!] %s" % (time
.time() % 100000, self
.nick
, "!!! NOMSG")
184 __import__('traceback').print_stack()
186 def msg(self
, target
, msg
):
187 if target
is None or msg
is None:
188 return self
.__debug
_nomsg
(target
, msg
)
190 self
.msgqueue
.append((target
, msg
))
191 if not self
.msgtimer
.is_alive():
192 self
.msgtimer
.start()
195 def slowmsg(self
, target
, msg
):
196 if target
is None or msg
is None:
197 return self
.__debug
_nomsg
(target
, msg
)
199 self
.slowmsgqueue
.append((target
,msg
))
200 if not self
.msgtimer
.is_alive():
201 self
.msgtimer
.start()
204 def fastmsg(self
, target
, msg
):
205 if target
is None or msg
is None:
206 return __debug_nomsg(target
, msg
)
208 if isinstance(target
, self
.parent
.User
): target
= target
.nick
209 elif isinstance(target
, self
.parent
.Channel
): target
= target
.name
210 elif not isinstance(target
, basestring
): raise TypeError('Bot.msg() "target" must be Erebus.User, Erebus.Channel, or string')
212 if target
[0] == '#': command
= "PRIVMSG %s :%s" % (target
, msg
)
213 else: command
= "NOTICE %s :%s" % (target
, msg
)
215 self
.conn
.send(command
)
221 self
.fastmsg(*self
.msgqueue
.popleft())
222 self
.msgtimer
.start()
225 self
.fastmsg(*self
.slowmsgqueue
.popleft())
226 self
.msgtimer
.start()
230 def makemsgtimer(self
):
231 self
.msgtimer
= threading
.Timer(2, self
._popmsg
)
232 self
.msgtimer
.daemon
= True
234 def join(self
, chan
):
235 self
.conn
.send("JOIN %s" % (chan
))
237 def part(self
, chan
):
238 self
.conn
.send("PART %s" % (chan
))
240 def quit(self
, reason
="Shutdown"):
241 self
.conn
.send("QUIT :%s" % (reason
))
243 def __str__(self
): return self
.nick
244 def __repr__(self
): return "<Bot %r>" % (self
.nick
)
246 class BotConnection(object):
247 def __init__(self
, parent
, bind
, server
, port
):
254 self
.port
= int(port
)
256 self
.state
= 0 # 0=disconnected, 1=registering, 2=connected
259 self
.socket
= socket
.socket(socket
.AF_INET
, socket
.SOCK_STREAM
)
260 self
.socket
.bind((self
.bind
, 0))
261 self
.socket
.connect((self
.server
, self
.port
))
265 self
.send("NICK %s" % (self
.parent
.nick
))
266 self
.send("USER %s 0 * :%s" % (self
.parent
.user
, self
.parent
.realname
))
270 def registered(self
, done
=False):
271 if done
: self
.state
= 2
272 return self
.state
== 2
274 def send(self
, line
):
275 print "%09.3f %s [O] %s" % (time
.time() % 100000, self
.parent
.nick
, line
)
278 def _write(self
, line
):
279 self
.socket
.sendall(line
+"\r\n")
282 self
.buffer += self
.socket
.recv(8192)
285 while "\r\n" in self
.buffer:
286 pieces
= self
.buffer.split("\r\n", 1)
287 print "%09.3f %s [I] %s" % (time
.time() % 100000, self
.parent
.nick
, pieces
[0])
288 # print (time.time() % 1460000000), self.parent.nick, '[I]', pieces[0]
289 lines
.append(pieces
[0])
290 self
.buffer = pieces
[1]
294 def __str__(self
): return self
.nick
295 def __repr__(self
): return "<BotConnection %r (%r)>" % (self
.socket
.fileno(), self
.parent
.nick
)