Skip to content

Instantly share code, notes, and snippets.

@williballenthin
Last active January 2, 2016 18:59
Show Gist options
  • Save williballenthin/8347711 to your computer and use it in GitHub Desktop.
Save williballenthin/8347711 to your computer and use it in GitHub Desktop.
An IRC server that translates messages to/from a running instance of Skype.
#!/usr/bin/python
# TODOs
# Add support for:
# - WHOIS/WHO/WHOWAS
# - AWAY
# - creating chats
# - TOPIC (setting)
import logging
import socket
import select
import re
import datetime
from Skype4Py import Skype
import Skype4Py
import six
from irc import events
from irc import client
from irc import buffer
MOD_HANDLE = "[MOD]"
UNREAD_HANDLE = "UNREAD"
SRV_WELCOME = "Welcome to %s v%s, the ugliest IRC server in the world." % (
__name__, client.VERSION)
class IRCError(Exception):
"""
Exception thrown by IRC command handlers to notify
client of a server/client error.
"""
def __init__(self, code, value):
self.code = code
self.value = value
def __str__(self):
return repr(self.value)
@classmethod
def from_name(cls, name, value):
return cls(events.codes[name], value)
class IRCClient(six.moves.socketserver.BaseRequestHandler):
"""
IRC client connect and command handling. Client connection is handled by
the `handle` method which sets up a two-way communication with the client.
It then handles commands sent by the client by dispatching them to the
handle_ methods.
"""
class Disconnect(BaseException):
pass
def __init__(self, request, client_address, server):
self.user = None
self.host = client_address # Client's hostname / ip.
self.realname = None # Client's real name
self.nick = None # Client's currently registered nickname
self.send_queue = [] # Messages to send to client (strings)
self._joined_chats = set() # Skype chats that the client has interacted with
self._chat_channelnames = [] # list(chat.Name, irc_channel_name)
self._logger = logging.getLogger("IRCClient")
six.moves.socketserver.BaseRequestHandler.__init__(self, request,
client_address, server)
def handle(self):
self._logger.info('Client connected: %s', self.client_ident())
self.buffer = buffer.LineBuffer()
try:
while True:
self._handle_one()
except self.Disconnect:
self.request.close()
def _handle_one(self):
"""
Handle one read/write cycle.
"""
ready_to_read, ready_to_write, in_error = select.select(
[self.request], [self.request], [self.request], 0.1)
if in_error:
raise self.Disconnect()
# Write any commands to the client
while self.send_queue and ready_to_write:
msg = self.send_queue.pop(0)
self._send(msg)
# See if the client has any commands for us.
if ready_to_read:
self._handle_incoming()
def _handle_incoming(self):
try:
data = self.request.recv(1024)
except Exception:
raise self.Disconnect()
if not data:
raise self.Disconnect()
self.buffer.feed(data)
for line in self.buffer:
line = line.decode('utf-8')
self._handle_line(line)
def _handle_line(self, line):
try:
self._logger.debug('from %s: %s' % (self.client_ident(), line))
command, sep, params = line.partition(' ')
handler = getattr(self, 'handle_%s' % command.lower(), None)
if not handler:
self._logger.info('No handler for command: %s. '
'Full line: %s' % (command, line))
raise IRCError.from_name('unknowncommand',
'%s :Unknown command' % command)
response = handler(params)
except AttributeError as e:
self._logger.error(six.text_type(e))
raise
except IRCError as e:
response = ':%s %s %s' % (self.server.servername, e.code, e.value)
self._logger.error(response)
except Exception as e:
response = ':%s ERROR %r' % (self.server.servername, e)
self._logger.error(response)
raise
if response:
self._send(response)
def _send(self, msg):
self._logger.debug('to %s: %s', self.client_ident(), msg)
self.request.send(msg.encode('utf-8') + b'\r\n')
def handle_nick(self, params):
"""
Handle the initial setting of the user's nickname and nick changes.
"""
nick = params
# Valid nickname?
if re.search('[^a-zA-Z0-9\-\[\]\'`^{}_]', nick):
raise IRCError.from_name('erroneusnickname', ':%s' % nick)
if self.server.clients.get(nick, None) == self:
# Already registered to user
return
if nick in self.server.clients:
# Someone else is using the nick
raise IRCError.from_name('nicknameinuse', 'NICK :%s' % (nick))
if not self.nick:
# New connection and nick is available; register and send welcome
# and MOTD.
self.nick = nick
self.server.clients[nick] = self
response = ':%s %s %s :%s' % (self.server.servername,
events.codes['welcome'], self.nick, SRV_WELCOME)
self.send_queue.append(response)
response = ':%s 376 %s :End of MOTD command.' % (
self.server.servername, self.nick)
self.send_queue.append(response)
return
# Nick is available. Change the nick.
message = ':%s NICK :%s' % (self.client_ident(), nick)
self.server.clients.pop(self.nick)
self.nick = nick
self.server.clients[self.nick] = self
# Send a notification of the nick change to the client itself
return message
def handle_user(self, params):
"""
Handle the USER command which identifies the user to the server.
"""
params = params.split(' ', 3)
if len(params) != 4:
raise IRCError.from_name('needmoreparams',
'USER :Not enough parameters')
user, mode, unused, realname = params
self.user = user
self.mode = mode
self.realname = realname
return ''
def handle_ping(self, params):
"""
Handle client PING requests to keep the connection alive.
"""
response = ':%s PONG :%s' % (self.server.servername, self.server.servername)
return response
def handle_join(self, params):
# TODO: this is a disaster!
channel_names = params.split(' ', 1)[0] # Ignore keys
for channel_name in channel_names.split(','):
r_channel_name = channel_name.strip()
self._logger.info("Request to JOIN %s" % (r_channel_name))
found = False
for chat in self.server.skype.Chats:
if chat.Name == r_channel_name:
found = True
self._ensure_client_joined(chat)
# TODO: i don't like the early returns, but need a refactor first
return
if not found:
stripped_channel_name = r_channel_name.lstrip("#")
matches = []
for chat in self.server.skype.Chats:
if stripped_channel_name in chat.FriendlyName:
if stripped_channel_name == chat.FriendlyName:
self._ensure_client_joined(chat)
return
else:
matches.append(chat)
if len(matches) == 1:
self._ensure_client_joined(matches[0])
return
elif len(matches) > 1:
message = "%s :Ambiguous channel name matches [%s]" % \
(stripped_channel_name, ", ".join(map(lambda c: c.FriendlyName, matches)))
raise IRCError.from_name('nosuchchannel', message)
else:
self.build_all_friendly_channelnames()
matches = []
for skypename, channelname in self._chat_channelnames:
print stripped_channel_name, channelname
if stripped_channel_name in channelname:
if stripped_channel_name == channelname:
# perfect match
# TODO: refactor this out
found = False
for chat in self.server.skype.Chats:
if chat.Name == skypename:
self._ensure_client_joined(chat)
return
if not found:
raise IRCError.from_name('nosuchchannel',
'%s :No such channel (0) ' % stripped_channel_name)
else:
matches.append((skypename, channelname))
if len(matches) == 1:
found = False
for chat in self.server.skype.Chats:
if chat.Name == matches[0][0]:
self._ensure_client_joined(chat)
return
if not found:
raise IRCError.from_name('nosuchchannel',
'%s :No such channel (0) ' % stripped_channel_name)
elif len(matches) == 0:
raise IRCError.from_name('nosuchchannel',
'%s :No such channel (1) ' % stripped_channel_name)
else:
message = "%s :Ambiguous channel name matches [%s]" % \
(stripped_channel_name, ", ".join(map(lambda p: p[1], matches)))
raise IRCError.from_name('nosuchchannel', message)
def _mark_all_chat_messages_read(self, chat):
for message in chat.Messages:
if message.Status == "RECEIVED":
message.MarkAsSeen()
def handle_privmsg(self, params):
# TODO(wb): send privmsg to all other connected clients
target, sep, msg = params.partition(' ')
if not msg:
raise IRCError.from_name('needmoreparams',
'PRIVMSG :Not enough parameters')
msg = msg.lstrip(":")
if target.startswith('#') or target.startswith('$'):
self._logger.info("message to %s: %s" % (target, msg))
the_chat = None
for chat in self.server.skype.Chats:
if chat.Name == target:
the_chat = chat
if the_chat is None:
names = self.get_skypename_from_friendly_channelname(target)
if len(names) == 0:
raise IRCError.from_name('cannotsendtochan',
'%s :Cannot send to channel' % target)
elif len(names) > 0:
raise IRCError.from_name('cannotsendtochan',
'%s :Cannot send to channel %s, ambiguous: [%s]' % (target, ", ".join(names)))
else:
the_chat = names
the_chat.SendMessage(msg)
self._mark_all_chat_messages_read(the_chat)
else:
self._logger.info("user message [UNSUPPORTED!] to %s: %s" % (target, msg))
def _queue_client_message(self, id_, message):
composed = ":%s %d %s %s" % (self.server.servername, id_, self.client_ident(), message)
self._logger.info("SENDING: %s" % (composed))
self.send_queue.append(composed)
def _queue_client_mod_message(self, chat, message):
composed = ":%s NOTICE %s :%s" % (MOD_HANDLE, chat.Name, message)
self._logger.info("SENDING: %s" % (composed))
self.send_queue.append(composed)
def get_friendly_channelname_from_skypename(self, chat):
# first, try to find a previously generated name
for skypename, channelname in self._chat_channelnames:
if chat.Name == skypename:
return channelname
# next, try to create a nice name
channelname = None
members = [m for m in chat.Members]
if len(members) == 2:
if members[0] == self.server.skype.User():
channelname = members[1].Handle + "-priv"
else:
channelname = members[0].Handle + "-priv"
else:
is_generic_name = False
for m in members:
if m.FullName in chat.FriendlyName:
is_generic_name = True
break
if not is_generic_name:
channelname = chat.FriendlyName.replace(" ", "-")[:16]
elif chat.Topic != "":
channelname = chat.Topic.replace(" ", "-")[:16]
else:
channelname = chat.Name
while True:
if len(self._chat_channelnames) == 0:
self._chat_channelnames.append((chat.Name, channelname))
return channelname
has_conflict = False
for skypename, existing_channelname in self._chat_channelnames:
if existing_channelname == channelname and skypename == chat.Name:
# race with other channel, take the other one
return channelname
elif existing_channelname == channelname:
# name conflict, change ours and retry
channelname += "Q"
has_conflict = True
break
if not has_conflict:
self._chat_channelnames.append((chat.Name, channelname))
return channelname
def build_all_friendly_channelnames(self):
for chat in self.server.skype.Chats:
self.get_friendly_channelname_from_skypename(chat)
def get_skypename_from_friendly_channelname(self, friendlychannelname):
skypenames = []
for skypename, channelname in self._chat_channelnames:
if channelname == friendlychannelname:
skypenames.append(skypename)
return skypenames
def handle_list(self, params):
self._queue_client_message(321, "NAME :FRIENDLYNAME TIMESTAMP")
sortable_chats = []
unsortable_chats = []
for c in self.server.skype.Chats:
try:
ts = c.Messagse[0].Timestamp
sortable_chats.append(c)
except Exception:
self._logger.warn("Unable to get TS from chat")
unsortable_chats.append(c)
for chat in sorted(sortable_chats, key=lambda c: c.Messages[0].Timestamp):
ts = datetime.datetime.fromtimestamp(chat.Messages[0].Timestamp)
#self._queue_client_message(322, "%s :%s %s" % (chat.Name, chat.FriendlyName, ts))
self._queue_client_message(322, "%s : | %s | %s" % (self.get_friendly_channelname_from_skypename(chat), chat.FriendlyName, ts))
for chat in unsortable_chats:
#self._queue_client_message(322, "%s :%s %s" % (chat.Name, chat.FriendlyName, "UNKNOWN"))
self._queue_client_message(322, "%s : | %s | %s" % (self.get_friendly_channelname_from_skypename(chat), chat.FriendlyName, "UNKNOWN"))
self._queue_client_message(323, ":End of /LIST")
print self._chat_channelnames
def handle_mode(self, params):
pass
def client_ident(self):
"""
Return the client identifier as included in many command replies.
"""
return client.NickMask.from_params(self.nick, self.user,
self.server.servername)
def finish(self):
"""
The client conection is finished. Do some cleanup to ensure that the
client doesn't linger around in any channel or the client list, in case
the client didn't properly close the connection with PART and QUIT.
"""
self._logger.info('Client disconnected: %s', self.client_ident())
if self.nick is not None:
self.server.clients.pop(self.nick)
else:
print self.server.clients
self._logger.info('Connection finished: %s', self.client_ident())
def __repr__(self):
"""
Return a user-readable description of the client
"""
return '<%s %s!%s@%s (%s)>' % (
self.__class__.__name__,
self.nick,
self.user,
self.host[0],
self.realname,
)
def _ensure_client_joined(self, chat):
if chat.Name not in self._joined_chats:
self._logger.info("joining to %s" % (chat.Name))
message = ":%s JOIN :%s" % (self.client_ident(), chat.Name)
self.send_queue.append(message)
message = ":%s TOPIC %s :%s" % (MOD_HANDLE, chat.Name, chat.FriendlyName)
self.send_queue.append(message)
nicks = [m.Handle for m in chat.Members]
self._queue_client_message(353, "= %s :%s" % (chat.Name, " ".join(nicks)))
self._queue_client_message(366, "%s :End of /NAMES list" % (chat.Name))
unread_messages = sorted([m for m in chat.Messages if m.Status == "RECEIVED"], key=lambda m: m.Timestamp)
if len(unread_messages) > 0:
self._queue_client_mod_message(chat, "Begin of UNREAD messages")
for message in unread_messages:
self._logger.info("unread message: %s %s %s" % (message.Timestamp, message.FromHandle, message.Body))
ts = datetime.datetime.fromtimestamp(message.Timestamp)
# TODO(wb): handle long messages
# TODO(wb): handle messages with newlines
for part in message.Body.split("\n"):
if part.rstrip("\r\n ") == "":
continue
m = "<%s %s> %s" % (ts.isoformat(), message.FromHandle, part)
composed = ":%s NOTICE %s :%s" % (UNREAD_HANDLE, chat.Name, m)
self.send_queue.append(composed)
self._queue_client_mod_message(chat, "End of UNREAD messages")
self._joined_chats.add(chat.Name)
def handle_skype_incoming(self, message, status): #chat_name, user_handle, message):
self._ensure_client_joined(message.Chat)
# TODO(wb): handle long messages
# TODO(wb): handle messages with newlines
self._logger.info("incoming message: %s %s %s" % (message.Chat.Name, message.FromHandle, message.Body))
for part in message.Body.split("\n"):
if part.rstrip("\r\n ") == "":
continue
m = ":%s PRIVMSG %s :%s" % (message.FromHandle, message.Chat.Name, part)
self.send_queue.append(m)
class IRCServer(six.moves.socketserver.ThreadingMixIn,
six.moves.socketserver.TCPServer):
daemon_threads = True
allow_reuse_address = True
clients = {}
"Connected clients (IRCClient instances) by nick name"
def __init__(self, *args, **kwargs):
self._logger = logging.getLogger("IRCServer")
self.servername = 'localhost'
self.clients = {}
self.skype = Skype()
self.skype.Attach()
self.skype.OnUserAuthorizationRequestReceived = self._handle_auth_req
self.skype.OnMessageStatus = self._handle_recv_message
six.moves.socketserver.TCPServer.__init__(self, *args, **kwargs)
def _handle_auth_req(self, user):
self._logger.info("Received authorization request: %s" % (user))
def _handle_recv_message(self, message, status):
if status == Skype4Py.cmsReceived:
for client in self.clients.values():
client.handle_skype_incoming(message, status)
def main():
logger = logging.getLogger("main")
logging.basicConfig(format='%(asctime)s %(message)s', level=logging.INFO)
address = "0.0.0.0"
port = 6667
try:
ircserver = IRCServer((address, port), IRCClient)
logger.info("Server started on %s:%d" % (address, port))
ircserver.serve_forever()
except socket.error as e:
logger.error(repr(e))
raise SystemExit(-2)
if __name__ == "__main__":
import sys
if len(sys.argv) > 1 and sys.argv[1] == "profile":
import cProfile
cProfile.run("main()")
else:
main()
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment