This repository has been archived on 2023-04-13. You can view files and clone it, but you cannot make any changes to it's state, such as pushing and creating new issues, pull requests or comments.
CloudBot/core/irc.py
2013-10-02 19:35:44 +13:00

285 lines
9.5 KiB
Python
Executable file

import re
import socket
import time
import thread
import threading
import Queue
from ssl import wrap_socket, CERT_NONE, CERT_REQUIRED, SSLError
irc_prefix_rem = re.compile(r'(.*?) (.*?) (.*)').match
irc_noprefix_rem = re.compile(r'()(.*?) (.*)').match
irc_netmask_rem = re.compile(r':?([^!@]*)!?([^@]*)@?(.*)').match
irc_param_ref = re.compile(r'(?:^|(?<= ))(:.*|[^ ]+)').findall
def decode(txt):
for codec in ('utf-8', 'iso-8859-1', 'shift_jis', 'cp1252'):
try:
return txt.decode(codec)
except UnicodeDecodeError:
continue
return txt.decode('utf-8', 'ignore')
def censor(text):
return text
class RecieveThread(threading.Thread):
"""recieves messages from IRC and puts them in the input_queue"""
def __init__(self, socket, input_queue, timeout):
self.input_buffer = ""
self.input_queue = input_queue
self.socket = socket
self.timeout = timeout
threading.Thread.__init__(self)
def recv_from_socket(self, nbytes):
return self.socket.recv(nbytes)
def handle_receive_exception(self, error, last_timestamp):
print error
if time.time() - last_timestamp > self.timeout:
self.input_queue.put(StopIteration)
self.socket.close()
return True
return False
def get_timeout_exception_type(self):
return socket.timeout
def run(self):
last_timestamp = time.time()
while True:
try:
data = self.recv_from_socket(4096)
self.input_buffer += data
if data:
last_timestamp = time.time()
else:
if time.time() - last_timestamp > self.timeout:
self.input_queue.put(StopIteration)
self.socket.close()
return
time.sleep(1)
except (self.get_timeout_exception_type(), socket.error) as e:
if self.handle_receive_exception(e, last_timestamp):
return
continue
while '\r\n' in self.input_buffer:
line, self.input_buffer = self.input_buffer.split('\r\n', 1)
self.input_queue.put(decode(line))
class SSLRecieveThread(RecieveThread):
def __init__(self, socket, input_queue, timeout):
RecieveThread.Thread.__init__(self, socket, input_queue, timeout)
def recv_from_socket(self, nbytes):
return self.socket.read(nbytes)
def get_timeout_exception_type(self):
return SSLError
def handle_receive_exception(self, error, last_timestamp):
# this is terrible
if not "timed out" in error.args[0]:
raise
return RecieveThread.handle_receive_exception(self, error, last_timestamp)
class SendThread(threading.Thread):
"""sends messages from output_queue to IRC"""
def __init__(self, socket, conn_name, output_queue):
self.output_buffer = ""
self.output_queue = output_queue
self.conn_name = conn_name
self.socket = socket
self.shutdown = False
threading.Thread.__init__(self)
def run(self):
while not self.shutdown:
line = self.output_queue.get().splitlines()[0][:500]
print u"[{}]: {}".format(self.conn_name, line)
self.output_buffer += line.encode('utf-8', 'replace') + '\r\n'
while self.output_buffer:
sent = self.socket.send(self.output_buffer)
self.output_buffer = self.output_buffer[sent:]
class ParseThread(threading.Thread):
"""parses messages from input_queue and puts them in parsed_queue"""
def __init__(self, input_queue, output_queue, parsed_queue):
self.input_queue = input_queue # lines that were received
self.output_queue = output_queue # lines to be sent out
self.parsed_queue = parsed_queue # lines that have been parsed
threading.Thread.__init__(self)
def run(self):
while True:
# get a message from the input queue
msg = self.input_queue.get()
#if msg == StopIteration:
# self.irc.connect()
# continue
# parse the message
if msg.startswith(":"): # has a prefix
prefix, command, params = irc_prefix_rem(msg).groups()
else:
prefix, command, params = irc_noprefix_rem(msg).groups()
nick, user, host = irc_netmask_rem(prefix).groups()
mask = nick + "!" + user + "@" + host
paramlist = irc_param_ref(params)
lastparam = ""
if paramlist:
if paramlist[-1].startswith(':'):
paramlist[-1] = paramlist[-1][1:]
lastparam = paramlist[-1]
# put the parsed message in the response queue
self.parsed_queue.put([msg, prefix, command, params, nick, user, host,
mask, paramlist, lastparam])
# if the server pings us, pong them back
if command == "PING":
str = "PONG :" + paramlist[0]
self.output_queue.put(str)
class Connection(object):
"""handles an IRC connection"""
def __init__(self, name, host, port, input_queue, output_queue):
self.output_queue = output_queue # lines to be sent out
self.input_queue = input_queue # lines that were received
self.socket = self.create_socket()
self.conn_name = name
self.host = host
self.port = port
self.timeout = 300
def create_socket(self):
return socket.socket(socket.AF_INET, socket.TCP_NODELAY)
def connect(self):
self.socket.connect((self.host, self.port))
self.recieve_thread = RecieveThread(self.socket, self.input_queue, self.timeout)
self.recieve_thread.daemon = True
self.recieve_thread.start()
self.send_thread = SendThread(self.socket, self.conn_name, self.output_queue)
self.send_thread.daemon = True
self.send_thread.start()
def stop(self):
self.send_thread.shutdown = True
time.sleep(.1)
self.socket.close()
class SSLConnection(Connection):
"""handles a SSL IRC connection"""
def __init__(self, name, host, port, input_queue, output_queue, ignore_cert_errors):
self.ignore_cert_errors = ignore_cert_errors
Connection.__init__(self, name, host, port, input_queue, output_queue)
def create_socket(self):
return wrap_socket(Connection.create_socket(self), server_side=False,
cert_reqs=CERT_NONE if self.ignore_cert_errors else
CERT_REQUIRED)
class IRC(object):
"""handles the IRC protocol"""
def __init__(self, name, server, nick, port=6667, channels=[], conf={}):
self.name = name
self.channels = channels
self.conf = conf
self.server = server
self.port = port
self.nick = nick
self.vars = {}
self.parsed_queue = Queue.Queue() # responses from the server are placed here
# format: [rawline, prefix, command, params,
# nick, user, host, paramlist, msg]
self.parsed_queue = Queue.Queue()
self.input_queue = Queue.Queue()
self.output_queue = Queue.Queue()
self.connection = Connection(self.name, self.server, self.port,
self.input_queue, self.output_queue)
self.connection.connect()
self.set_pass(self.conf.get('server_password'))
self.set_nick(self.nick)
self.cmd("USER",
[self.conf.get('user', 'cloudbot'), "3", "*", self.conf.get('realname',
'CloudBot - http://git.io/cloudbot')])
self.parse_thread = ParseThread(self.input_queue, self.output_queue,
self.parsed_queue)
self.parse_thread.daemon = True
self.parse_thread.start()
def stop(self):
self.connection.stop()
def set_pass(self, password):
if password:
self.cmd("PASS", [password])
def set_nick(self, nick):
self.cmd("NICK", [nick])
def join(self, channel):
""" makes the bot join a channel """
self.send("JOIN {}".format(channel))
if channel not in self.channels:
self.channels.append(channel)
def part(self, channel):
""" makes the bot leave a channel """
self.cmd("PART", [channel])
if channel in self.channels:
self.channels.remove(channel)
def msg(self, target, text):
""" makes the bot send a PRIVMSG to a target """
self.cmd("PRIVMSG", [target, text])
def ctcp(self, target, ctcp_type, text):
""" makes the bot send a PRIVMSG CTCP to a target """
out = u"\x01{} {}\x01".format(ctcp_type, text)
self.cmd("PRIVMSG", [target, out])
def cmd(self, command, params=None):
if params:
params[-1] = ':' + params[-1]
self.send(command + ' ' + ' '.join(map(censor, params)))
else:
self.send(command)
def send(self, str):
self.output_queue.put(str)
class SSLIRC(IRC):
def __init__(self, name, server, nick, port=6667, channels=[], conf={},
ignore_certificate_errors=True):
self.ignore_cert_errors = ignore_certificate_errors
IRC.__init__(self, name, server, nick, port, channels, conf)
def create_connection(self):
return crlf_ssl_tcp(self.name, self.server, self.port, self.ignore_cert_errors)