180 lines
5.5 KiB
Python
Executable File
180 lines
5.5 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
|
|
from collections import deque
|
|
from discord.utils import setup_logging
|
|
from typing import Optional
|
|
import argparse
|
|
import discord
|
|
import logging
|
|
import os.path
|
|
import random
|
|
import re
|
|
import shutil
|
|
import subprocess as sp
|
|
|
|
discord.utils.setup_logging()
|
|
_log = logging.getLogger(__name__)
|
|
|
|
_GAMES = "games.txt"
|
|
_DB = "holy-heck.db"
|
|
_DB_DEST = f"/var/lib/grafana/{_DB}"
|
|
|
|
|
|
def _write_game(content: str):
|
|
try:
|
|
with open(_GAMES, "a") as f:
|
|
f.write(content)
|
|
f.write("\n")
|
|
except:
|
|
_log.exception(f"failed writing game {content}")
|
|
|
|
|
|
def _update_db():
|
|
try:
|
|
games = []
|
|
with open(_GAMES) as f:
|
|
for line in f:
|
|
games.append(line.strip())
|
|
sp.run(["./index.py", "-o", _DB] + games)
|
|
shutil.move(_DB, _DB_DEST)
|
|
_log.info("updated db")
|
|
except:
|
|
_log.exception(f"failed updating db")
|
|
|
|
|
|
class BotClient(discord.Client):
|
|
def __init__(
|
|
self,
|
|
intents: discord.Intents,
|
|
leaguefacts: Optional[list[str]] = None,
|
|
):
|
|
super().__init__(intents=intents)
|
|
self._recentfacts = deque(maxlen=min(len(leaguefacts) - 1, 5))
|
|
self._leaguefacts = leaguefacts or []
|
|
self._leaguefacts.append(
|
|
f"I know {len(self._leaguefacts)} leaguefacts but you'll never see most of them."
|
|
)
|
|
random.seed()
|
|
|
|
async def on_ready(self):
|
|
_log.info(f"ready as {self.user}")
|
|
|
|
async def on_message(self, message: discord.Message):
|
|
content = message.content
|
|
if self.is_command(message):
|
|
await self.on_command(message)
|
|
elif self.is_replay(message):
|
|
await self.on_replay(message)
|
|
elif self.is_leaguefact(message):
|
|
await self.on_leaguefact(message)
|
|
|
|
def is_command(self, message: discord.Message) -> bool:
|
|
return message.content.startswith("%")
|
|
|
|
async def on_command(self, message: discord.Message):
|
|
match re.split(" +", message.content):
|
|
case ["%calc?"] | ["%calc", "?"]:
|
|
await message.reply(content=self._help_calculate())
|
|
case ["%calc", *args]:
|
|
try:
|
|
calc = await self._calculate(args)
|
|
await message.reply(content=calc.strip())
|
|
except Exception as e:
|
|
_log.exception("running calculation")
|
|
await message.reply(
|
|
content="```\n" + str(e) + "\n```", delete_after=10
|
|
)
|
|
case _:
|
|
_log.info(f"Unrecognised command {message.content}")
|
|
await message.reply(content="Unrecognised command.")
|
|
|
|
def _help_calculate(self) -> str:
|
|
return (
|
|
"Run Showdown damage calculations.\n"
|
|
"Example format:\n"
|
|
"` %calc -2 8 SpA Choice Specs Torkoal Overheat vs. 252 HP / 4+ SpD Assault Vest Abomasnow in Sun through Light Screen`\n"
|
|
"Supported: attacker/defender boosts, EVs, tera, item, ability, species; weather/terrain; screens."
|
|
"Not supported (popular): non-default multi hits; hazards."
|
|
)
|
|
|
|
async def _calculate(self, args: list[str]) -> str:
|
|
proc = sp.run(
|
|
["node", "calc_main.js", "--"] + args, stdout=sp.PIPE, stderr=sp.PIPE
|
|
)
|
|
stderr = proc.stderr.decode()
|
|
if proc.returncode != 0:
|
|
raise Exception(proc.stderr.decode())
|
|
if stderr:
|
|
_log.warning(f"running calculation '{args}': {stderr}")
|
|
return proc.stdout.decode()
|
|
|
|
def is_replay(self, message: discord.Message) -> bool:
|
|
if re.match("https://replay.pokemonshowdown.com/dl-.*", message.content):
|
|
return True
|
|
return False
|
|
|
|
async def on_replay(self, message: discord.Message):
|
|
_log.info(f"Recognised {message.content} as a League game")
|
|
_write_game(message.content)
|
|
_update_db()
|
|
|
|
def is_leaguefact(self, message: discord.Message) -> bool:
|
|
return message.content.lower() in ["leaguefact", "leaguefacts"]
|
|
|
|
async def on_leaguefact(self, message: discord.Message):
|
|
_log.info("leaguefact requested")
|
|
fact = self._select_leaguefact()
|
|
if fact:
|
|
await message.channel.send(f"Did you know? {fact}")
|
|
else:
|
|
await message.channel.send("There are no league facts.")
|
|
|
|
def _select_leaguefact(self) -> Optional[str]:
|
|
if not self._leaguefacts:
|
|
return None
|
|
choice = None
|
|
while True:
|
|
choice = random.choice(self._leaguefacts)
|
|
if choice not in self._recentfacts:
|
|
break
|
|
self._recentfacts.append(choice)
|
|
return choice
|
|
|
|
|
|
def main():
|
|
parser = argparse.ArgumentParser()
|
|
parser.add_argument(
|
|
"-t",
|
|
"--token-file",
|
|
metavar="FILE",
|
|
default="token",
|
|
help="file containing Discord API token",
|
|
)
|
|
parser.add_argument(
|
|
"-f",
|
|
"--facts",
|
|
metavar="FILE",
|
|
default="facts.txt",
|
|
help="file containing leagefacts",
|
|
)
|
|
|
|
args = parser.parse_args()
|
|
|
|
facts = []
|
|
if os.path.exists(args.facts):
|
|
with open(args.facts) as f:
|
|
for line in f:
|
|
facts.append(line)
|
|
|
|
intents = discord.Intents.default()
|
|
intents.message_content = True
|
|
client = BotClient(leaguefacts=facts, intents=intents)
|
|
|
|
with open(args.token_file) as f:
|
|
token = f.read().strip()
|
|
client.run(token, log_handler=None)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
main()
|