2 # vim: fileencoding=utf-8
4 # Erebus IRC bot - Author: John Runyon
5 # "Bot" and "BotConnection" classes (handling a specific "arm")
7 import socket
, sys
, time
, threading
, os
, random
8 from collections
import deque
10 MAXLEN
= 400 # arbitrary max length of a command generated by Bot.msg functions
12 if sys
.version_info
.major
< 3:
13 timerbase
= threading
._Timer
15 timerbase
= threading
.Timer
16 class MyTimer(timerbase
):
17 def __init__(self
, *args
, **kwargs
):
18 timerbase
.__init
__(self
, *args
, **kwargs
)
22 #bots = {'erebus': bot.Bot(nick='Erebus', user='erebus', bind='', server='irc.quakenet.org', port=6667, realname='Erebus')}
24 def __init__(self
, parent
, nick
, user
, bind
, authname
, authpass
, server
, port
, realname
):
29 self
.realname
= realname
31 self
.authname
= authname
32 self
.authpass
= authpass
34 curs
= self
.parent
.query("SELECT chname FROM chans WHERE bot = %s AND active = 1", (self
.nick
,))
36 chansres
= curs
.fetchall()
38 self
.chans
= [self
.parent
.newchannel(self
, row
['chname']) for row
in chansres
]
42 self
.conn
= BotConnection(self
, bind
, server
, port
)
44 self
.lastreceived
= time
.time() #time we last received a line from the server
47 self
.msgqueue
= deque()
48 self
.slowmsgqueue
= deque()
54 curs
= self
.parent
.query("UPDATE bots SET connected = 0 WHERE nick = %s", (self
.nick
,))
59 if time
.time() > int(self
.parent
.cfg
.get('watchdog', 'maxtime', default
=300))+self
.lastreceived
:
60 self
.parse("ERROR :Fake-error from watchdog timer.")
61 self
.watchdogtimer
= MyTimer(int(self
.parent
.cfg
.get('watchdog', 'interval', default
=30)), self
.watchdog
)
63 def log(self
, *args
, **kwargs
):
64 self
.parent
.log(self
.nick
, *args
, **kwargs
)
67 if self
.conn
.connect():
68 self
.parent
.newfd(self
, self
.conn
.socket
.fileno())
71 self
.lastreceived
= time
.time()
72 return self
.conn
.read()
74 def _checknick(self
): # check if we're using the right nick, try changing
75 if self
.nick
!= self
.permnick
and self
.conn
.registered():
76 self
.conn
.send("NICK %s" % (self
.permnick
))
78 def parse(self
, line
):
79 if self
.parent
.cfg
.getboolean('debug', 'io'):
84 zero
= { #things to look for without source
85 'NOTICE': self
._gotconnected
,
86 'PING': self
._gotping
,
87 'ERROR': self
._goterror
,
89 one
= { #things to look for after source
91 '376': self
._gotRegistered
,
92 '422': self
._gotRegistered
,
93 'PRIVMSG': self
._gotprivmsg
,
94 '353': self
._got
353, #NAMES
95 '354': self
._got
354, #WHO
96 '433': self
._got
433, #nick in use
97 'JOIN': self
._gotjoin
,
98 'PART': self
._gotpart
,
99 'KICK': self
._gotkick
,
100 'QUIT': self
._gotquit
,
101 'NICK': self
._gotnick
,
102 'MODE': self
._gotmode
,
105 if self
.parent
.hasnumhook(pieces
[1]):
106 hooks
= self
.parent
.getnumhook(pieces
[1])
107 for callback
in hooks
:
111 self
.__debug
_cbexception
("numhook", line
)
113 if pieces
[0] in zero
:
114 zero
[pieces
[0]](pieces
)
115 elif pieces
[1] in one
:
116 one
[pieces
[1]](pieces
)
118 def _gotconnected(self
, pieces
):
119 if not self
.conn
.registered():
121 def _gotping(self
, pieces
):
122 self
.conn
.send("PONG %s" % (pieces
[1]))
124 def _goterror(self
, pieces
):
126 self
.quit("Error detected: %s" % ' '.join(pieces
))
127 curs
= self
.parent
.query("UPDATE bots SET connected = 0")
132 def _got001(self
, pieces
):
133 pass # wait until the end of MOTD instead
134 def _gotRegistered(self
, pieces
):
135 self
.conn
.registered(True)
137 curs
= self
.parent
.query("UPDATE bots SET connected = 1 WHERE nick = %s", (self
.nick
,))
138 if curs
: curs
.close()
140 self
.conn
.send("MODE %s +x" % (pieces
[2]))
141 if self
.authname
is not None and self
.authpass
is not None:
142 self
.conn
.send("AUTH %s %s" % (self
.authname
, self
.authpass
))
145 def _gotprivmsg(self
, pieces
):
146 nick
= pieces
[0].split('!')[0][1:]
147 user
= self
.parent
.user(nick
)
149 msg
= ' '.join(pieces
[3:])[1:]
150 self
.parsemsg(user
, target
, msg
)
151 def _got353(self
, pieces
):
152 prefixes
= {'@': 'op', '+': 'voice'}
153 chan
= self
.parent
.channel(pieces
[4])
155 names
[0] = names
[0][1:] #remove colon
158 user
= self
.parent
.user(n
[1:])
159 chan
.userjoin(user
, prefixes
[n
[0]])
161 user
= self
.parent
.user(n
)
164 def _got354(self
, pieces
):
167 nick
, auth
= pieces
[4:6]
170 chan
, nick
, auth
= pieces
[4:7]
171 chan
= self
.parent
.channel(chan
)
172 user
= self
.parent
.user(nick
)
179 if qt
== 2: # triggered by !auth
182 self
.msg(nick
, "You are now known as #%s (access level: %s)" % (auth
, user
.glevel
))
184 self
.msg(nick
, "You are now known as #%s (not staff)" % (auth
))
186 self
.msg(nick
, "I tried, but you're not authed!")
187 def _got433(self
, pieces
):
188 if not self
.conn
.registered(): #we're trying to connect
189 newnick
= "%s%d" % (self
.nick
, random
.randint(111, 999))
190 self
.conn
.send("NICK %s" % (newnick
))
192 def _gotjoin(self
, pieces
):
193 nick
= pieces
[0].split('!')[0][1:]
194 chan
= self
.parent
.channel(pieces
[2])
196 if nick
== self
.nick
:
197 self
.conn
.send("WHO %s c%%cant,3" % (chan
))
199 user
= self
.parent
.user(nick
, justjoined
=True)
202 def _clientLeft(self
, nick
, chan
):
203 if nick
!= self
.nick
:
204 gone
= self
.parent
.user(nick
).part(chan
)
205 chan
.userpart(self
.parent
.user(nick
))
207 self
.parent
.user(nick
).quit()
208 del self
.parent
.users
[nick
.lower()]
209 def _gotpart(self
, pieces
):
210 nick
= pieces
[0].split('!')[0][1:]
211 chan
= self
.parent
.channel(pieces
[2])
212 self
._clientLeft
(nick
, chan
)
213 def _gotkick(self
, pieces
):
215 chan
= self
.parent
.channel(pieces
[2])
216 self
._clientLeft
(nick
, chan
)
217 def _gotquit(self
, pieces
):
218 nick
= pieces
[0].split('!')[0][1:]
219 if nick
!= self
.nick
:
220 for chan
in self
.parent
.user(nick
).chans
:
221 chan
.userpart(self
.parent
.user(nick
))
222 self
.parent
.user(nick
).quit()
223 del self
.parent
.users
[nick
.lower()]
224 def _gotnick(self
, pieces
):
225 oldnick
= pieces
[0].split('!')[0][1:]
226 newnick
= pieces
[2][1:]
227 if newnick
.lower() != oldnick
.lower():
228 self
.parent
.users
[newnick
.lower()] = self
.parent
.users
[oldnick
.lower()]
229 del self
.parent
.users
[oldnick
.lower()]
230 self
.parent
.users
[newnick
.lower()].nickchange(newnick
)
231 def _gotmode(self
, pieces
):
232 source
= pieces
[0].split('!')[0][1:]
234 if not chan
.startswith("#"): return
235 chan
= self
.parent
.channel(pieces
[2])
247 chan
.userop(self
.parent
.user(args
.pop(0)))
249 chan
.userdeop(self
.parent
.user(args
.pop(0)))
252 chan
.uservoice(self
.parent
.user(args
.pop(0)))
254 chan
.userdevoice(self
.parent
.user(args
.pop(0)))
256 pass # don't care about other modes
258 def __debug_cbexception(self
, source
, *args
, **kwargs
):
259 if self
.parent
.cfg
.getboolean('debug', 'cbexc'):
260 self
.conn
.send("PRIVMSG %s :%09.3f \ 34\1f!!! CBEXC\1f\ 3 %s" % (self
.parent
.cfg
.get('debug', 'owner'), time
.time() % 100000, source
))
261 __import__('traceback').print_exc()
262 self
.log('!', "CBEXC %s %r %r" % (source
, args
, kwargs
))
263 # print "%09.3f %s [!] CBEXC %s %r %r" % (time.time() % 100000, self.nick, source, args, kwargs)
266 def parsemsg(self
, user
, target
, msg
):
267 if user
.glevel
<= -2: return # short circuit if user is IGNORED
269 chanparam
= None # was the channel specified as part of the command?
273 if target
== self
.nick
:
274 if msg
.startswith("\001"): #ctcp
275 msg
= msg
.strip("\001")
277 self
.msg(user
, "\001VERSION Erebus v%d.%d - http://github.com/zonidjan/erebus" % (self
.parent
.APIVERSION
, self
.parent
.RELEASE
))
280 triggerused
= msg
.startswith(self
.parent
.trigger
)
281 if triggerused
: msg
= msg
[len(self
.parent
.trigger
):]
284 if target
!= self
.nick
: # message was sent to a channel
286 if msg
.startswith('*'): # message may be addressed to bot by "*BOTNICK" trigger?
287 if pieces
[0][1:].lower() == self
.nick
.lower():
288 pieces
.pop(0) # command actually starts with next word
289 msg
= ' '.join(pieces
) # command actually starts with next word
292 return # "message" is empty
299 if chanword
.startswith('#'):
300 chanparam
= self
.parent
.channel(chanword
)
302 if target
!= self
.nick
: # message was sent to a channel
303 chan
= self
.parent
.channel(target
)
305 if self
.parent
.haschanhook(target
.lower()):
306 for callback
in self
.parent
.getchanhook(target
.lower()):
308 cbret
= callback(self
, user
, chan
, *pieces
)
309 except NotImplementedError:
310 self
.msg(user
, "Command not implemented.")
312 self
.msg(user
, "Command failed. Code: CBEXC%09.3f" % (time
.time() % 100000))
313 self
.__debug
_cbexception
("chanhook", user
=user
, target
=target
, msg
=msg
)
314 return # not to bot, don't process!
316 cmd
= pieces
[0].lower()
318 if self
.parent
.hashook(cmd
):
319 for callback
in self
.parent
.gethook(cmd
):
320 if chanparam
is not None and (callback
.needchan
or callback
.wantchan
):
323 if chan
is None and callback
.needchan
:
325 self
.msg(user
, "You need to specify a channel for that command.")
326 elif user
.glevel
>= callback
.reqglevel
and (not callback
.needchan
or chan
.levelof(user
.auth
) >= callback
.reqclevel
):
329 cbret
= callback(self
, user
, chan
, target
, *pieces
[1:])
330 if cbret
is NotImplemented:
331 raise NotImplementedError
332 except NotImplementedError:
333 self
.msg(user
, "Command not implemented.")
335 self
.msg(user
, "Command failed. Code: CBEXC%09.3f" % (time
.time() % 100000))
336 self
.__debug
_cbexception
("hook", user
=user
, target
=target
, msg
=msg
)
337 except SystemExit as e
:
339 curs
= self
.parent
.query("UPDATE bots SET connected = 0")
345 self
.msg(user
, "I don't know that command.")
347 self
.msg(user
, "You don't have enough access to run that command.")
349 def __debug_nomsg(self
, target
, msg
):
350 if self
.parent
.cfg
.getboolean('debug', 'nomsg'):
351 self
.conn
.send("PRIVMSG %s :%09.3f \ 34\1f!!! NOMSG\1f\ 3 %r, %r" % (self
.parent
.cfg
.get('debug', 'owner'), time
.time() % 100000, target
, msg
))
352 self
.log('!', "!!! NOMSG")
353 # print "%09.3f %s [!] %s" % (time.time() % 100000, self.nick, "!!! NOMSG")
354 __import__('traceback').print_stack()
356 def msg(self
, target
, msg
):
357 if self
.parent
.cfg
.getboolean('erebus', 'nofakelag'): return self
.fastmsg(target
, msg
)
358 cmd
= self
._formatmsg
(target
, msg
)
359 if len(cmd
) > MAXLEN
: return False
360 if self
.conn
.exceeded
or self
.conn
.bytessent
+len(cmd
) >= self
.conn
.recvq
:
361 self
.msgqueue
.append(cmd
)
364 self
.conn
.exceeded
= True
367 def slowmsg(self
, target
, msg
):
368 if self
.parent
.cfg
.getboolean('erebus', 'nofakelag'): return self
.fastmsg(target
, msg
)
369 cmd
= self
._formatmsg
(target
, msg
)
370 if len(cmd
) > MAXLEN
: return False
371 if self
.conn
.exceeded
or self
.conn
.bytessent
+len(cmd
) >= self
.conn
.recvq
:
372 self
.slowmsgqueue
.append(cmd
)
375 self
.conn
.exceeded
= True
378 def fastmsg(self
, target
, msg
):
379 cmd
= self
._formatmsg
(target
, msg
)
380 if len(cmd
) > MAXLEN
: return False
382 self
.conn
.exceeded
= True
385 def _formatmsg(self
, target
, msg
):
386 if target
is None or msg
is None:
387 return self
.__debug
_nomsg
(target
, msg
)
391 if target
.startswith('#'): command
= "PRIVMSG %s :%s" % (target
, msg
)
392 else: command
= "NOTICE %s :%s" % (target
, msg
)
398 self
.conn
.bytessent
-= self
.conn
.recvq
/3
399 if self
.conn
.bytessent
< 0: self
.conn
.bytessent
= 0
400 self
.conn
.exceeded
= False
403 cmd
= self
.msgqueue
.popleft()
404 if not self
.conn
.exceeded
and self
.conn
.bytessent
+len(cmd
) < self
.conn
.recvq
:
406 self
.conn
.exceeded
= True
407 else: raise IndexError
410 cmd
= self
.slowmsgqueue
.popleft()
411 if not self
.conn
.exceeded
and self
.conn
.bytessent
+len(cmd
) < self
.conn
.recvq
:
413 self
.conn
.exceeded
= True
416 self
.msgtimer
.start()
418 def makemsgtimer(self
):
419 self
.msgtimer
= threading
.Timer(3, self
._popmsg
)
420 self
.msgtimer
.daemon
= True
422 def join(self
, chan
):
423 self
.conn
.send("JOIN %s" % (chan
))
425 def part(self
, chan
):
426 self
.conn
.send("PART %s" % (chan
))
428 def quit(self
, reason
="Shutdown"):
429 self
.conn
.send("QUIT :%s" % (reason
))
431 def __str__(self
): return self
.nick
432 def __repr__(self
): return "<Bot %r>" % (self
.nick
)
434 class BotConnection(object):
435 def __init__(self
, parent
, bind
, server
, port
):
437 self
.buffer = bytearray(8192)
442 self
.port
= int(port
)
444 self
.state
= 0 # 0=disconnected, 1=registering, 2=connected
448 self
.exceeded
= False
451 self
.socket
= socket
.socket(socket
.AF_INET
, socket
.SOCK_STREAM
)
452 self
.socket
.bind((self
.bind
, 0))
453 self
.socket
.connect((self
.server
, self
.port
))
457 self
.send("NICK %s" % (self
.parent
.nick
))
458 self
.send("USER %s 0 * :%s" % (self
.parent
.user
, self
.parent
.realname
))
462 def registered(self
, done
=False):
463 if done
: self
.state
= 2
464 return self
.state
== 2
466 def send(self
, line
):
467 if self
.parent
.parent
.cfg
.getboolean('debug', 'io'):
468 self
.parent
.log('O', line
)
469 self
.bytessent
+= len(line
)
472 def _write(self
, line
):
473 self
.socket
.sendall(line
.encode('utf-8', 'backslashreplace')+b
"\r\n")
476 self
.buffer += self
.socket
.recv(8192)
479 while b
"\r\n" in self
.buffer:
480 pieces
= self
.buffer.split(b
"\r\n", 1)
481 lines
.append(pieces
[0].decode('utf-8', 'backslashreplace'))
482 self
.buffer = pieces
[1]
486 def __str__(self
): return self
.parent
.nick
487 def __repr__(self
): return "<BotConnection %r (%r)>" % (self
.socket
.fileno(), self
.parent
.nick
)