3
0
mirror of https://github.com/jlu5/PyLink.git synced 2025-01-23 10:44:09 +01:00

Add pseudoserver spawning, adapting _sendFromServer and spawnClient accordingly.

Now you can spawn multiple servers for a multi-server botnet!

Also, create proto.isInternalServer() / utils.isServerName() checkers.

Closes #22.
This commit is contained in:
James Lu 2015-06-21 15:00:33 -07:00
parent 0a1754dd4b
commit c686523a6e
3 changed files with 88 additions and 20 deletions

View File

@ -18,9 +18,18 @@ class IrcUser():
return repr(self.__dict__) return repr(self.__dict__)
class IrcServer(): class IrcServer():
def __init__(self, uplink): """PyLink IRC Server class.
uplink: The SID of this IrcServer instance's uplink. This is set to None
for the main PyLink PseudoServer!
name: The name of the server.
internal: Whether the server is an internal PyLink PseudoServer.
"""
def __init__(self, uplink, name, internal=False):
self.uplink = uplink self.uplink = uplink
self.users = [] self.users = []
self.internal = internal
self.name = name.lower()
def __repr__(self): def __repr__(self):
return repr(self.__dict__) return repr(self.__dict__)

View File

@ -8,16 +8,23 @@ from copy import copy
import traceback import traceback
from classes import * from classes import *
uidgen = utils.TS6UIDGenerator() uidgen = {}
def _sendFromServer(irc, msg): def _sendFromServer(irc, sid, msg):
irc.send(':%s %s' % (irc.sid, msg)) irc.send(':%s %s' % (sid, msg))
def _sendFromUser(irc, numeric, msg): def _sendFromUser(irc, numeric, msg):
irc.send(':%s %s' % (numeric, msg)) irc.send(':%s %s' % (numeric, msg))
def spawnClient(irc, nick, ident, host, modes=[], *args): def spawnClient(irc, nick, ident, host, modes=[], server=None, *args):
uid = uidgen.next_uid(irc.sid) server = server or irc.sid
if not isInternalServer(irc, server):
raise ValueError('Server %r is not a PyLink internal PseudoServer!' % server)
# We need a separate UID generator instance for every PseudoServer
# we spawn. Otherwise, things won't wrap around properly.
if server not in uidgen:
uidgen[server] = utils.TS6UIDGenerator()
uid = uidgen[server].next_uid(server)
ts = int(time.time()) ts = int(time.time())
if modes: if modes:
modes = utils.joinModes(modes) modes = utils.joinModes(modes)
@ -25,21 +32,22 @@ def spawnClient(irc, nick, ident, host, modes=[], *args):
modes = '+' modes = '+'
if not utils.isNick(nick): if not utils.isNick(nick):
raise ValueError('Invalid nickname %r.' % nick) raise ValueError('Invalid nickname %r.' % nick)
_sendFromServer(irc, "UID {uid} {ts} {nick} {host} {host} {ident} 0.0.0.0 " _sendFromServer(irc, server, "UID {uid} {ts} {nick} {host} {host} {ident} 0.0.0.0 "
"{ts} {modes} + :PyLink Client".format(ts=ts, host=host, "{ts} {modes} + :PyLink Client".format(ts=ts, host=host,
nick=nick, ident=ident, uid=uid, nick=nick, ident=ident, uid=uid,
modes=modes)) modes=modes))
u = irc.users[uid] = IrcUser(nick, ts, uid, ident, host, *args) u = irc.users[uid] = IrcUser(nick, ts, uid, ident, host, *args)
irc.servers[irc.sid].users.append(uid) irc.servers[server].users.append(uid)
return u return u
def joinClient(irc, client, channel): def joinClient(irc, client, channel):
# One channel per line here! server = isInternalClient(irc, client)
if not isInternalClient(irc, client): if not server:
raise LookupError('No such PyLink PseudoClient exists.') raise LookupError('No such PyLink PseudoClient exists.')
if not utils.isChannel(channel): if not utils.isChannel(channel):
raise ValueError('Invalid channel name %r.' % channel) raise ValueError('Invalid channel name %r.' % channel)
_sendFromServer(irc, "FJOIN {channel} {ts} + :,{uid}".format( # One channel per line here!
_sendFromServer(irc, server, "FJOIN {channel} {ts} + :,{uid}".format(
ts=int(time.time()), uid=client, channel=channel)) ts=int(time.time()), uid=client, channel=channel))
def partClient(irc, client, channel, reason=None): def partClient(irc, client, channel, reason=None):
@ -71,9 +79,19 @@ def removeClient(irc, numeric):
def isInternalClient(irc, numeric): def isInternalClient(irc, numeric):
"""<irc object> <client numeric> """<irc object> <client numeric>
Returns whether <client numeric> is a PyLink PseudoClient. Checks whether <client numeric> is a PyLink PseudoClient,
returning the SID of the PseudoClient's server if True.
""" """
return numeric in irc.servers[irc.sid].users for sid in irc.servers:
if irc.servers[sid].internal and numeric in irc.servers[sid].users:
return sid
def isInternalServer(irc, sid):
"""<irc object> <sid>
Returns whether <sid> is an internal PyLink PseudoServer.
"""
return (sid in irc.servers and irc.servers[sid].internal)
def quitClient(irc, numeric, reason): def quitClient(irc, numeric, reason):
"""<irc object> <client numeric> """<irc object> <client numeric>
@ -111,7 +129,7 @@ def nickClient(irc, numeric, newnick):
def connect(irc): def connect(irc):
irc.start_ts = ts = int(time.time()) irc.start_ts = ts = int(time.time())
host = irc.serverdata["hostname"] host = irc.serverdata["hostname"]
irc.servers[irc.sid] = IrcServer(None) irc.servers[irc.sid] = IrcServer(None, host, internal=True)
f = irc.send f = irc.send
f('CAPAB START 1203') f('CAPAB START 1203')
@ -218,18 +236,25 @@ def handle_uid(irc, numeric, command, args):
irc.servers[numeric].users.append(uid) irc.servers[numeric].users.append(uid)
def handle_quit(irc, numeric, command, args): def handle_quit(irc, numeric, command, args):
# :1SRAAGB4T QUIT :Quit: quit message goes here # <- :1SRAAGB4T QUIT :Quit: quit message goes here
removeClient(irc, numeric) removeClient(irc, numeric)
def handle_burst(irc, numeric, command, args): def handle_burst(irc, numeric, command, args):
# :70M BURST 1433044587 # BURST is sent by our uplink when we link.
irc.servers[numeric] = IrcServer(None) # <- :70M BURST 1433044587
# This is handled in handle_events, since our uplink
# only sends its name in the initial authentication phase,
# not in any following BURST commands.
pass
def handle_server(irc, numeric, command, args): def handle_server(irc, numeric, command, args):
# :70M SERVER millennium.overdrive.pw * 1 1ML :a relatively long period of time... (Fremont, California) # SERVER is sent by our uplink or any other server to introduce others.
# <- :00A SERVER test.server * 1 00C :testing raw message syntax
# <- :70M SERVER millennium.overdrive.pw * 1 1ML :a relatively long period of time... (Fremont, California)
servername = args[0] servername = args[0]
sid = args[3] sid = args[3]
irc.servers[sid] = IrcServer(numeric) irc.servers[sid] = IrcServer(numeric, servername)
def handle_nick(irc, numeric, command, args): def handle_nick(irc, numeric, command, args):
# <- :70MAAAAAA NICK GL-devel 1434744242 # <- :70MAAAAAA NICK GL-devel 1434744242
@ -294,10 +319,12 @@ def handle_events(irc, data):
if args and args[0] == 'SERVER': if args and args[0] == 'SERVER':
# SERVER whatever.net abcdefgh 0 10X :something # SERVER whatever.net abcdefgh 0 10X :something
servername = args[1] servername = args[1]
numeric = args[4]
if args[2] != irc.serverdata['recvpass']: if args[2] != irc.serverdata['recvpass']:
# Check if recvpass is correct # Check if recvpass is correct
print('Error: recvpass from uplink server %s does not match configuration!' % servername) print('Error: recvpass from uplink server %s does not match configuration!' % servername)
sys.exit(1) sys.exit(1)
irc.servers[numeric] = IrcServer(None, servername)
return return
try: try:
real_args = [] real_args = []
@ -333,3 +360,20 @@ def handle_events(irc, data):
func(irc, numeric, command, args) func(irc, numeric, command, args)
except KeyError: # unhandled event except KeyError: # unhandled event
pass pass
def spawnServer(irc, name, sid, uplink=None, desc='PyLink Server'):
# -> :0AL SERVER test.server * 1 0AM :some silly pseudoserver
uplink = uplink or irc.sid
name = name.lower()
if sid in irc.servers:
raise ValueError('A server with SID %r already exists!' % sid)
for server in irc.servers.values():
if name == server.name:
raise ValueError('A server named %r already exists!' % name)
if not isInternalServer(irc, uplink):
raise ValueError('Server %r is not a PyLink internal PseudoServer!' % uplink)
if not utils.isServerName(name):
raise ValueError('Invalid server name %r' % name)
_sendFromServer(irc, uplink, 'SERVER %s * 1 %s :%s' % (name, sid, desc))
_sendFromServer(irc, uplink, 'ENDBURST')
irc.servers[sid] = IrcServer(uplink, name, internal=True)

