FireBot/bot.py

372 lines
15 KiB
Python
Raw Normal View History

2023-11-04 20:37:35 -05:00
#!/usr/bin/python3
from socket import socket, AF_INET, SOCK_STREAM
from overrides import bytes, bbytes
2023-11-07 09:05:23 -06:00
import logs
2023-11-04 20:37:35 -05:00
import re
2023-11-07 07:25:53 -06:00
from typing import NoReturn, Union
2023-11-06 18:18:52 -06:00
import commands as cmds
2023-11-07 09:05:23 -06:00
import config as conf
2023-11-07 07:53:14 -06:00
from time import sleep
2023-11-07 09:05:23 -06:00
from importlib import reload
import threads
2023-11-07 18:30:06 -06:00
import random as r
import handlers
import bare
2024-03-06 12:33:04 -06:00
from threading import Thread
from markov import MarkovBot
2024-05-23 13:12:46 -05:00
from traceback import format_exc
2023-11-05 18:50:04 -06:00
2023-11-07 20:30:35 -06:00
2023-11-07 07:25:53 -06:00
def mfind(message: str, find: list, usePrefix: bool = True) -> bool:
if usePrefix:
2023-11-07 09:05:23 -06:00
return any(message[: len(match) + 1] == conf.prefix + match for match in find)
2023-11-07 07:25:53 -06:00
else:
return any(message[: len(match)] == match for match in find)
2023-11-07 20:30:35 -06:00
class bot(bare.bot):
2023-11-06 16:23:31 -06:00
def __init__(self, server: str):
2023-11-04 20:37:35 -05:00
self.gmode = False
self.server = server
self.nicklen = 30
2023-11-07 09:05:23 -06:00
self.address = conf.servers[server]["address"]
2023-11-07 20:30:35 -06:00
self.port = (
conf.servers[server]["port"] if "port" in conf.servers[server] else 6667
)
2023-11-07 09:05:23 -06:00
self.channels = conf.servers[server]["channels"]
2024-04-06 23:45:38 -05:00
self.adminnames = (
conf.servers[server]["admins"] if "admins" in conf.servers[server] else []
)
self.ignores = (
2024-04-11 21:45:40 -05:00
conf.servers[server]["ignores"] if "ignores" in conf.servers[server] else []
2024-04-06 23:45:38 -05:00
)
self.__version__ = conf.__version__
self.npallowed = conf.npallowed
2023-11-04 20:37:35 -05:00
self.interval = (
2023-11-07 20:30:35 -06:00
conf.servers[server]["interval"]
if "interval" in conf.servers[server]
else 50
2023-11-04 20:37:35 -05:00
)
2024-04-20 02:48:43 -05:00
self.nick = (
conf.servers[server]["nick"]
if "nick" in conf.servers[server]
else "FireBot"
)
2024-02-14 21:17:08 -06:00
self.queue: list[bbytes] = [] # pyright: ignore [reportInvalidTypeForm]
2024-05-24 12:06:41 -05:00
self.statuses = {"firepup": {}}
self.ops = {}
2023-11-05 20:36:33 -06:00
self.sock = socket(AF_INET, SOCK_STREAM)
2023-11-20 17:52:18 -06:00
self.current = "user"
2024-04-06 23:45:38 -05:00
self.threads = (
conf.servers[server]["threads"] if "threads" in conf.servers[server] else []
)
self.onIdntCmds = (
2024-04-20 02:48:43 -05:00
conf.servers[server]["onIdntCmds"]
if "onIdntCmds" in conf.servers[server]
else []
)
self.onJoinCmds = (
2024-04-20 02:48:43 -05:00
conf.servers[server]["onJoinCmds"]
if "onJoinCmds" in conf.servers[server]
else []
)
self.onStrtCmds = (
2024-04-20 02:48:43 -05:00
conf.servers[server]["onStrtCmds"]
if "onStrtCmds" in conf.servers[server]
else []
)
self.autoMethod = (
2024-05-10 19:43:14 -05:00
conf.servers[server]["autoMethod"]
if "autoMethod" in conf.servers[server]
else "QUOTE"
)
2024-05-18 17:38:49 -05:00
self.dnsblMode = (
conf.servers[server]["dnsblMode"]
if "dnsblMode" in conf.servers[server]
else "none"
)
2024-06-10 04:20:45 -05:00
self.dns = {}
2024-03-06 12:33:04 -06:00
self.lastfmLink = conf.lastfmLink
with open("mastermessages.txt") as f:
TMFeed = []
for line in f.readlines():
TMFeed.extend([line.strip().split()])
self.markov = MarkovBot(TMFeed)
2024-05-11 22:30:31 -05:00
conf.prefix = (
conf.servers[server]["prefix"] if "prefix" in conf.servers[server] else "."
)
2023-11-07 07:53:14 -06:00
self.log(f"Start init for {self.server}")
2023-11-04 20:37:35 -05:00
2023-11-05 18:49:57 -06:00
def connect(self) -> None:
2023-11-07 07:53:14 -06:00
self.log(f"Joining {self.server}...")
2023-11-05 20:36:33 -06:00
self.sock.connect((self.address, self.port))
2024-04-20 02:48:43 -05:00
self.send("\n") # Just for sanity
if self.onStrtCmds:
for cmd in self.onStrtCmds:
self.send(cmd + "\n")
if "serverPass" in conf.servers[self.server]:
2024-04-20 02:48:43 -05:00
self.send(f"PASS {conf.servers[self.server]['serverPass']}\n")
2023-11-05 21:08:47 -06:00
self.send(f"USER {self.nick} {self.nick} {self.nick} {self.nick}\n")
self.send(f"NICK {self.nick}\n")
ircmsg = ""
while True:
ircmsg = self.recv().safe_decode()
2023-11-05 18:49:57 -06:00
if ircmsg != "":
2023-11-08 21:20:50 -06:00
code = 0
try:
code = int(ircmsg.split(" ", 2)[1].strip())
except (IndexError, ValueError):
2023-11-14 22:47:38 -06:00
pass
2023-11-05 18:49:57 -06:00
print(bytes(ircmsg).lazy_decode())
2023-11-14 18:50:26 -06:00
if "NICKLEN" in ircmsg:
2023-11-05 18:49:57 -06:00
self.nicklen = int(ircmsg.split("NICKLEN=")[1].split(" ")[0])
2023-11-07 07:53:14 -06:00
self.log(f"NICKLEN set to {self.nicklen}")
2023-11-14 18:37:46 -06:00
if code == 433:
2023-11-05 20:24:43 -06:00
self.log("Nickname in use", "WARN")
2023-11-05 21:08:47 -06:00
self.nick = f"{self.nick}{r.randint(0,1000)}"
self.send(f"NICK {self.nick}\n")
self.log(f"nick is now {self.nick}")
2023-11-14 18:37:46 -06:00
if code in [376, 422]:
self.log(f"Success by code: {code}")
break
2023-11-14 22:55:41 -06:00
if " MODE " in ircmsg or " PRIVMSG " in ircmsg:
self.log(f"Success by MSG/MODE")
break
2023-11-14 18:37:46 -06:00
if ircmsg.startswith("PING "):
2023-11-05 18:49:57 -06:00
self.ping(ircmsg)
2023-11-14 18:37:46 -06:00
if len(ircmsg.split("\x01")) == 3:
2023-11-09 15:36:34 -06:00
handlers.CTCP(self, ircmsg)
2023-11-14 18:50:26 -06:00
if "Closing link" in ircmsg:
2023-11-05 18:49:57 -06:00
self.exit("Closing Link")
else:
self.exit("Lost connection to the server")
2023-11-07 07:53:14 -06:00
self.log(f"Joined {self.server} successfully!")
2023-11-05 18:49:57 -06:00
2023-11-06 18:18:52 -06:00
def join(self, chan: str, origin: str, lock: bool = True) -> None:
2023-11-06 16:20:54 -06:00
self.log(f"Joining {chan}...")
2023-11-05 21:26:41 -06:00
chan = chan.replace(" ", "")
if "," in chan:
chans = chan.split(",")
for subchan in chans:
2023-11-06 16:20:54 -06:00
self.join(subchan, origin)
2023-11-05 21:26:41 -06:00
return
2023-11-17 21:24:10 -06:00
if chan.startswith("0") or (
chan == "#main" and lock and self.server != "replirc"
):
2023-11-05 21:26:41 -06:00
if origin != "null":
2023-11-07 22:23:40 -06:00
self.msg(f"Refusing to join channel {chan} (protected)", origin)
2023-11-05 21:26:41 -06:00
return
2023-11-07 07:53:14 -06:00
if chan in self.channels and lock:
2023-11-05 21:26:41 -06:00
if origin != "null":
2023-11-07 22:23:40 -06:00
self.msg(f"I'm already in {chan}.", origin)
2023-11-05 21:26:41 -06:00
return
2023-11-06 16:20:54 -06:00
self.send(f"JOIN {chan}\n")
2023-11-05 21:26:41 -06:00
while True:
ircmsg = self.recv().safe_decode()
2023-11-05 21:26:41 -06:00
if ircmsg != "":
code = 0
try:
code = int(ircmsg.split(" ", 2)[1].strip())
except (IndexError, ValueError):
pass
2023-11-05 21:26:41 -06:00
print(bytes(ircmsg).lazy_decode())
if ircmsg.startswith("PING "):
self.ping(ircmsg)
elif ircmsg.startswith("ERROR "):
self.exit("Lost connection to the server while joining a channel")
2023-11-05 21:26:41 -06:00
elif len(ircmsg.split("\x01")) == 3:
2023-11-09 15:36:34 -06:00
handlers.CTCP(self, ircmsg)
elif code == 403:
2023-11-05 21:26:41 -06:00
self.log(f"Joining {chan} failed", "WARN")
if origin != "null":
self.msg(f"{chan} is an invalid channel", origin)
2023-11-05 21:26:41 -06:00
break
elif code == 473:
self.log(f"Joining {chan} failed (+i)", "WARN")
2023-11-05 21:26:41 -06:00
if origin != "null":
self.msg(f"{chan} is +i, and I'm not invited.", origin)
2023-11-05 21:26:41 -06:00
break
2024-02-15 21:25:42 -06:00
elif code == 474:
self.log(f"Joining {chan} failed (+b)", "WARN")
if origin != "null":
self.msg(f"I'm banned from {chan}.", origin)
break
elif code == 475:
self.log(f"Joining {chan} failed (+k without/with bad key)")
if origin != "null":
self.msg(f"{chan} is +k, and either you didn't give me a key, or you gave me the wrong one.", origin)
break
2024-02-15 22:09:53 -06:00
elif code == 480:
self.log(f"Joining {chan} failed (+S)", "WARN")
if origin != "null":
2024-04-07 00:38:49 -05:00
self.msg(
f"{chan} is +S, and I'm not connected over SSL.", origin
)
2024-02-15 22:09:53 -06:00
break
elif code == 519:
self.log(f"Joining {chan} failed (+A)", "WARN")
if origin != "null":
self.msg(f"{chan} is +A, and I'm not an admin.", origin)
break
2023-11-06 16:04:46 -06:00
elif code == 520:
self.log(f"Joining {chan} failed (+O)", "WARN")
if origin != "null":
self.msg(f"{chan} is +O, and I'm not an operator.", origin)
break
elif code == 405:
self.log(f"Joining {chan} failed (too many channels)", "WARN")
if origin != "null":
self.msg(f"I'm in too many channels to join {chan}", origin)
break
elif code == 471:
self.log(f"Joining {chan} failed (+l)", "WARN")
if origin != "null":
self.msg(f"{chan} is +l, and is full", origin)
break
elif code == 366:
2023-11-07 07:53:14 -06:00
self.log(f"Joining {chan} succeeded")
2023-11-05 21:26:41 -06:00
if origin != "null":
self.msg(f"Joined {chan}", origin)
2023-11-05 21:26:41 -06:00
self.channels[chan] = 0
break
2023-11-07 07:59:01 -06:00
def ping(self, ircmsg: str) -> int:
pong = f"PONG :{ircmsg.split('PING :')[1]}\n"
print(pong, end="")
return self.send(pong)
2023-11-04 20:37:35 -05:00
def send(self, command: str) -> int:
2023-11-06 16:20:54 -06:00
return self.sock.send(bytes(command))
2023-11-04 20:37:35 -05:00
def recv(self) -> bytes:
2023-11-05 16:26:50 -06:00
if self.queue:
return bytes(self.queue.pop(0))
2024-05-18 22:23:36 -05:00
data = bytes(self.sock.recv(2048))
if data.lazy_decode() == "":
return data
2024-05-18 22:52:40 -05:00
while not data.endswith(b"\r\n"):
2024-05-18 22:23:36 -05:00
data += bytes(self.sock.recv(2048))
2024-05-18 22:52:40 -05:00
data = bytes(data.strip(b"\r\n"))
2023-11-05 16:26:50 -06:00
if b"\r\n" in data:
self.queue.extend(data.split(b"\r\n"))
return bytes(self.queue.pop(0))
return data
2023-11-04 20:37:35 -05:00
2023-11-08 21:20:50 -06:00
def log(self, message: str, level: str = "LOG") -> None:
2023-11-17 20:24:30 -06:00
logs.log(message, self.server, level)
2023-11-04 20:37:35 -05:00
2023-11-08 21:20:50 -06:00
def exit(self, message: str) -> NoReturn:
2023-11-07 09:05:23 -06:00
logs.log(message, self.server, "EXIT")
2023-11-04 20:37:35 -05:00
exit(1)
2023-11-06 00:29:00 -06:00
2023-11-07 07:25:53 -06:00
def msg(self, msg: str, target: str) -> None:
if not (target == "NickServ" and mfind(msg, ["IDENTIFY"], False)):
self.log(f"Sending {bytes(msg).lazy_decode()} to {target}")
else:
self.log("Identifying myself...")
self.send(f"PRIVMSG {target} :{msg}\n")
def op(self, name: str, chan: str) -> Union[int, None]:
if name != "":
self.log(f"Attempting op of {name} in {chan}...")
return self.send(f"MODE {chan} +o {name}\n")
2023-11-08 21:20:50 -06:00
return None
2023-11-07 07:25:53 -06:00
def notice(self, msg: str, target: str, silent: bool = False) -> int:
if not silent:
self.log(f"Sending {bytes(msg).lazy_decode()} to {target} (NOTICE)")
return self.send(f"NOTICE {target} :{msg}\n")
def sendraw(self, command: str) -> int:
self.log(f"RAW sending {command}")
command = f"{command}\n"
return self.send(command.replace("$BOTNICK", self.nick))
2023-11-07 07:25:53 -06:00
2023-11-06 18:18:52 -06:00
def mainloop(self) -> NoReturn:
self.log("Starting connection..")
self.connect()
2023-11-07 09:05:23 -06:00
if "pass" in conf.servers[self.server]:
2023-11-07 20:30:35 -06:00
self.msg(
f"IDENTIFY FireBot {conf.servers[self.server]['pass']}", "NickServ"
)
2023-11-06 18:18:52 -06:00
sleep(0.5)
if self.onIdntCmds:
for cmd in self.onIdntCmds:
self.send(cmd + "\n")
2023-11-06 18:18:52 -06:00
for chan in self.channels:
self.join(chan, "null", False)
if self.onJoinCmds:
for cmd in self.onJoinCmds:
self.send(cmd + "\n")
2024-03-06 12:33:04 -06:00
tMgr = None
if self.threads:
tdict = {}
for thread in self.threads:
tdict[thread] = threads.data[thread]
if tdict[thread]["passInstance"]:
2024-03-06 12:33:04 -06:00
tdict[thread]["args"] = [self]
tMgr = Thread(target=threads.threadManager, args=(tdict,))
2024-03-06 12:33:04 -06:00
tMgr.daemon = True
tMgr.start()
2023-11-06 18:18:52 -06:00
while 1:
raw = self.recv()
ircmsg = raw.safe_decode()
2023-11-06 18:18:52 -06:00
if ircmsg == "":
2023-11-07 07:59:01 -06:00
self.exit("Probably a netsplit")
2023-11-06 18:18:52 -06:00
else:
print(raw.lazy_decode(), sep="\n")
action = "Unknown"
try:
action = ircmsg.split(" ", 2)[1].strip()
except IndexError:
2024-04-30 21:19:28 -05:00
pass
2024-02-14 21:17:08 -06:00
self.tmpHost = ""
if action in handlers.handles:
res, chan = handlers.handles[action](self, ircmsg)
2023-11-17 23:23:46 -06:00
if res == "reload" and type(chan) == str:
2024-05-24 12:06:41 -05:00
try:
reload(conf)
self.adminnames = (
conf.servers[self.server]["admins"]
if "admins" in conf.servers[self.server]
else []
)
self.ignores = (
conf.servers[self.server]["ignores"]
if "ignores" in conf.servers[self.server]
else []
)
self.__version__ = conf.__version__
self.npallowed = conf.npallowed
self.interval = (
conf.servers[self.server]["interval"]
if "interval" in conf.servers[self.server]
else 50
)
conf.prefix = (
conf.servers[self.server]["prefix"]
if "prefix" in conf.servers[self.server]
else "."
)
reload(cmds)
reload(handlers)
self.msg("Reloaded successfully", chan)
except Exception:
Err = format_exc()
for line in Err.split("\n"):
self.log(line, "ERROR")
self.msg(
"Reload failed, likely partially reloaded. Please check error logs.",
chan,
)
2023-11-07 08:01:37 -06:00
else:
if ircmsg.startswith("PING "):
2023-11-07 08:02:05 -06:00
self.ping(ircmsg)
2023-11-07 08:01:37 -06:00
elif ircmsg.startswith("ERROR :Closing Link"):
2023-11-07 08:02:05 -06:00
self.exit("I got killed :'(")
2023-11-07 08:01:37 -06:00
elif ircmsg.startswith("ERROR :Ping "):
2023-11-07 08:02:05 -06:00
self.exit("Ping timeout")
else:
self.log("Unrecognized server request!", "WARN")
2023-11-08 21:24:43 -06:00
self.exit("While loop broken")