You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

271 lines
7.7 KiB
Python

12 years ago
#!/usr/bin/python
import math, socket, ssl, select, time, threading, random, string, os
from collections import deque
12 years ago
EOC = "\r\n"
config_ownhost = "ircd.local"
config_password = ""
config_netname = "Cryto IRC"
config_version = "circd 0.1"
12 years ago
config_motd = "sample.motd"
12 years ago
def remove_from_list(ls, val):
return [value for value in ls if value is not val]
def split_irc(message):
if ":" in message:
first, second = message.split(":", 2)
return first.rstrip().split(" ") + [second]
else:
return message.split(" ")
class ircd:
channels = {}
users = {}
12 years ago
motd = ""
def __init__(self):
if config_motd != "":
try:
self.motd = open(config_motd, "r").read()
except IOError:
print "WARNING: Could not read MOTD file."
12 years ago
class listener:
ssl = False
server = None
12 years ago
client_list = []
client_map = {}
select_inputs = []
select_outputs = []
def __init__(self, server):
self.server = server
12 years ago
def start(self, interface, port, cert_path, key_path):
bindsocket = socket.socket()
bindsocket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
bindsocket.bind((interface, port))
bindsocket.listen(5)
self.select_inputs = [ bindsocket ]
while self.select_inputs:
readable, writable, error = select.select(self.select_inputs, self.select_outputs, self.select_inputs)
if len(readable) > 0:
for sock in readable:
try:
if sock is bindsocket:
newsocket, fromaddr = bindsocket.accept()
remote_ip, remote_port = fromaddr
if self.ssl == True:
connstream = ssl.wrap_socket(newsocket, server_side=True, certfile=cert_path, keyfile=key_path, ssl_version=ssl.PROTOCOL_TLSv1)
else:
connstream = newsocket
new_client = client(connstream, remote_ip, server, self)
12 years ago
self.select_inputs.append(connstream)
self.select_outputs.append(connstream)
self.client_map[connstream.fileno()] = new_client
self.client_list.append(new_client)
else:
data = sock.recv(1024)
cur_client = self.client_map[sock.fileno()]
if data:
cur_client.process_data(data)
else:
cur_client.end()
self.select_inputs = remove_from_list(self.select_inputs, sock)
print "NOTICE: Client disconnected"
except ssl.SSLError, err:
if err.args[0] == ssl.SSL_ERROR_WANT_READ:
select.select([sock], [], [])
elif err.args[0] == ssl.SSL_ERROR_WANT_WRITE:
select.select([], [sock], [])
else:
raise
else:
time.sleep(0.010)
class client:
buff = ""
stream = None
user = None
listener = None
12 years ago
ip = ""
def __init__(self, connstream, ip, server, listener):
12 years ago
self.ip = ip
self.stream = connstream
self.user = user(self, server)
self.listener = listener
12 years ago
def send_chunk(self, chunk):
try:
self.stream.send(chunk + EOC)
except socket.error:
# TODO: Log quit reason
try:
self.stream.shutdown(2)
self.close()
except socket.error:
pass
self.end()
self.listener.select_inputs = remove_from_list(self.listener.select_inputs, self.stream)
print "NOTICE: Client disconnected, possibly due to socket error"
12 years ago
def send_global_notice(self, notice):
self.send_chunk(":%s NOTICE %s" % (config_ownhost, notice))
def send_numeric(self, numeric, notice):
self.send_chunk(":%s %s %s %s" % (config_ownhost, numeric, self.user.nickname, notice))
12 years ago
def process_data(self, data):
self.buff += data
stack = self.buff.split("\n")
self.buff = stack.pop()
for chunk in stack:
print chunk
self.process_chunk(chunk.rstrip())
def process_chunk(self, chunkdata):
data = split_irc(chunkdata)
if data[0].upper() == "PING":
self.send_chunk("PONG %s" % data[1])
12 years ago
else:
self.user.process_data(data)
def end(self):
if self.user is not None:
self.user.end()
12 years ago
class channel:
presences = {}
name = ""
registered = False
class user:
client = None
server = None
12 years ago
registered = 0
registered_nick = False
registered_user = False
nickname = "*"
12 years ago
ident = ""
realname = ""
masked_host = ""
real_host = ""
ip = ""
def __init__(self, client, server):
self.server = server
12 years ago
self.client = client
self.client.send_global_notice("AUTH :*** Looking up your hostname...")
hostname, aliaslist, ipaddrlist = socket.gethostbyaddr(self.client.ip)
self.real_host = hostname
self.masked_host = hostname
self.client.send_global_notice("AUTH :*** Found your hostname")
if config_password == "":
self.registered = 1
def process_data(self, data):
data = deque(data)
if data[0].startswith(":"):
origin = data.popleft()
else:
origin = ""
data[0] = data[0].upper()
if data[0] in ["USER", "NICK"] and self.registered == 1:
if data[0] == "USER":
12 years ago
if len(data) >= 5:
self.ident = data[1]
self.realname = data[4]
self.registered_user = True
self.verify_registration()
else:
self.client.send_numeric("461", "USER :Not enough parameters.")
elif data[0] == "NICK":
12 years ago
if len(data) >= 2:
if data[1] not in self.server.users:
self.nickname = data[1]
self.registered_nick = True
self.verify_registration()
else:
self.client.send_numeric("433", "%s :Nickname is already in use." % data[1])
12 years ago
else:
self.client.send_numeric("461", "NICK :Not enough parameters.")
elif self.registered == 2 and data[0] == "PONG":
if data[1] == self.challenge:
self.finish_registration()
elif self.registered < 2:
self.client.send_numeric("451", "%s :You have not registered." % data[0])
elif self.registered < 3:
self.client.send_numeric("451", "%s :You have not completed the challenge PING." % data[0])
else:
12 years ago
if data[0] == "LUSERS":
self.send_lusers()
12 years ago
else:
self.client.send_numeric("421", "%s :Unknown command." % data[0])
12 years ago
def verify_registration(self):
if self.registered_nick == True and self.registered_user == True:
self.registered = 2
print "Client %s registered from IP %s, sending challenge string." % (self.nickname, self.client.ip)
self.send_challenge()
def send_challenge(self):
self.challenge = ''.join(random.choice(string.ascii_uppercase + string.digits) for i in xrange(8))
self.client.send_chunk("PING :%s" % self.challenge)
def finish_registration(self):
self.registered = 3
self.server.users[self.nickname] = self
self.client.send_numeric("001", ":Welcome to %s, %s!%s@%s" % (config_netname, self.nickname, self.ident, self.real_host))
self.client.send_numeric("002", ":Your host is %s, running %s." % (config_ownhost, config_version))
self.client.send_numeric("003", ":This server has been running since unknown.")
self.client.send_numeric("004", ":%s %s %s %s" % (config_ownhost, config_version, "", ""))
self.send_lusers()
12 years ago
self.send_motd()
def send_lusers(self):
self.client.send_numeric("251", ":There are %d users and 0 invisible on 1 server." % len(self.server.users))
self.client.send_numeric("252", "%d :operator(s) online" % 0)
self.client.send_numeric("254", "%d :channel(s) formed" % 0)
self.client.send_numeric("255", ":I have %d clients and 1 server." % len(self.server.users)) # TODO: Sum all clients of all listenersm rather than taking the usercount.
12 years ago
def send_motd(self):
if server.motd == "":
self.client.send_numeric("422", ":No MOTD was set.")
else:
self.client.send_numeric("375", ":- %s Message of the day -" % config_ownhost)
for line in server.motd.split("\n"):
if line.rstrip() != "":
self.client.send_numeric("372", ":- %s" % line.rstrip())
self.client.send_numeric("375", ":End of MOTD for %s." % config_ownhost)
def end(self):
del self.server.users[self.nickname]
12 years ago
class presence:
user = None
status = "none"
joined = 0
server = ircd()
l = listener(server)
12 years ago
l.start("0.0.0.0", 6667, "sample.cert", "sample.key")