2015-03-19 20:55:18 +01:00
|
|
|
#!/usr/bin/python3
|
|
|
|
|
|
|
|
import imp
|
|
|
|
import os
|
2015-04-03 09:17:03 +02:00
|
|
|
import socket
|
2015-04-03 21:35:55 +02:00
|
|
|
import time
|
2015-04-25 07:37:07 +02:00
|
|
|
import sys
|
2015-06-24 04:29:53 +02:00
|
|
|
from collections import defaultdict
|
2015-07-10 03:29:00 +02:00
|
|
|
import threading
|
2015-04-25 07:37:07 +02:00
|
|
|
|
2015-07-05 22:22:17 +02:00
|
|
|
from log import log
|
2015-07-06 04:19:49 +02:00
|
|
|
import conf
|
2015-06-24 04:29:53 +02:00
|
|
|
import classes
|
2015-07-10 03:29:00 +02:00
|
|
|
import utils
|
2015-03-19 20:55:18 +01:00
|
|
|
|
2015-04-18 04:55:48 +02:00
|
|
|
class Irc():
|
2015-07-10 03:29:00 +02:00
|
|
|
def __init__(self, netname, proto, conf):
|
2015-04-04 03:45:18 +02:00
|
|
|
# Initialize some variables
|
2015-04-18 04:55:48 +02:00
|
|
|
self.connected = False
|
2015-07-10 03:29:00 +02:00
|
|
|
self.name = netname
|
2015-04-25 07:37:07 +02:00
|
|
|
self.conf = conf
|
2015-07-05 08:12:00 +02:00
|
|
|
# Server, channel, and user indexes to be populated by our protocol module
|
2015-05-31 07:15:19 +02:00
|
|
|
self.servers = {}
|
2015-07-05 08:12:00 +02:00
|
|
|
self.users = {}
|
|
|
|
self.channels = defaultdict(classes.IrcChannel)
|
|
|
|
# Sets flags such as whether to use halfops, etc. The default RFC1459
|
|
|
|
# modes are implied.
|
|
|
|
self.cmodes = {'op': 'o', 'secret': 's', 'private': 'p',
|
|
|
|
'noextmsg': 'n', 'moderated': 'm', 'inviteonly': 'i',
|
|
|
|
'topiclock': 't', 'limit': 'l', 'ban': 'b',
|
2015-07-06 04:19:49 +02:00
|
|
|
'voice': 'v', 'key': 'k',
|
|
|
|
# Type A, B, and C modes
|
|
|
|
'*A': 'b',
|
|
|
|
'*B': 'k',
|
|
|
|
'*C': 'l',
|
|
|
|
'*D': 'imnpstr'}
|
2015-07-05 08:12:00 +02:00
|
|
|
self.umodes = {'invisible': 'i', 'snomask': 's', 'wallops': 'w',
|
2015-07-06 04:19:49 +02:00
|
|
|
'oper': 'o',
|
|
|
|
'*A': '', '*B': '', '*C': 's', '*D': 'iow'}
|
2015-07-05 08:12:00 +02:00
|
|
|
self.maxnicklen = 30
|
2015-07-06 04:19:49 +02:00
|
|
|
self.prefixmodes = 'ov'
|
2015-04-03 09:17:03 +02:00
|
|
|
|
2015-07-10 03:29:00 +02:00
|
|
|
self.serverdata = conf['servers'][netname]
|
2015-07-06 04:19:49 +02:00
|
|
|
self.sid = self.serverdata["sid"]
|
2015-07-08 22:54:45 +02:00
|
|
|
self.botdata = conf['bot']
|
2015-07-06 04:19:49 +02:00
|
|
|
self.proto = proto
|
|
|
|
self.connect()
|
|
|
|
|
|
|
|
def connect(self):
|
2015-04-03 09:17:03 +02:00
|
|
|
ip = self.serverdata["ip"]
|
|
|
|
port = self.serverdata["port"]
|
2015-07-05 22:29:01 +02:00
|
|
|
log.info("Connecting to network %r on %s:%s", self.name, ip, port)
|
2015-04-03 09:17:03 +02:00
|
|
|
self.socket = socket.socket()
|
2015-07-10 03:29:00 +02:00
|
|
|
self.socket.setblocking(0)
|
2015-07-12 05:10:21 +02:00
|
|
|
self.socket.settimeout(180)
|
2015-04-03 09:17:03 +02:00
|
|
|
self.socket.connect((ip, port))
|
2015-07-06 04:19:49 +02:00
|
|
|
self.proto.connect(self)
|
2015-04-25 07:37:07 +02:00
|
|
|
self.loaded = []
|
2015-07-10 03:29:00 +02:00
|
|
|
reading_thread = threading.Thread(target = self.run)
|
2015-06-03 01:55:04 +02:00
|
|
|
self.connected = True
|
2015-07-10 03:29:00 +02:00
|
|
|
reading_thread.start()
|
|
|
|
|
|
|
|
def disconnect(self):
|
|
|
|
self.connected = False
|
|
|
|
self.socket.close()
|
2015-04-03 09:17:03 +02:00
|
|
|
|
2015-04-03 21:35:55 +02:00
|
|
|
def run(self):
|
2015-04-18 07:11:49 +02:00
|
|
|
buf = ""
|
|
|
|
data = ""
|
2015-04-18 04:55:48 +02:00
|
|
|
while self.connected:
|
2015-04-03 21:35:55 +02:00
|
|
|
try:
|
2015-05-31 07:15:19 +02:00
|
|
|
data = self.socket.recv(2048).decode("utf-8")
|
2015-04-25 07:37:07 +02:00
|
|
|
buf += data
|
2015-04-18 07:11:49 +02:00
|
|
|
if not data:
|
|
|
|
break
|
|
|
|
while '\n' in buf:
|
|
|
|
line, buf = buf.split('\n', 1)
|
2015-07-10 03:29:00 +02:00
|
|
|
log.debug("(%s) <- %s", self.name, line)
|
2015-04-25 07:37:07 +02:00
|
|
|
proto.handle_events(self, line)
|
2015-07-10 03:29:00 +02:00
|
|
|
except (socket.error, classes.ProtocolError) as e:
|
|
|
|
log.error('Disconnected from network %r: %s: %s, exiting.',
|
|
|
|
self.name, type(e).__name__, str(e))
|
|
|
|
self.disconnect()
|
2015-04-03 09:17:03 +02:00
|
|
|
|
|
|
|
def send(self, data):
|
2015-07-10 02:25:10 +02:00
|
|
|
# Safeguard against newlines in input!! Otherwise, each line gets
|
|
|
|
# treated as a separate command, which is particularly nasty.
|
|
|
|
data = data.replace('\n', ' ')
|
2015-04-03 09:17:03 +02:00
|
|
|
data = data.encode("utf-8") + b"\n"
|
2015-07-10 03:29:00 +02:00
|
|
|
log.debug("(%s) -> %s", self.name, data.decode("utf-8").strip("\n"))
|
2015-04-03 09:17:03 +02:00
|
|
|
self.socket.send(data)
|
2015-03-19 20:55:18 +01:00
|
|
|
|
2015-04-25 07:37:07 +02:00
|
|
|
def load_plugins(self):
|
2015-07-06 04:19:49 +02:00
|
|
|
to_load = conf.conf['plugins']
|
2015-04-25 07:37:07 +02:00
|
|
|
plugins_folder = [os.path.join(os.getcwd(), 'plugins')]
|
|
|
|
# Here, we override the module lookup and import the plugins
|
|
|
|
# dynamically depending on which were configured.
|
|
|
|
for plugin in to_load:
|
2015-06-08 02:04:23 +02:00
|
|
|
try:
|
|
|
|
moduleinfo = imp.find_module(plugin, plugins_folder)
|
2015-07-12 05:11:48 +02:00
|
|
|
pl = imp.load_source(plugin, moduleinfo[1])
|
|
|
|
self.loaded.append(pl)
|
2015-06-08 02:04:23 +02:00
|
|
|
except ImportError as e:
|
|
|
|
if str(e).startswith('No module named'):
|
2015-07-05 22:29:01 +02:00
|
|
|
log.error('Failed to load plugin %r: the plugin could not be found.', plugin)
|
2015-06-08 02:04:23 +02:00
|
|
|
else:
|
2015-07-05 22:29:01 +02:00
|
|
|
log.error('Failed to load plugin %r: import error %s', plugin, str(e))
|
2015-07-12 05:11:48 +02:00
|
|
|
else:
|
|
|
|
if hasattr(pl, 'main'):
|
|
|
|
log.debug('Calling main() function of plugin %r', pl)
|
|
|
|
pl.main(irc)
|
2015-07-05 22:44:48 +02:00
|
|
|
log.info("loaded plugins: %s", self.loaded)
|
2015-04-25 07:37:07 +02:00
|
|
|
|
2015-05-31 21:20:09 +02:00
|
|
|
if __name__ == '__main__':
|
2015-07-05 22:44:48 +02:00
|
|
|
log.info('PyLink starting...')
|
2015-07-06 04:19:49 +02:00
|
|
|
if conf.conf['login']['password'] == 'changeme':
|
2015-07-05 22:44:48 +02:00
|
|
|
log.critical("You have not set the login details correctly! Exiting...")
|
2015-06-17 05:05:41 +02:00
|
|
|
sys.exit(2)
|
|
|
|
protocols_folder = [os.path.join(os.getcwd(), 'protocols')]
|
2015-07-10 03:29:00 +02:00
|
|
|
for network in conf.conf['servers']:
|
|
|
|
protoname = conf.conf['servers'][network]['protocol']
|
|
|
|
try:
|
|
|
|
moduleinfo = imp.find_module(protoname, protocols_folder)
|
|
|
|
proto = imp.load_source(protoname, moduleinfo[1])
|
|
|
|
except ImportError as e:
|
|
|
|
if str(e).startswith('No module named'):
|
|
|
|
log.critical('Failed to load protocol module %r: the file could not be found.', protoname)
|
|
|
|
else:
|
|
|
|
log.critical('Failed to load protocol module: import error %s', protoname, str(e))
|
|
|
|
sys.exit(2)
|
2015-06-17 05:05:41 +02:00
|
|
|
else:
|
2015-07-12 05:11:48 +02:00
|
|
|
utils.networkobjects[network] = Irc(network, proto, conf.conf)
|
|
|
|
# This is a separate loop to make sure that ALL networks have their
|
|
|
|
# Irc objects added into utils.networkobjects, before we load any plugins
|
|
|
|
# that may require them.
|
|
|
|
for irc in utils.networkobjects.values():
|
|
|
|
irc.load_plugins()
|