hybridbot/hybridbot.py

388 lines
12 KiB
Python
Raw Permalink Normal View History

2017-06-04 12:32:26 +00:00
#!/usr/bin/env python3
2016-07-25 18:01:51 +00:00
2017-06-04 12:32:26 +00:00
import asyncio
2016-08-19 13:50:46 +00:00
import re
import signal
2016-07-25 18:01:51 +00:00
import sys
import time
2017-06-04 12:32:26 +00:00
import slixmpp
from configparser import SafeConfigParser
2017-06-03 19:12:30 +00:00
from irc.bot import SingleServerIRCBot
2016-07-25 18:01:51 +00:00
class IRCBot:
def __init__(self, opts, inter):
self.client = SingleServerIRCBot([(opts['server'], opts['port'])],
2017-06-04 12:32:26 +00:00
opts['nick'], opts['nick'])
self.conn = self.client.connection
2016-08-01 12:44:08 +00:00
self.nick = opts['nick']
self.pure_nick = self.nick
self.chan = opts['chan']
2016-08-01 12:44:08 +00:00
self.inter = inter
self.loop = None
2016-07-25 18:01:51 +00:00
def register_handlers(self):
self.conn.add_global_handler('welcome', self.on_session_start)
self.conn.add_global_handler('pubmsg', self.on_message)
self.conn.add_global_handler('action', self.on_message)
self.conn.add_global_handler('join', self.on_presence)
self.conn.add_global_handler('part', self.on_presence)
self.conn.add_global_handler('namreply', self.on_namreply)
self.conn.add_global_handler('kick', self.on_kick)
self.conn.add_global_handler('nicknameinuse', self.on_nicknameinuse)
def join_chan(self):
self.conn.part(self.chan, message='Replaced by new connection')
self.conn.join(self.chan)
def on_session_start(self, conn, event):
print('Connected to IRC')
self.join_chan()
2016-07-25 18:01:51 +00:00
def on_message(self, conn, event):
nick = event.source.split('!')[0]
body = ''.join(event.arguments)
typ = event.type
2016-07-25 18:01:51 +00:00
if typ == 'action':
body = '/me ' + body
2016-07-25 18:01:51 +00:00
self.call_outside(self.inter.relay_message, 'irc', nick, body)
2016-07-25 18:01:51 +00:00
def on_presence(self, conn, event):
try:
typ = event.type
nick = event.source.nick
2016-07-25 18:01:51 +00:00
if typ == 'part':
if nick != self.nick:
if nick in self.inter.get_irc_users():
self.call_outside(self.inter.remove_irc_user, nick)
2016-07-25 18:01:51 +00:00
else:
if nick != self.nick:
if nick not in self.inter.get_irc_users():
self.call_outside(self.inter.append_irc_user, nick)
except Exception as e:
2017-06-04 12:32:26 +00:00
print(str(e), file=sys.stderr)
2016-07-25 18:01:51 +00:00
def on_namreply(self, conn, event):
for nick in event.arguments[2].split():
if nick != conn.get_nickname():
self.call_outside(self.inter.append_irc_user, nick)
2016-07-25 18:01:51 +00:00
def on_kick(self, conn, event):
self.nick = self.pure_nick
conn.nick(self.nick)
time.sleep(0.5)
self.join_chan()
def on_nicknameinuse(self, conn, event):
self.nick = self.nick + '_'
conn.nick(self.nick)
2016-07-25 18:01:51 +00:00
2016-08-11 16:31:40 +00:00
def send_message(self, msg, lead_prefix='', prefix=''):
buf = 460
result = []
try:
2016-08-11 16:31:40 +00:00
for line in msg:
for i in range(0, len(line), buf):
result.append(line[i:i + buf])
self.conn.privmsg(self.chan, lead_prefix + result[0])
for r in result[1:]:
time.sleep(0.5)
self.conn.privmsg(self.chan, prefix + r)
except Exception as e:
2017-06-04 12:32:26 +00:00
print(str(e), file=sys.stderr)
2016-07-30 10:11:20 +00:00
def call_outside(self, func, *args):
assert self.loop
self.loop.call_soon_threadsafe(func, *args)
@asyncio.coroutine
2017-06-04 12:32:26 +00:00
def run(self):
self.loop = asyncio.get_event_loop()
self.register_handlers()
yield from self.loop.run_in_executor(None, self.client.start)
2016-07-30 10:11:20 +00:00
2016-07-25 18:01:51 +00:00
class XMPPBot:
def __init__(self, opts, inter, timeout=5.0):
2017-06-04 12:32:26 +00:00
self.client = slixmpp.ClientXMPP(opts['jid'], opts['passwd'])
2017-06-26 16:17:42 +00:00
self.client.register_plugin('xep_0199') # XMPP Ping.
self.client_ping = self.client.plugin['xep_0199']
self.client.register_plugin('xep_0045') # XMPP MUC.
self.client_muc = self.client.plugin['xep_0045']
self.muc_is_joined = False
self.client_ping.timeout = self.timeout = timeout
self.client_ping.keepalive = True
2016-08-01 12:44:08 +00:00
self.nick = opts['nick']
self.pure_nick = self.nick
self.muc = opts['muc']
2016-08-01 12:44:08 +00:00
self.inter = inter
2016-07-25 18:01:51 +00:00
def register_handlers(self):
2017-06-04 12:32:26 +00:00
self.client.add_event_handler('failed_all_auth',
self.on_failed_all_auth)
self.client.add_event_handler('session_start', self.on_session_start)
self.client.add_event_handler('got_online', self.on_got_online)
self.client.add_event_handler('disconnected', self.on_disconnected)
self.client.add_event_handler('groupchat_message', self.on_message)
self.client.add_event_handler('muc::%s::presence' % self.muc,
self.on_presence)
2017-06-03 19:12:30 +00:00
def connect(self):
2017-06-04 12:32:26 +00:00
self.client.connect()
2017-06-03 19:12:30 +00:00
2017-06-19 16:47:42 +00:00
def join_muc(self):
@asyncio.coroutine
def loop_cycle():
if self._join_muc_block > 0:
return
self._join_muc_block += 1
2017-06-19 16:47:42 +00:00
while not self.muc_is_joined:
2017-06-26 16:17:42 +00:00
self.client_muc.join_muc(self.muc, self.nick)
2017-06-19 16:47:42 +00:00
yield from asyncio.sleep(self.timeout)
self._join_muc_block -= 1
asyncio.async(loop_cycle())
_join_muc_block = 0
2017-06-04 12:32:26 +00:00
def on_failed_all_auth(event):
# print('could not connect!', file=sys.stderr)
print('Could not connect to the server, or password mismatch!',
file=sys.stderr)
sys.exit(1)
def on_session_start(self, event):
2017-06-04 12:32:26 +00:00
# print('connected with %s' %con, file=sys.stderr)
print('Connected to XMPP')
2017-06-04 12:32:26 +00:00
self.client.get_roster()
self.client.send_presence()
def on_got_online(self, event):
self.join_muc()
2016-07-28 18:14:33 +00:00
@asyncio.coroutine
def on_disconnected(self, event):
self.muc_is_joined = False
print('Connection lost, reattempting in %d seconds' % self.timeout)
yield from asyncio.sleep(self.timeout)
2017-06-03 19:12:30 +00:00
self.connect()
def on_message(self, event):
body = event['body']
nick = event['mucnick']
2016-07-25 18:01:51 +00:00
self.inter.relay_message('xmpp', nick, body)
2016-07-25 18:01:51 +00:00
@asyncio.coroutine
def on_presence(self, event):
2016-07-25 18:01:51 +00:00
try:
muc_plugin = self.client.plugin['xep_0045']
typ = event['muc']['type']
2016-07-28 18:14:33 +00:00
nick = event['muc']['nick']
2016-07-25 18:01:51 +00:00
if not typ:
typ = event['type']
if not nick:
2017-06-04 12:32:26 +00:00
nick = muc_plugin.get_nick(self.muc, event['from'])
if typ == 'available':
self.muc_is_joined = True
elif typ == 'error':
self.muc_is_joined = False
if event['error']['code'] == '409':
2017-06-26 16:17:42 +00:00
self.nick += '_'
self.join_muc()
elif typ == 'unavailable':
if nick != self.nick:
if nick in self.inter.get_xmpp_users():
self.inter.remove_xmpp_user(nick)
else:
self.muc_is_joined = False
self.nick = self.pure_nick
yield from asyncio.sleep(0.5)
self.join_muc()
2016-07-25 18:01:51 +00:00
else:
if nick != self.nick:
if nick not in self.inter.get_xmpp_users():
self.inter.append_xmpp_user(nick)
2016-07-25 18:01:51 +00:00
except Exception as e:
2017-06-04 12:32:26 +00:00
print(str(e), file=sys.stderr)
2016-07-25 18:01:51 +00:00
2016-08-11 16:31:40 +00:00
def send_message(self, msg, prefix=''):
try:
2016-08-11 16:31:40 +00:00
msg[0] = prefix + msg[0]
result = '\n'.join(msg)
self.client.send_message(mto=self.muc, mbody=result,
mtype='groupchat')
2016-08-11 16:31:40 +00:00
except Exception as e:
2017-06-04 12:32:26 +00:00
print(str(e), file=sys.stderr)
2017-06-04 12:32:26 +00:00
@asyncio.coroutine
def run(self):
2017-06-03 19:12:30 +00:00
self.register_handlers()
self.connect()
2016-07-25 18:01:51 +00:00
class Intermedia:
def __init__(self, shared_opts, irc_chan, xmpp_muc):
2016-08-01 12:44:08 +00:00
self.irc_chan = irc_chan
self.xmpp_muc = xmpp_muc
self.ircbot = None
self.xmppbot = None
self.irc_users = []
self.xmpp_users = []
2016-08-01 12:44:08 +00:00
self.prefix = shared_opts['prefix']
self.owner = shared_opts['owner']
2016-08-01 12:44:08 +00:00
def set_bots(self, ircbot, xmppbot):
self.ircbot = ircbot
self.xmppbot = xmppbot
2016-08-11 16:31:40 +00:00
def to_irc(self, msg, lead_prefix='', prefix=''):
if self.ircbot:
2016-08-11 16:31:40 +00:00
self.ircbot.send_message(msg, lead_prefix, prefix)
2016-08-11 16:31:40 +00:00
def to_xmpp(self, msg, prefix=''):
2016-08-01 12:44:08 +00:00
if self.xmppbot:
self.xmppbot.send_message(msg, prefix)
2016-07-25 18:01:51 +00:00
def relay_message(self, from_net, nick, body):
if not self.ircbot or not self.xmppbot:
return
if from_net != 'irc' and from_net != 'xmpp':
return
if from_net == 'irc' and nick == self.ircbot.nick or \
from_net == 'xmpp' and nick == self.xmppbot.nick:
return
if not body or len(body) <= 0:
return
try:
msg = body.replace('\r\n', '\n').replace('\r', '\n').split('\n')
if msg and len(msg) > 0:
if len(msg) == 1 and msg[0] == self.prefix + 'users':
irc_users = ', '.join(self.get_irc_users())
xmpp_users = ', '.join(self.get_xmpp_users())
if irc_users:
irc_users = '[ IRC Users ] ' + irc_users
if xmpp_users:
xmpp_users = '[ XMPP Users ] ' + xmpp_users
if from_net == 'irc':
for answer in [xmpp_users]:
2016-08-11 16:31:40 +00:00
self.to_irc([answer])
elif from_net == 'xmpp':
for answer in [irc_users]:
2016-08-11 16:31:40 +00:00
self.to_xmpp([answer])
elif len(msg) == 1 and msg[0] == self.prefix + 'help':
answer = 'The only command I have is \'' + self.prefix + \
'users\'. Also, my owner is ' + self.owner + '.'
if from_net == 'irc':
2016-08-11 16:31:40 +00:00
self.to_irc([answer])
elif from_net == 'xmpp':
2016-08-11 16:31:40 +00:00
self.to_xmpp([answer])
else:
nick_prefix = '[' + nick + '] '
nick_prefix_me = '***' + nick + ' '
if (not re.match('^/me .+$', msg[0])):
2016-08-11 16:31:40 +00:00
nick_prefix_lead = nick_prefix
else:
2016-08-11 16:31:40 +00:00
msg[0] = re.split('^/me ', msg[0])[1]
nick_prefix_lead = nick_prefix_me
if from_net == 'irc':
2016-08-11 16:31:40 +00:00
self.to_xmpp(msg, prefix=nick_prefix_lead)
elif from_net == 'xmpp':
2016-08-11 16:31:40 +00:00
self.to_irc(msg,
lead_prefix=nick_prefix_lead,
prefix=nick_prefix)
except Exception as e:
2017-06-04 12:32:26 +00:00
print(str(e), file=sys.stderr)
2016-07-25 18:01:51 +00:00
def get_irc_users(self):
return self.irc_users
def append_irc_user(self, user):
self.irc_users.append(user)
def remove_irc_user(self, user):
self.irc_users.remove(user)
def get_xmpp_users(self):
return self.xmpp_users
def append_xmpp_user(self, user):
self.xmpp_users.append(user)
def remove_xmpp_user(self, user):
self.xmpp_users.remove(user)
if __name__ == '__main__':
config = SafeConfigParser()
shared_opts = {}
2016-08-01 12:44:08 +00:00
xmpp_opts = {}
irc_opts = {}
2016-07-25 18:01:51 +00:00
if len(sys.argv) > 1:
config.read(sys.argv[1])
2016-07-25 18:01:51 +00:00
else:
config.read('config.ini')
2016-07-25 18:01:51 +00:00
if not config.sections():
2017-06-04 12:32:26 +00:00
print('Error: Configuration file does not exist or is empty.',
file=sys.stderr)
sys.exit(1)
shared_opts['prefix'] = config.get('Shared', 'prefix')
shared_opts['owner'] = config.get('Shared', 'owner')
2016-07-25 18:01:51 +00:00
irc_opts['chan'] = config.get('IRC', 'channel')
irc_opts['nick'] = config.get('IRC', 'nick')
irc_opts['server'] = config.get('IRC', 'server')
irc_opts['port'] = int(config.get('IRC', 'port'))
xmpp_opts['jid'] = config.get('XMPP', 'jid')
xmpp_opts['passwd'] = config.get('XMPP', 'password')
xmpp_opts['muc'] = config.get('XMPP', 'muc')
xmpp_opts['nick'] = config.get('XMPP', 'nick')
2016-07-25 18:01:51 +00:00
2016-08-10 21:31:07 +00:00
signal.signal(signal.SIGINT, signal.SIG_DFL)
2017-06-04 12:32:26 +00:00
loop = asyncio.get_event_loop()
2016-08-10 21:31:07 +00:00
inter = Intermedia(shared_opts, irc_opts['chan'], xmpp_opts['muc'])
2017-06-04 12:32:26 +00:00
ircbot = IRCBot(irc_opts, inter)
xmppbot = XMPPBot(xmpp_opts, inter)
inter.set_bots(ircbot, xmppbot)
2016-08-01 12:44:08 +00:00
2017-06-04 12:32:26 +00:00
asyncio.async(xmppbot.run())
asyncio.async(ircbot.run())
2017-06-04 12:32:26 +00:00
loop.run_forever()