IRCat/server.py

662 lines
40 KiB
Python
Raw Normal View History

2024-12-08 16:54:04 -08:00
#!/usr/bin/python3
2024-12-08 20:06:57 -08:00
__version__ = "0.0.1-pre-alpha"
2024-12-24 16:50:04 -08:00
print(f"Codename IRCat v{__version__}")
2024-12-08 16:54:04 -08:00
print("Welcome! /ᐠ ˵> ⩊ <˵マ")
2025-01-08 22:03:18 -08:00
import socket, ssl, time, threading, traceback, sys, subprocess, yaml, sqlite3, os, bcrypt, importlib
2024-12-08 16:54:04 -08:00
from requests import get
2024-12-08 18:55:43 -08:00
if not len(sys.argv) == 2:
print("IRCat requires the following arguments: config.yml")
2024-12-08 19:04:37 -08:00
sys.exit(1)
server = "127.0.0.1"
displayname = "foo"
2024-12-09 15:33:56 -08:00
identifier = "somewhere in the universe"
2024-12-11 12:54:12 -08:00
admin_nick = "admin"
data_path = ""
2024-12-15 15:39:48 -08:00
motd = """
____ _ ___ ____ ____ _
/ ___|___ __| | ___ _ __ __ _ _ __ ___ ___ |_ _| _ \ / ___|__ _| |_
| | / _ \ / _` |/ _ \ '_ \ / _` | '_ ` _ \ / _ \ | || |_) | | / _` | __|
| |__| (_) | (_| | __/ | | | (_| | | | | | | __/ | || _ <| |__| (_| | |_
2024-12-16 09:15:45 -08:00
\____\___/ \__,_|\___|_| |_|\__,_|_| |_| |_|\___| |___|_| \_\\\\____\__,_|\__|
2024-12-15 15:39:48 -08:00
https://ircat.xyz
2024-12-22 16:54:52 -08:00
This server doesn't have a MOTD in its configuration, or is invalid."""
2024-12-15 15:39:48 -08:00
motd_file = None
2024-12-15 19:30:22 -08:00
ping_timeout = 255
2024-12-26 16:23:58 -08:00
restrict_ip = ''
2025-01-03 16:44:42 -08:00
global banlist
banlist = {}
2025-01-08 21:53:37 -08:00
global modules
modules = {"sql_provider": None, "command": [], "allsocket": [], "banprovider": None}
2025-01-03 16:44:42 -08:00
def updateklines():
global banlist
try:
klines = open(data["klinepath"]).read().split("\n")
2025-01-03 17:45:50 -08:00
print(open(data["klinepath"]).read())
2025-01-03 16:44:42 -08:00
for i in klines:
2025-01-03 16:48:54 -08:00
specifiedip = i.split(" ")[0]
specifiedreason = " ".join(i.split(" ")[1:])
2025-01-03 16:44:42 -08:00
banlist[specifiedip] = specifiedreason
2025-01-03 17:11:56 -08:00
print(f"Updated ban list! {banlist}")
2025-01-03 16:44:42 -08:00
except:
2025-01-03 17:11:56 -08:00
print("Failed to update banlist...")
print(traceback.format_exc())
2025-01-03 16:44:42 -08:00
banlist = {}
2024-12-08 19:04:37 -08:00
with open(sys.argv[1], 'r') as file:
data = yaml.safe_load(file)
try: server = data["host"]
except: print("using fallback server address")
try: displayname = data["name"]
except: print("using fallback display name")
2024-12-09 15:39:05 -08:00
try: identifier = data["identifier"]
2024-12-09 15:33:56 -08:00
except: print("using fallback identifier")
2024-12-11 12:54:12 -08:00
try: admin_nick = data["admin-nick"]
except: print("using fallback admin nick")
try: data_path = data["data-path"]
except:
print("IRCat requires \"data-path\" in config.yml")
sys.exit(1)
2024-12-15 15:39:48 -08:00
try: motd = data["motd"]
except: print("using fallback MOTD")
try: motd_file = data["motd-file"]
except: print("Not reading MOTD from a file.")
2024-12-15 19:30:22 -08:00
try: ping_timeout = int(data["ping-timeout"])
except: print("Using 255 as a ping timeout.")
2024-12-26 16:23:58 -08:00
try: restrict_ip = data["restrict-ip"]
2024-12-26 16:22:39 -08:00
except: print("Listening on all hosts possible.")
2025-01-02 19:57:09 -08:00
try: ssl_option = bool(data["ssl"])
2025-01-02 19:45:42 -08:00
except:
print("SSL will be off.")
2025-01-02 19:57:09 -08:00
ssl_option = False
2025-01-02 19:45:42 -08:00
if ssl_option:
try: ssl_cert = data["ssl_cert"]
except:
print("IRCat needs an SSL cert to use SSL!")
sys.exit(1)
try: ssl_pkey = data["ssl_pkey"]
except:
print("IRCat needs an SSL Private Key to use SSL!")
2025-01-08 21:53:04 -08:00
try: modules = data["modules"]
except:
print("IRCat needs at least one module enabled.")
sys.exit(1)
2025-01-03 16:44:42 -08:00
updateklines()
2024-12-08 19:04:37 -08:00
file.close()
2024-12-08 20:03:55 -08:00
print("Successfully loaded config!")
2025-01-08 22:03:18 -08:00
for mod in modules:
i = mod
2025-01-08 21:53:04 -08:00
if not os.path.isabs(i):
2025-01-08 21:59:31 -08:00
i = os.path.dirname(__file__) + "/modules/" + i
2025-01-08 21:53:04 -08:00
try:
2025-01-08 22:03:18 -08:00
print(f"Importing module {mod}...")
2025-01-08 22:04:42 -08:00
spc = importlib.util.spec_from_file_location(mod, f"{i}.py")
temp_module = importlib.util.module_from_spec(spc)
spc.loader.exec_module(temp_module)
2025-01-08 21:53:04 -08:00
if temp_module.__ircat_type__ == "sql.provider":
if modules["sql_provider"] != None:
modules["sql_provider"] = temp_module
else:
2025-01-08 22:03:18 -08:00
raise Exception(f"Tried to import {mod} as an SQL provider, but something's already the SQL provider.")
2025-01-08 21:53:04 -08:00
elif temp_module.__ircat_type__ == "command":
modules["command"].append(temp_module)
except:
print(f"Module {i} failed to load.")
print(traceback.format_exc())
sys.exit(1)
if modules["sql_provider"] == None:
print("IRCat needs an SQL provider.")
sys.exit(1)
config = modules["sql_provider"].broker()
2025-01-02 19:45:42 -08:00
sockets = {}
sockets_ssl = {}
# Open the specified non-SSL sockets.
for i in restrict_ip.split(" "):
sockets[i] = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sockets[i].setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
sockets[i].bind((i,6667))
sockets[i].listen(1)
context = ssl.SSLContext(ssl.PROTOCOL_TLS_SERVER)
if ssl_option:
2025-01-02 19:59:36 -08:00
print(f"Loading SSL cert {ssl_cert} with key {ssl_pkey}")
2025-01-02 19:59:58 -08:00
context.load_cert_chain(ssl_cert, keyfile=ssl_pkey)
2025-01-02 19:45:42 -08:00
for i in restrict_ip.split(" "):
sockets_ssl[i] = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sockets_ssl[i].setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
sockets_ssl[i].bind((i,6697))
sockets_ssl[i].listen(1)
2024-12-11 14:59:32 -08:00
opened=True
2024-12-09 14:35:01 -08:00
reserved = ["nickserv", "chanserv", "gitserv"] # Reserved nicknames
2024-12-09 12:27:44 -08:00
nickname_list = {} # Stores nicknames and the respective sockets
2024-12-13 13:03:06 -08:00
lower_nicks = {"gitserv": "GitServ", "nickserv": "NickServ"} # Nicknames in lowercase
2024-12-09 12:27:44 -08:00
channels_list = {} # Store channels and their user lists
2024-12-27 20:16:23 -08:00
lower_chans = {} # Channel names in lowercase
2025-01-06 15:42:48 -08:00
property_list = {"GitServ": {"host": "IRCatCore", "username": "IRCat", "realname": "Codename IRCat Integrated services - Updates bot", "modes": "iw", "away": False},"NickServ": {"host": "IRCatCore", "username": "IRCat", "realname": "Codename IRCat Integrated services - Login bot", "away": False, "modes": "iw"}} # Stores properties for active users and channels
2024-12-09 16:45:21 -08:00
def pinger(nick, connection):
2024-12-10 21:35:50 -08:00
global property_list
2024-12-09 16:45:21 -08:00
while nick in property_list:
2024-12-15 19:29:27 -08:00
if (time.time() - property_list[nick]["last_ping"]) > 30 and not property_list[nick]["ping_pending"]:
2024-12-12 13:35:02 -08:00
if nick in property_list:
print("Sent ping message to " + nick)
property_list[nick]["ping_pending"] = True
time.sleep(0.5)
2025-01-06 15:38:03 -08:00
try:
connection.sendall(bytes(f"PING {server}\r\n","UTF-8"))
except Exception as ex:
property_list[nick]["cause"] = "Send error: " + str(ex)
print("SHUTTING DOWN FOR " + nick)
connection.shutdown(socket.SHUT_WR)
connection.close()
break
2024-12-15 19:29:27 -08:00
elif property_list[nick]["ping_pending"] and ((time.time() - property_list[nick]["last_ping"]) > ping_timeout):
2024-12-12 13:35:02 -08:00
if nick in property_list:
2024-12-15 19:29:27 -08:00
property_list[nick]["cause"] = f"Ping timeout: {ping_timeout} seconds"
2024-12-12 13:45:48 -08:00
print("SHUTTING DOWN FOR " + nick)
2024-12-12 13:35:02 -08:00
connection.shutdown(socket.SHUT_WR)
connection.close()
break
2025-01-02 19:45:42 -08:00
def session(connection, client, ip, ssl=False):
2024-12-15 19:08:18 -08:00
global property_list
2024-12-09 09:30:20 -08:00
pending = "*" # The nickname of the client
2024-12-08 20:02:59 -08:00
already_set = False # If the client gave the server a NICK packet
ready = False # If the client gave the server a USER packet
finished = False # If the server gave the client its information, indicating it's ready.
2024-12-09 14:08:31 -08:00
username = "oreo" # Username/ident specified by client
hostname = "" # Hostname, can be IP or domain
realname = "realname" # Realname specified by client
safe_quit = False # If the client safely exited, or if the server should manually drop the connection
cause = "Unknown" # The cause of the unexpected exit
2024-12-08 16:54:04 -08:00
try:
print("Connected to client IP: {}".format(client))
connection.sendall(bytes(f":{server} NOTICE * :*** Looking for your hostname...\r\n","UTF-8"))
2024-12-09 12:41:46 -08:00
try:
hostname = socket.gethostbyaddr(client[0])[0]
connection.sendall(bytes(f":{server} NOTICE * :*** Got it! {hostname}\r\n","UTF-8"))
except:
hostname = client[0]
connection.sendall(bytes(f":{server} NOTICE * :*** Oof! Can't find your hostname, using IP...\r\n","UTF-8"))
2025-01-03 16:46:26 -08:00
updateklines()
global banlist
2025-01-03 16:47:23 -08:00
if client[0] in banlist:
2025-01-03 16:44:42 -08:00
print("Specified IP is banned, killing now.")
2025-01-03 16:47:23 -08:00
reason = banlist[client[0]]
2025-01-03 16:44:42 -08:00
connection.sendall(bytes(f":{server} 465 * :You are banned from this server\r\n","UTF-8"))
2025-01-03 16:57:45 -08:00
connection.sendall(bytes(f"ERROR :Closing Link: {hostname} (K-Lined: {reason})\r\n","UTF-8"))
2025-01-03 16:49:37 -08:00
time.sleep(3)
2025-01-03 16:44:42 -08:00
raise Exception("Killed connection, IP is banned.")
2024-12-08 16:54:04 -08:00
while True:
try:
data = connection.recv(2048)
2025-01-06 15:38:03 -08:00
if not data:
cause = "Remote host closed the connection"
break
2024-12-09 21:58:05 -08:00
except Exception as ex:
cause = "Read error: " + str(ex)
break
2024-12-08 16:54:04 -08:00
print("Received data: {}".format(data))
try:
textt = data.decode()
2024-12-11 22:32:38 -08:00
for text in textt.replace("\r", "").split("\n"):
2024-12-09 11:16:30 -08:00
command = text.split(" ")[0].upper()
try:
args = text.split(" ")[1:]
except:
pass
2024-12-12 13:12:10 -08:00
if command == "NICK" and not finished:
2024-12-08 16:54:04 -08:00
pending = text.split(" ")[1]
2024-12-09 15:33:56 -08:00
if pending[0] == ":": pending[1:]
2024-12-09 21:43:11 -08:00
if "!" in pending or ":" in pending or "#" in pending or "*" in pending:
connection.sendall(bytes(f":{server} 432 * {pending} :Erroneus nickname\r\n","UTF-8"))
pending = "*"
elif pending.lower() in lower_nicks or pending in reserved:
2024-12-08 16:54:04 -08:00
connection.sendall(bytes(f":{server} 433 * {pending} :Nickname is already in use.\r\n","UTF-8"))
2024-12-09 09:30:20 -08:00
pending = "*"
2024-12-08 16:54:04 -08:00
else:
2024-12-12 08:45:11 -08:00
already_set = True
print(f"User {pending} set nick")
2024-12-09 11:07:26 -08:00
elif command == "USER":
2024-12-08 16:54:04 -08:00
if not ready:
username = text.split(" ")[1]
2024-12-09 13:44:15 -08:00
realname = " ".join(text.split(" ")[4:])[1:]
2024-12-08 20:06:57 -08:00
ready = True
2024-12-09 13:33:03 -08:00
elif command == "CAP":
2024-12-09 13:33:26 -08:00
if args[0] == "LS":
connection.sendall(bytes(f":{server} CAP * LS :ircat.xyz/foo\r\n", "UTF-8"))
2024-12-08 20:02:59 -08:00
elif (ready and already_set) and not finished:
2024-12-15 19:29:27 -08:00
cleanup_manual()
2024-12-11 22:01:28 -08:00
print(f"User {pending} successfully logged in.")
2024-12-09 14:35:01 -08:00
nickname_list[pending] = connection
2024-12-22 16:54:52 -08:00
property_list[pending] = {"host": hostname, "username": username, "realname": realname, "modes": "iw", "last_ping": time.time(), "ping_pending": False, "away": False}
2024-12-09 15:57:24 -08:00
lower_nicks[pending.lower()] = pending
2024-12-09 16:45:36 -08:00
threading.Thread(target=pinger, args=[pending, connection]).start()
2024-12-08 20:02:59 -08:00
connection.sendall(bytes(f":{server} 001 {pending} :Welcome to the {displayname} Internet Relay Chat Network {pending}\r\n", "UTF-8"))
connection.sendall(bytes(f":{server} 002 {pending} :Your host is {server}[{ip}/6667], running version IRCat-v{__version__}\r\n", "UTF-8"))
connection.sendall(bytes(f":{server} 004 {pending} {server} IRCat-{__version__} iow ovmsitnlbkq\r\n", "UTF-8"))
2024-12-26 16:47:12 -08:00
connection.sendall(bytes(f":{server} 005 {pending} CHANMODES=bq,k,l,irmnpst CHANTYPES=# NETWORK={displayname} :are supported by this server\r\n", "UTF-8"))
2024-12-15 15:39:48 -08:00
# connection.sendall(bytes(f":{server} 251 {pending} :There are {allusers} users and {allinvis} invisible in {servers} servers\r\n", "UTF-8")) Not supported as there isn't multi-server capability (yet)
ops = 0 # Placeholder, will replace with caclulating how much people have +o
connection.sendall(bytes(f":{server} 252 {pending} {ops} :IRC Operators online\r\n", "UTF-8"))
connection.sendall(bytes(f":{server} 253 {pending} 0 :unknown connection(s)\r\n", "UTF-8")) # Replace 0 with a variable of not setup clients.
chans = len(channels_list)
connection.sendall(bytes(f":{server} 254 {pending} {chans} :channels formed\r\n", "UTF-8"))
cleints = len(nickname_list)
servers = 1
connection.sendall(bytes(f":{server} 255 {pending} :I have {cleints} clients and {servers} servers\r\n", "UTF-8"))
# Start the MOTD
if motd_file != None:
motd = open(motd_file).read()
connection.sendall(bytes(f":{server} 375 {pending} :- {server} Message of the Day -\r\n", "UTF-8"))
for i in motd.rstrip().split("\n"):
connection.sendall(bytes(f":{server} 376 {pending} :- {i}\r\n", "UTF-8"))
connection.sendall(bytes(f":{server} 372 {pending} :End of /MOTD command\r\n", "UTF-8"))
# End the MOTD
2024-12-08 20:02:59 -08:00
connection.sendall(bytes(f":{pending} MODE {pending} +iw\r\n","UTF-8"))
finished = True
2024-12-09 11:07:26 -08:00
elif command == "PING":
2024-12-09 21:56:12 -08:00
e = text.split(" ")[1]
2024-12-10 10:25:15 -08:00
print("Replying with \"" + str([f":{server} PONG {server} :{e}\r\n"]) + "\"")
connection.sendall(bytes(f":{server} PONG {server} :{e}\r\n","UTF-8"))
2024-12-15 15:39:48 -08:00
elif command == "MOTD":
if motd_file != None:
motd = open(motd_file).read()
connection.sendall(bytes(f":{server} 375 {pending} :- {server} Message of the Day -\r\n", "UTF-8"))
for i in motd.rstrip().split("\n"):
connection.sendall(bytes(f":{server} 376 {pending} :- {i}\r\n", "UTF-8"))
connection.sendall(bytes(f":{server} 372 {pending} :End of /MOTD command\r\n", "UTF-8"))
2024-12-12 13:12:10 -08:00
elif finished:
2024-12-09 11:07:26 -08:00
if command == "JOIN":
2024-12-09 12:04:47 -08:00
channels = text.split(" ")[1]
for channelt in channels.split(","):
channel = channelt.strip()
2024-12-27 20:10:42 -08:00
if channel.lower() in lower_chans:
channel = lower_chans[channel.lower()]
2024-12-09 12:04:47 -08:00
success = True
if channel in channels_list:
if pending in channels_list[channel]:
success=False
print(f"{pending} is already in {channel} , ignoring JOIN request.")
if success:
try:
2024-12-09 12:04:47 -08:00
if channel in channels_list:
channels_list[channel].append(pending)
else:
channels_list[channel] = [pending]
2024-12-27 20:10:42 -08:00
lower_chans[channel.lower()] = channel
except:
2024-12-09 12:04:47 -08:00
connection.sendall(bytes(f":{server} NOTICE * :*** Could not join {channel}\r\n","UTF-8"))
print(channels_list)
for i in channels_list[channel]:
try:
2024-12-09 13:26:23 -08:00
nickname_list[i].sendall(bytes(f":{pending}!~{username}@{hostname} JOIN {channel}\r\n","UTF-8"))
2024-12-09 12:04:47 -08:00
except:
pass
2024-12-09 13:26:23 -08:00
# Code re-used in the NAMES command
2024-12-09 12:04:47 -08:00
if channel in channels_list:
if pending in channels_list[channel]:
users = " ".join(channels_list[channel])
connection.sendall(bytes(f":{server} 353 {pending} = {channel} :{users}\r\n","UTF-8"))
connection.sendall(bytes(f":{server} 366 {pending} {channel} :End of /NAMES list.\r\n","UTF-8"))
2024-12-09 12:41:46 -08:00
print("Successfully pre-loaded /NAMES list")
2024-12-09 16:20:40 -08:00
elif command == "PONG":
e = text.split(" ")[1]
if e == server:
2024-12-09 16:46:15 -08:00
print(pending + " replied to PING.")
2024-12-09 16:45:21 -08:00
property_list[pending]["last_ping"] = time.time()
property_list[pending]["ping_pending"] = False
2024-12-12 13:22:59 -08:00
elif command == "NICK":
2024-12-12 13:12:10 -08:00
if len(args) == 0:
connection.sendall(bytes(f":{server} 461 {pending} {command} :Not enough parameters\r\n","UTF-8"))
2024-12-13 13:37:52 -08:00
elif text.split(" ")[1] == pending:
pass
2024-12-12 13:12:10 -08:00
else:
pending2 = text.split(" ")[1]
if pending2[0] == ":": pending2[1:]
if "!" in pending2 or ":" in pending2 or "#" in pending2 or "*" in pending2:
connection.sendall(bytes(f":{server} 432 {pending} {pending2} :Erroneus nickname\r\n","UTF-8"))
2024-12-12 13:19:48 -08:00
elif pending2.lower() in lower_nicks or pending2 in reserved:
2024-12-12 13:12:10 -08:00
connection.sendall(bytes(f":{server} 433 {pending} {pending2} :Nickname is already in use.\r\n","UTF-8"))
else:
2024-12-12 13:52:18 -08:00
print("Sending nickname change...")
2024-12-12 13:56:22 -08:00
connection.sendall(bytes(f":{pending}!~{username}@{hostname} NICK {pending2}\r\n","UTF-8"))
2024-12-12 13:12:10 -08:00
# Broadcast the nickname change
done = []
for i, users in channels_list.items():
2024-12-12 13:15:01 -08:00
if pending in users:
for j in users:
2024-12-12 13:28:45 -08:00
if j != pending and j != pending2 and not j in done:
2024-12-16 09:28:49 -08:00
print("Broadcasting on " + j)
2024-12-12 13:15:01 -08:00
nickname_list[j].sendall(bytes(f":{pending}!~{username}@{hostname} {text}\r\n","UTF-8"))
done.append(j)
# Replace the nickname
try:
2024-12-12 13:45:48 -08:00
print("Changing on " + i)
2024-12-12 13:15:01 -08:00
channels_list[i].remove(pending)
channels_list[i].append(pending2)
except:
print(traceback.format_exc())
2024-12-16 09:28:49 -08:00
print("Moving config...")
property_list[pending2] = property_list.pop(pending)
nickname_list[pending2] = nickname_list.pop(pending)
del lower_nicks[pending.lower()]
lower_nicks[pending2.lower()] = pending2
print("starting pinger...")
pending = pending2
property_list[pending2]["ping_pending"] = False
property_list[pending2]["last_ping"] = time.time()
threading.Thread(target=pinger, args=[pending, connection]).start()
print(f"User {pending} set nick")
print("Broadcasting nickname change...")
2024-12-09 11:07:26 -08:00
elif command == "PART":
2024-12-09 18:27:40 -08:00
if len(args) == 0:
2024-12-09 21:56:12 -08:00
connection.sendall(bytes(f":{server} 461 {pending} {command} :Not enough parameters\r\n","UTF-8"))
2024-12-09 18:27:40 -08:00
else:
channel = text.split(" ")[1]
for i in channels_list[channel]:
try:
nickname_list[i].sendall(bytes(f":{pending}!~{username}@{hostname} {text}\r\n","UTF-8"))
except:
pass
2024-12-08 20:02:59 -08:00
try:
2024-12-09 18:27:40 -08:00
channels_list[channel].remove(pending)
2024-12-08 20:02:59 -08:00
except:
2024-12-09 18:27:40 -08:00
print(traceback.format_exc())
2024-12-22 16:54:52 -08:00
elif command == "AWAY":
if len(args) == 0:
property_list[pending]["away"] = False
connection.sendall(bytes(f":{server} 305 {pending} :You are no longer marked as being away\r\n","UTF-8"))
else:
reasons = " ".join(args)
if reasons[0] == ":": reasons = reasons[1:]
property_list[pending]["away"] = True
property_list[pending]["reason"] = reasons
connection.sendall(bytes(f":{server} 306 {pending} :You have been marked as being away\r\n","UTF-8"))
2024-12-09 11:07:26 -08:00
elif command == "WHO":
2024-12-09 18:27:40 -08:00
if len(args) == 0:
2024-12-09 21:56:12 -08:00
connection.sendall(bytes(f":{server} 461 {pending} {command} :Not enough parameters\r\n","UTF-8"))
2024-12-09 18:27:40 -08:00
else:
channel = text.split(" ")[1]
if channel in channels_list:
for i in channels_list[channel]:
who_host = property_list[i]["host"]
who_user = property_list[i]["username"]
who_realname = property_list[i]["realname"]
2024-12-22 16:54:52 -08:00
who_away = "G" if property_list[i]["away"] else "H"
connection.sendall(bytes(f":{server} 352 {pending} {channel} ~{who_user} {who_host} {server} {i} {who_away} :0 {who_realname}\r\n","UTF-8"))
2024-12-09 18:27:40 -08:00
elif channel in nickname_list:
who_host = property_list[channel]["host"]
who_user = property_list[channel]["username"]
who_realname = property_list[channel]["realname"]
2024-12-22 16:54:52 -08:00
who_away = "G" if property_list[channel]["away"] else "H"
connection.sendall(bytes(f":{server} 352 {pending} * ~{who_user} {who_host} {server} {channel} {who_away} :0 {who_realname}\r\n","UTF-8"))
2024-12-09 12:27:44 -08:00
2024-12-12 16:38:26 -08:00
connection.sendall(bytes(f":{server} 315 {pending} {channel} :End of /WHO list.\r\n","UTF-8"))
2024-12-09 14:35:01 -08:00
elif command == "WHOIS":
2024-12-09 18:27:40 -08:00
if len(args) == 0:
2024-12-09 21:56:12 -08:00
connection.sendall(bytes(f":{server} 461 {pending} {command} :Not enough parameters\r\n","UTF-8"))
2024-12-09 14:35:01 -08:00
else:
2024-12-09 18:27:40 -08:00
target = text.split(" ")[1]
if target.lower() in lower_nicks:
target = lower_nicks[target.lower()]
2024-12-09 18:27:40 -08:00
if target in property_list:
who_user = property_list[target]["username"]
who_realname = property_list[target]["realname"]
who_host = property_list[target]["host"]
2024-12-09 19:31:48 -08:00
try:
who_flags = property_list[target]["modes"]
except:
who_flags = None
connection.sendall(bytes(f":{server} 311 {pending} {target} ~{who_user} {who_host} * :{who_realname}\r\n","UTF-8"))
2024-12-09 18:27:40 -08:00
connection.sendall(bytes(f":{server} 312 {pending} {target} {server} :{identifier}\r\n","UTF-8"))
2024-12-22 16:54:52 -08:00
if "o" in who_flags: connection.sendall(bytes(f":{server} 313 {pending} {target} :is an IRC operator\r\n","UTF-8"))
who_away = property_list[target]["away"]
if who_away:
who_reason = who_away = property_list[target]["reason"]
connection.sendall(bytes(f":{server} 301 {pending} {target} :{who_reason}\r\n","UTF-8"))
#connection.sendall(bytes(f":{server} 317 {pending} {target} {time} :seconds idle\r\n","UTF-8")) # I haven't implemented idle time yet.
2024-12-09 19:31:48 -08:00
if who_flags != None and who_flags != "iw":
connection.sendall(bytes(f":{server} 379 {pending} {target} :Is using modes +{who_flags}\r\n","UTF-8"))
2024-12-09 18:27:40 -08:00
connection.sendall(bytes(f":{server} 318 {pending} {target} :End of /WHOIS list\r\n","UTF-8"))
else:
connection.sendall(bytes(f":{server} 401 {pending} {target} :No such nick/channel\r\n","UTF-8"))
2024-12-09 13:26:23 -08:00
elif command == "NAMES":
2024-12-09 18:27:40 -08:00
if len(args) == 0:
2024-12-09 21:56:12 -08:00
connection.sendall(bytes(f":{server} 461 {pending} {command} :Not enough parameters\r\n","UTF-8"))
2024-12-09 18:27:40 -08:00
else:
channel = text.split(" ")[1]
if channel in channels_list:
if pending in channels_list[channel]:
users = " ".join(channels_list[channel])
connection.sendall(bytes(f":{server} 353 {pending} = {channel} :{users}\r\n","UTF-8"))
connection.sendall(bytes(f":{server} 366 {pending} {channel} :End of /NAMES list.\r\n","UTF-8"))
2024-12-09 18:15:10 -08:00
elif command == "NOTICE":
2024-12-09 18:27:40 -08:00
if len(args) >= 2:
target = text.split(" ")[1]
if target.lower() in lower_nicks:
target = lower_nicks[target.lower()]
2024-12-09 18:27:40 -08:00
if target in channels_list:
if pending in channels_list[target]:
for i in channels_list[channel]:
try:
if i != pending:
nickname_list[i].sendall(bytes(f":{pending}!~{username}@{hostname} {text}\r\n","UTF-8"))
except:
pass
elif target in nickname_list:
nickname_list[target].sendall(bytes(f":{pending}!~{username}@{hostname} {text}\r\n","UTF-8"))
else:
connection.sendall(bytes(f":{server} 401 {pending} {target} :No such nick/channel\r\n","UTF-8"))
2024-12-08 20:02:59 -08:00
else:
2024-12-09 21:56:12 -08:00
connection.sendall(bytes(f":{server} 461 {pending} {command} :Not enough parameters\r\n","UTF-8"))
2024-12-09 11:07:26 -08:00
elif command == "QUIT":
2024-12-08 20:02:59 -08:00
# Parse the quit message.
done = []
2024-12-26 16:15:51 -08:00
if len(text.split(" ")) == 1:
msg = "Client Quit"
2024-12-08 20:02:59 -08:00
else:
2024-12-26 16:15:51 -08:00
msg = text.split(" ")[1:]
2024-12-26 16:53:30 -08:00
if msg[0][0] == ":":
msg[0]=msg[0][1:]
2024-12-26 16:15:51 -08:00
if len(msg) > 0:
mse = " ".join(msg)
msg = f"Quit: {mse}"
2024-12-08 20:02:59 -08:00
text = f"QUIT :{msg}"
2024-12-08 20:21:29 -08:00
# Broadcast all users in the joined channels that the person quit.
2024-12-08 20:02:59 -08:00
for i, users in channels_list.items():
if pending in users:
for j in users:
if j != pending and not j in done:
2024-12-09 13:26:23 -08:00
nickname_list[j].sendall(bytes(f":{pending}!~{username}@{hostname} {text}\r\n","UTF-8"))
2024-12-08 20:02:59 -08:00
done.append(j)
# Remove the quitting user from the channel.
2024-12-08 16:54:04 -08:00
try:
2024-12-08 20:02:59 -08:00
channels_list[i].remove(pending)
2024-12-08 16:54:04 -08:00
except:
2024-12-08 20:02:59 -08:00
print(traceback.format_exc())
2024-12-08 20:48:30 -08:00
# Confirm QUIT and close the socket.
2024-12-09 13:53:01 -08:00
try:
2024-12-09 13:48:45 -08:00
connection.sendall(bytes(f":{pending}!~{username}@{hostname} {text}\r\n","UTF-8"))
connection.sendall(bytes(f"ERROR :Closing Link: {hostname} ({msg})\r\n","UTF-8"))
2024-12-09 21:43:11 -08:00
finally:
connection.close()
safe_quit = True
2024-12-09 17:29:46 -08:00
break
2024-12-09 19:31:48 -08:00
elif command == "MODE":
2024-12-11 15:12:48 -08:00
target = args[0]
2024-12-09 19:31:48 -08:00
if len(args) == 0:
2024-12-09 21:56:12 -08:00
connection.sendall(bytes(f":{server} 461 {pending} {command} :Not enough parameters\r\n","UTF-8"))
2024-12-09 19:31:48 -08:00
elif len(args) == 1:
if args[0] == pending:
yourmodes = property_list[pending]["modes"]
2024-12-11 15:04:03 -08:00
connection.sendall(bytes(f":{server} 221 {pending} +{yourmodes}\r\n","UTF-8"))
2024-12-09 21:43:11 -08:00
elif args[0] in channels_list:
if args[0] in property_list:
if "modes" in property_list[args[0]]:
# Get the modes + parameters, then print them out.
modes = property_list[args[0]]["modes"]
params = property_list[args[0]]["params"]
2024-12-09 21:56:12 -08:00
connection.sendall(bytes(f":{server} 221 {pending} {target} +{modes} {params}\r\n","UTF-8"))
2024-12-09 21:43:11 -08:00
else:
# Default channel mode
2024-12-09 21:56:12 -08:00
connection.sendall(bytes(f":{server} 324 {pending} {target} +n\r\n","UTF-8"))
2024-12-09 21:43:11 -08:00
else:
# Default channel mode
2024-12-09 21:56:12 -08:00
connection.sendall(bytes(f":{server} 324 {pending} {target} +n\r\n","UTF-8"))
2024-12-09 21:43:11 -08:00
else:
if args[0][0] == "#":
2024-12-09 21:56:12 -08:00
connection.sendall(bytes(f":{server} 403 {pending} {target} :No such channel\r\n","UTF-8"))
2024-12-09 21:43:11 -08:00
else:
2024-12-09 21:56:12 -08:00
connection.sendall(bytes(f":{server} 505 {pending} :Cant change mode for other users\r\n","UTF-8"))
2024-12-09 19:31:48 -08:00
2024-12-11 15:28:52 -08:00
elif command == "GITSERV" or (command == "PRIVMSG" and args[0].lower() == "gitserv"):
if command == "PRIVMSG":
args = args[1:]
2024-12-15 19:10:18 -08:00
if args[0][0] == ":":
2024-12-28 14:58:26 -08:00
args[0] = args[0][1:]
2024-12-09 18:27:40 -08:00
if len(args) == 0:
2024-12-09 21:56:12 -08:00
connection.sendall(bytes(f":{server} 461 {pending} {command} :Not enough parameters\r\n","UTF-8"))
2024-12-09 18:27:40 -08:00
elif args[0].upper() == "PULL":
2024-12-09 18:30:43 -08:00
updater = subprocess.run(["git", "pull"], stdout=subprocess.PIPE)
2024-12-09 18:27:40 -08:00
if updater.stdout.decode().strip() == "Already up to date.":
2024-12-09 18:57:05 -08:00
connection.sendall(bytes(f":GitServ!~IRCat@IRCatCore NOTICE {pending} :Codename IRCat is already up-to-date.\r\n","UTF-8"))
2024-12-09 18:27:40 -08:00
else:
2024-12-09 18:57:05 -08:00
connection.sendall(bytes(f":GitServ!~IRCat@IRCatCore NOTICE {pending} :Done, it is recommended to use /RESTART if you're an IRC op\r\n","UTF-8"))
2024-12-09 18:27:40 -08:00
elif args[0].upper() == "VERSION":
connection.sendall(bytes(f":GitServ!~IRCat@IRCatCore NOTICE {pending} :Codename IRCat version {__version__}\r\n","UTF-8"))
connection.sendall(bytes(f":GitServ!~IRCat@IRCatCore NOTICE {pending} :This is Codename IRCat's integrated services.\r\n","UTF-8"))
else:
connection.sendall(bytes(f":GitServ!~IRCat@IRCatCore NOTICE {pending} :GitServ Usage:\r\n","UTF-8"))
connection.sendall(bytes(f":GitServ!~IRCat@IRCatCore NOTICE {pending} :PULL - Pulls the latest version of Codename IRCat\r\n","UTF-8"))
connection.sendall(bytes(f":GitServ!~IRCat@IRCatCore NOTICE {pending} :VERSION - Gets the version number of this service.\r\n","UTF-8"))
2024-12-13 13:03:06 -08:00
elif command == "NICKSERV" or (command == "PRIVMSG" and args[0].lower() == "nickserv"):
if command == "PRIVMSG":
args = args[1:]
2024-12-28 14:58:26 -08:00
if args[0][0] == ":":
args[0] = args[0][1:]
2024-12-13 13:03:06 -08:00
if len(args) == 0:
connection.sendall(bytes(f":{server} 461 {pending} {command} :Not enough parameters\r\n","UTF-8"))
elif args[0].upper() == "IDENTIFY":
2024-12-13 13:19:50 -08:00
pass
2024-12-13 13:03:06 -08:00
elif args[0].upper() == "VERSION":
connection.sendall(bytes(f":NickServ!~IRCat@IRCatCore NOTICE {pending} :Codename IRCat version {__version__}\r\n","UTF-8"))
connection.sendall(bytes(f":NickServ!~IRCat@IRCatCore NOTICE {pending} :This is Codename IRCat's integrated services.\r\n","UTF-8"))
else:
connection.sendall(bytes(f":NickServ!~IRCat@IRCatCore NOTICE {pending} :NickServ Usage:\r\n","UTF-8"))
2024-12-13 13:19:50 -08:00
connection.sendall(bytes(f":NickServ!~IRCat@IRCatCore NOTICE {pending} :IDENTIFY - Identifies your nickname\r\n","UTF-8"))
2024-12-13 13:03:06 -08:00
connection.sendall(bytes(f":NickServ!~IRCat@IRCatCore NOTICE {pending} :VERSION - Gets the version number of this service.\r\n","UTF-8"))
2024-12-11 14:50:16 -08:00
2024-12-11 14:55:35 -08:00
elif command == "RESTART":
if "o" in property_list[pending]["modes"]:
tcp_socket.shutdown(socket.SHUT_RDWR)
tcp_socket.close()
2024-12-11 15:22:20 -08:00
global opened
2024-12-11 14:59:32 -08:00
opened = False
2024-12-11 14:55:35 -08:00
else:
connection.sendall(bytes(f":{server} 481 {pending} :Permission Denied- You're not an IRC operator\r\n","UTF-8"))
2024-12-11 14:50:16 -08:00
elif command == "PRIVMSG":
if len(args) >= 2:
target = text.split(" ")[1]
if target.lower() in lower_nicks:
target = lower_nicks[target.lower()]
if target in channels_list:
if pending in channels_list[target]:
for i in channels_list[channel]:
try:
if i != pending:
nickname_list[i].sendall(bytes(f":{pending}!~{username}@{hostname} {text}\r\n","UTF-8"))
except:
pass
elif target in nickname_list:
nickname_list[target].sendall(bytes(f":{pending}!~{username}@{hostname} {text}\r\n","UTF-8"))
else:
connection.sendall(bytes(f":{server} 401 {pending} {target} :No such nick/channel\r\n","UTF-8"))
else:
connection.sendall(bytes(f":{server} 461 {pending} {command} :Not enough parameters\r\n","UTF-8"))
2024-12-09 10:22:43 -08:00
# Ignore empty text
elif text.split(" ")[0] == "":
pass
else:
# Unknown command
cmd = text.split(" ")[0]
connection.sendall(bytes(f":{server} 421 {pending} {cmd} :Unknown command\r\n","UTF-8"))
2024-12-08 21:44:32 -08:00
2024-12-09 14:08:31 -08:00
except Exception as ex:
2024-12-08 16:54:04 -08:00
print(traceback.format_exc())
2025-01-06 15:38:03 -08:00
cause = "Internal IRCat error: " + str(ex)
2024-12-08 16:54:04 -08:00
break
finally:
connection.close()
2024-12-15 19:29:27 -08:00
try:
if "cause" in property_list[pending]:
cause = property_list[pending]["cause"]
if pending != "*":
del nickname_list[pending]
del property_list[pending]
del lower_nicks[pending.lower()]
if not safe_quit:
done = []
for i, users in channels_list.items():
if pending in users:
for j in users:
if j != pending and not j in done:
2024-12-16 14:22:14 -08:00
try:
nickname_list[j].sendall(bytes(f":{pending}!~{username}@{hostname} QUIT :{cause}\r\n","UTF-8"))
done.append(j)
except:
print(traceback.format_exc())
2024-12-15 19:29:27 -08:00
# Remove the quitting user from the channel.
try:
channels_list[i].remove(pending)
except:
print(traceback.format_exc())
except:
2024-12-16 14:22:14 -08:00
print(traceback.format_exc())
2024-12-15 19:29:27 -08:00
cleanup_manual()
def cleanup():
while True:
time.sleep(15)
2024-12-15 19:39:41 -08:00
cleanup_manual()
2024-12-15 19:29:27 -08:00
def cleanup_manual():
global channels_list
global property_list
print("Cleaning up...")
2024-12-15 19:39:41 -08:00
for j, i in channels_list.items():
2024-12-15 19:29:27 -08:00
for h in i:
if not h in property_list:
print("Found a detached connection: " + h)
i.remove(h)
for k in channels_list[j]:
if k != h and k in nickname_list:
2024-12-16 09:28:49 -08:00
nickname_list[k].sendall(f":{h}!~DISCONNECTED@DISCONNECTED PART {j} :IRCat Cleanup: Found missing connection!!\r\n")
2025-01-02 19:45:42 -08:00
def tcp_session(sock):
2025-01-03 17:02:55 -08:00
while True:
2025-01-02 20:01:06 -08:00
try:
while opened:
print("Waiting for connection...")
2025-01-02 19:45:42 -08:00
connection, client = sock.accept()
ip_to = restrict_ip
threading.Thread(target=session, daemon=True, args=[connection, client, ip_to]).start()
2025-01-02 20:01:06 -08:00
except:
print("Something went wrong...")
print(traceback.format_exc())
2025-01-03 17:02:55 -08:00
def ssl_session(sock2):
with context.wrap_socket(sock2, server_side=True) as sock:
while True:
try:
while opened:
print("Waiting for connection...")
connection, client = sock.accept()
ip_to = restrict_ip
threading.Thread(target=session, daemon=True, args=[connection, client, ip_to]).start()
except:
print("Something went wrong...")
print(traceback.format_exc())
2025-01-02 19:52:51 -08:00
for ip, i in sockets.items():
2025-01-02 19:53:39 -08:00
print("Now listening on port 6667 with IP " + ip)
2025-01-02 19:52:13 -08:00
threading.Thread(target=tcp_session, args=[i]).start()
if ssl_option:
2025-01-02 19:52:51 -08:00
for ip, i in sockets_ssl.items():
2025-01-02 19:53:39 -08:00
print("Now listening on SSL port 6697 with IP " + ip)
2025-01-02 19:52:13 -08:00
threading.Thread(target=ssl_session, args=[i]).start()