View File

@ -48,6 +48,14 @@ def nickToUid(irc, nick):
if v.nick == nick: if v.nick == nick:
return k return k
def clientToServer(irc, numeric):
"""<irc object> <numeric>
Finds the server SID of user <numeric> and returns it."""
for server in irc.servers:
if numeric in irc.servers[server].users:
return server
# A+ regex # A+ regex
_nickregex = r'^[A-Za-z\|\\_\[\]\{\}\^\`][A-Z0-9a-z\-\|\\_\[\]\{\}\^\`]*$' _nickregex = r'^[A-Za-z\|\\_\[\]\{\}\^\`][A-Z0-9a-z\-\|\\_\[\]\{\}\^\`]*$'
def isNick(s, nicklen=None): def isNick(s, nicklen=None):
@ -56,7 +64,14 @@ def isNick(s, nicklen=None):
return bool(re.match(_nickregex, s)) return bool(re.match(_nickregex, s))
def isChannel(s): def isChannel(s):
return bool(s.startswith('#')) return s.startswith('#')
def _isASCIIPrintable(s):
return all(char in string.printable for char in s)
def isServerName(s):
return _isASCIIPrintable(s) and '.' in s and not s.startswith('.') \
and not s.endswith('.')
def parseModes(args): def parseModes(args):
"""['+mitl-o', '3', 'person'] => ['+m', '+i', '+t', '-o'] """['+mitl-o', '3', 'person'] => ['+m', '+i', '+t', '-o']