import socket import time import sys from utils import * from copy import copy import traceback from classes import * def _sendFromServer(irc, msg): irc.send(':%s %s' % (irc.sid, msg)) def _sendFromUser(irc, msg, user=None): if user is None: user = irc.pseudoclient.uid irc.send(':%s %s' % (user, msg)) def _nicktoUid(irc, nick): for k, v in irc.users.items(): if v.nick == nick: return k def spawnClient(irc, nick, ident, host, *args): uid = next_uid(irc.sid) ts = int(time.time()) _sendFromServer(irc, "UID {uid} {ts} {nick} {host} {host} {ident} 0.0.0.0 {ts} +o +" " :PyLink Client".format(ts=ts, host=host, nick=nick, ident=ident, uid=uid)) u = irc.users[uid] = IrcUser(nick, ts, uid, ident, host, *args) irc.servers[irc.sid].users.append(uid) return u def joinClient(irc, client, channel): # Channel list can be a comma-separated list of channels, per the # IRC specification. if not isInternalClient(irc, client): raise LookupError('No user/PyLink PseudoClient %r exists.' % client) _sendFromUser(irc, "JOIN {channel} {ts} +nt :,{uid}".format(sid=irc.sid, ts=int(time.time()), uid=client.uid, channel=channel)) def removeClient(irc, numeric): """ Removes a client from our internal databases, regardless of whether it's one of our pseudoclients or not.""" for k, v in copy(irc.channels).items(): irc.channels[k].users.discard(numeric) if not irc.channels[k].users: # Clear empty channels del irc.channels[k] sid = numeric[:3] print('Removing client %s from irc.users' % numeric) del irc.users[numeric] print('Removing client %s from irc.servers[%s]' % (numeric, sid)) irc.servers[sid].users.remove(numeric) def isInternalClient(irc, numeric): """ Returns whether is a PyLink PseudoClient. """ return numeric in irc.servers[irc.sid].users def quitClient(irc, numeric): """ Quits a PyLink PseudoClient.""" if isInternalClient(irc, numeric): _sendFromUser(irc, numeric, "QUIT :Client quit") else: raise LookupError("No user %r exists. If you're trying to remove " "a user that's not a PyLink PseudoClient from " "the internal state, use removeClient() instead.") def kickClient(irc, channel, numeric, target, reason=None): """ Sends a kick from a PyLink PseudoClient.""" if not isInternalClient(irc, numeric): raise LookupError('No user/PyLink PseudoClient %r exists.' % numeric) if reason is None: reason = irc.users[target].nick _sendFromUser(irc, numeric, 'KICK %s %s :%s' % (channel, target, reason)) def nickClient(irc, numeric, newnick, reason=None): """ Changes the nick of a PyLink PseudoClient.""" if not isInternalClient(irc, numeric): raise LookupError('No user/PyLink PseudoClient %r exists.' % numeric) _sendFromUser(irc, numeric, 'NICK %s' % newnick) def connect(irc): irc.start_ts = ts = int(time.time()) host = irc.serverdata["hostname"] irc.servers[irc.sid] = IrcServer(None) f = irc.send f('CAPAB START 1203') # This is hard coded atm... We should fix it eventually... f('CAPAB CAPABILITIES :NICKMAX=32 HALFOP=0 CHANMAX=65 MAXMODES=20' ' IDENTMAX=12 MAXQUIT=255 PROTOCOL=1203') f('CAPAB END') # TODO: check recvpass here f('SERVER {host} {Pass} 0 {sid} :PyLink Service'.format(host=host, Pass=irc.serverdata["sendpass"], sid=irc.sid)) f(':%s BURST %s' % (irc.sid, ts)) # InspIRCd documentation: # :751 UID 751AAAAAA 1220196319 Brain brainwave.brainbox.cc # netadmin.chatspike.net brain 192.168.1.10 1220196324 +Siosw # +ACKNOQcdfgklnoqtx :Craig Edwards irc.pseudoclient = spawnClient(irc, 'PyLink', 'pylink', host) f(':%s ENDBURST' % (irc.sid)) joinClient(irc, irc.pseudoclient, ','.join(irc.serverdata['channels'])) # :7NU PING 7NU 0AL def handle_ping(irc, servernumeric, command, args): if args[1] == irc.sid: _sendFromServer(irc, 'PONG %s' % args[1]) def handle_privmsg(irc, source, command, args): prefix = irc.conf['bot']['prefix'] if args[0] == irc.pseudoclient.uid: cmd_args = args[1].split(' ') cmd = cmd_args[0].lower() try: cmd_args = cmd_args[1:] except IndexError: cmd_args = [] try: func = bot_commands[cmd] except KeyError: msg(irc, source, 'Unknown command %r.' % cmd) return try: func(irc, source, cmd_args) except Exception as e: traceback.print_exc() msg(irc, source, 'Uncaught exception in command %r: %s: %s' % (cmd, type(e).__name__, str(e))) return def handle_kill(irc, source, command, args): killed = args[0] removeClient(irc, killed) if killed == irc.pseudoclient.uid: irc.pseudoclient = spawnClient(irc, 'PyLink', 'pylink', irc.serverdata["hostname"]) joinClient(irc, irc.pseudoclient, ','.join(irc.serverdata['channels'])) def handle_kick(irc, source, command, args): # :70MAAAAAA KICK #endlessvoid 70MAAAAAA :some reason channel = args[0] kicked = args[1] irc.channels[channel].users.discard(kicked) if kicked == irc.pseudoclient.uid: joinClient(irc, irc.pseudoclient, channel) def handle_part(irc, source, command, args): channel = args[0] # We should only get PART commands for channels that exist, right?? irc.channels[channel].users.discard(source) def handle_error(irc, numeric, command, args): print('Received an ERROR, killing!') irc.connected = False sys.exit(1) def handle_fjoin(irc, servernumeric, command, args): # :70M FJOIN #chat 1423790411 +AFPfjnt 6:5 7:5 9:5 :o,1SRAABIT4 v,1IOAAF53R <...> channel = args[0] if channel not in irc.channels.keys(): irc.channels[channel] = IrcChannel() # InspIRCd sends each user's channel data in the form of 'modeprefix(es),UID' userlist = args[-1].split() for user in userlist: modeprefix, user = user.split(',', 1) for mode in modeprefix: # Note that a user can have more than one mode prefix (e.g. they have both +o and +v), # so they would be added to both lists. ''' # left to right: m_ojoin, m_operprefix, owner (~/+q), admin (&/+a), and op (!/+o) if mode in 'Yyqao': irc.channels[channel].ops.append(user) if mode == 'h': irc.channels[channel].halfops.append(user) if mode == 'v': irc.channels[channel].voices.append(user) ''' irc.channels[channel].users.add(user) def handle_uid(irc, numeric, command, args): # :70M UID 70MAAAAAB 1429934638 GL 0::1 hidden-7j810p.9mdf.lrek.0000.0000.IP gl 0::1 1429934638 +Wioswx +ACGKNOQXacfgklnoqvx :realname uid, ts, nick, realhost, host, ident, ip = args[0:7] realname = args[-1] irc.users[uid] = IrcUser(nick, ts, uid, ident, host, realname, realhost, ip) irc.servers[numeric].users.append(uid) def handle_quit(irc, numeric, command, args): # :1SRAAGB4T QUIT :Quit: quit message goes here removeClient(irc, numeric) def handle_burst(irc, numeric, command, args): # :70M BURST 1433044587 irc.servers[numeric] = IrcServer(None) def handle_server(irc, numeric, command, args): # :70M SERVER millennium.overdrive.pw * 1 1ML :a relatively long period of time... (Fremont, California) servername = args[0] sid = args[3] irc.servers[sid] = IrcServer(numeric) def handle_nick(irc, numeric, command, args): newnick = args[0] irc.users[numeric].nick = newnick def handle_fmode(irc, numeric, command, args): # <- :70MAAAAAA FMODE #chat 1433653462 +hhT 70MAAAAAA 70MAAAAAD # Oh god, how are we going to handle this?! channel = args[0] modestrings = args[3:] def handle_squit(irc, numeric, command, args): # :70M SQUIT 1ML :Server quit by GL!gl@0::1 split_server = args[0] print('Netsplit on server %s' % split_server) # Prevent RuntimeError: dictionary changed size during iteration old_servers = copy(irc.servers) for sid, data in old_servers.items(): if data.uplink == split_server: print('Server %s also hosts server %s, removing those users too...' % (split_server, sid)) handle_squit(irc, sid, 'SQUIT', [sid, "PyLink: Automatically splitting leaf servers of %s" % sid]) for user in copy(irc.servers[split_server].users): print('Removing client %s (%s)' % (user, irc.users[user].nick)) removeClient(irc, user) del irc.servers[split_server] def handle_idle(irc, numeric, command, args): """Handle the IDLE command, sent between servers in remote WHOIS queries.""" # <- :70MAAAAAA IDLE 1MLAAAAIG # -> :1MLAAAAIG IDLE 70MAAAAAA 1433036797 319 sourceuser = numeric targetuser = args[0] _sendFromUser(irc, 'IDLE %s %s 0' % (sourceuser, irc.start_ts), user=targetuser) def handle_events(irc, data): # Each server message looks something like this: # :70M FJOIN #chat 1423790411 +AFPfjnt 6:5 7:5 9:5 :v,1SRAAESWE # : ... :final multi word argument args = data.split() if args and args[0] == 'SERVER': # SERVER whatever.net abcdefgh 0 10X :something servername = args[1] if args[2] != irc.serverdata['recvpass']: # Check if recvpass is correct print('Error: recvpass from uplink server %s does not match configuration!' % servername) sys.exit(1) return try: real_args = [] for arg in args: real_args.append(arg) # If the argument starts with ':' and ISN'T the first argument. # The first argument is used for denoting the source UID/SID. if arg.startswith(':') and args.index(arg) != 0: # : is used for multi-word arguments that last until the end # of the message. We can use list splicing here to turn them all # into one argument. index = args.index(arg) # Get the array index of the multi-word arg # Set the last arg to a joined version of the remaining args arg = args[index:] arg = ' '.join(arg)[1:] # Cut the original argument list right before the multi-word arg, # and then append the multi-word arg. real_args = args[:index] real_args.append(arg) break real_args[0] = real_args[0].split(':', 1)[1] args = real_args numeric = args[0] command = args[1] args = args[2:] except IndexError: return # We will do wildcard event handling here. Unhandled events are just ignored, yay! try: func = globals()['handle_'+command.lower()] func(irc, numeric, command, args) except KeyError: # unhandled event pass