]>
jfr.im git - erebus.git/blob - bot.py
3 # Erebus IRC bot - Author: John Runyon
4 # "Bot" and "BotConnection" classes (handling a specific "arm")
10 #bots = {'erebus': bot.Bot(nick='Erebus', user='erebus', bind='', server='irc.quakenet.org', port=6667, realname='Erebus')}
12 def __init__(self
, parent
, nick
, user
, bind
, server
, port
, realname
):
16 self
.realname
= realname
18 curs
= self
.parent
.db
.cursor()
19 if curs
.execute("SELECT chname FROM chans WHERE bot = %s AND active = 1", (self
.nick
,)):
20 chansres
= curs
.fetchall()
22 self
.chans
= [self
.parent
.newchannel(self
, row
['chname']) for row
in chansres
]
24 self
.conn
= BotConnection(self
, bind
, server
, port
)
26 if self
.conn
.connect():
27 self
.parent
.newfd(self
, self
.conn
.socket
.fileno())
30 return self
.conn
.read()
32 def parse(self
, line
):
35 if not self
.conn
.registered() and pieces
[0] == "NOTICE":
39 if self
.parent
.hasnumhook(pieces
[1]):
40 hooks
= self
.parent
.getnumhook(pieces
[1])
41 for callback
in hooks
:
44 if pieces
[1] == "001":
45 self
.conn
.registered(True)
49 elif pieces
[1] == "PRIVMSG":
50 nick
= pieces
[0].split('!')[0][1:]
51 user
= self
.parent
.user(nick
)
53 msg
= ' '.join(pieces
[3:])[1:]
54 self
.parsemsg(user
, target
, msg
)
56 elif pieces
[0] == "PING":
57 self
.conn
.send("PONG %s" % (pieces
[1]))
59 elif pieces
[1] == "354": # WHOX
63 self
.parent
.user(nick
).authed(auth
)
65 elif pieces
[1] == "JOIN":
66 nick
= pieces
[0].split('!')[0][1:]
67 chan
= self
.parent
.channel(pieces
[2])
70 self
.conn
.send("WHO %s %%ant,1" % (chan
))
72 user
= self
.parent
.user(nick
, justjoined
=True)
76 elif pieces
[1] == "PART":
77 nick
= pieces
[0].split('!')[0][1:]
78 chan
= self
.parent
.channel(pieces
[2])
81 self
.parent
.user(nick
).part(chan
)
82 chan
.userpart(self
.parent
.user(nick
))
84 elif pieces
[1] == "QUIT":
85 nick
= pieces
[0].split('!')[0][1:]
87 self
.parent
.user(nick
).quit()
88 del self
.parent
.users
[nick
.lower()]
90 elif pieces
[1] == "MODE": #TODO
94 def parsemsg(self
, user
, target
, msg
):
99 triggerused
= msg
[0] == self
.parent
.trigger
100 if triggerused
: msg
= msg
[1:]
103 if target
== self
.nick
:
106 if chanword
[0] == '#':
107 chan
= self
.parent
.channel(chanword
)
108 if chan
is not None: #if chan is still none, there's no bot on "chanword", and chanword is used as a parameter.
111 else: # message was sent to a channel
112 chan
= self
.parent
.channel(target
) #TODO check if bot's on channel --- in Erebus.channel() maybe?
114 if msg
[0] == '*': # message may be addressed to bot by "*BOTNICK" trigger?
115 if pieces
[0][1:].lower() == self
.nick
.lower():
116 pieces
.pop(0) # command actually starts with next word
117 msg
= ' '.join(pieces
) # command actually starts with next word
118 elif not triggerused
:
119 if self
.parent
.haschanhook(target
.lower()):
120 for callback
in self
.parent
.getchanhook(target
.lower()):
121 cbret
= callback(self
, user
, chan
, *pieces
)
122 if cbret
is NotImplemented:
123 self
.msg(user
, "Command not implemented.")
126 return # not to bot, don't process!
128 return # Fix if you feel like it /BiohZn
130 cmd
= pieces
[0].lower()
132 if self
.parent
.hashook(cmd
):
133 for callback
in self
.parent
.gethook(cmd
):
134 if chan
is None and callback
.needchan
:
135 self
.msg(user
, "You need to specify a channel for that command.")
136 elif user
.glevel
>= callback
.reqglevel
and (not callback
.needchan
or chan
.levelof(user
.auth
) >= callback
.reqclevel
):
137 cbret
= callback(self
, user
, chan
, target
, *pieces
[1:])
138 if cbret
is NotImplemented:
139 self
.msg(user
, "Command not implemented.")
141 def msg(self
, target
, msg
):
142 if target
is None or msg
is None: return
144 if isinstance(target
, self
.parent
.User
): self
.conn
.send("NOTICE %s :%s" % (target
.nick
, msg
))
145 elif isinstance(target
, self
.parent
.Channel
): self
.conn
.send("PRIVMSG %s :%s" % (target
.name
, msg
))
146 elif isinstance(target
, basestring
):
147 if target
[0] == '#': self
.conn
.send("PRIVMSG %s :%s" % (target
, msg
))
148 else: self
.conn
.send("NOTICE %s :%s" % (target
, msg
))
149 else: raise TypeError('Bot.msg() "target" must be Erebus.User, Erebus.Channel, or string')
151 def join(self
, chan
):
152 self
.conn
.send("JOIN %s" % (chan
))
154 def part(self
, chan
):
155 self
.conn
.send("PART %s" % (chan
))
157 def quit(self
, reason
="Shutdown"):
158 self
.conn
.send("QUIT :%s" % (reason
))
160 def __str__(self
): return self
.nick
161 def __repr__(self
): return "<Bot %r>" % (self
.nick
)
163 class BotConnection(object):
164 state
= 0 # 0=disconnected, 1=registering, 2=connected
166 def __init__(self
, parent
, bind
, server
, port
):
173 self
.port
= int(port
)
176 self
.socket
= socket
.socket(socket
.AF_INET
, socket
.SOCK_STREAM
)
177 self
.socket
.bind((self
.bind
, 0))
178 self
.socket
.connect((self
.server
, self
.port
))
182 self
.send("NICK %s" % (self
.parent
.nick
))
183 self
.send("USER %s 0 * :%s" % (self
.parent
.user
, self
.parent
.realname
))
187 def registered(self
, done
=False):
188 if done
: self
.state
= 2
189 return self
.state
== 2
191 #TODO: rewrite send() to queue
192 def send(self
, line
):
193 print self
.parent
.nick
, '[O]', str(line
)
196 def write(self
, line
):
197 self
.socket
.sendall(line
+"\r\n")
200 self
.buffer += self
.socket
.recv(8192)
203 while '\r\n' in self
.buffer:
204 pieces
= self
.buffer.split('\r\n', 1)
205 print self
.parent
.nick
, '[I]', pieces
[0]
206 lines
.append(pieces
[0])
207 self
.buffer = pieces
[1]
211 def __str__(self
): return self
.nick
212 def __repr__(self
): return "<BotConnection %r (%r)>" % (self
.socket
.fileno(), self
.parent
.nick
)