PluralKit/bot/pluralkit/bot.py

123 lines
4.0 KiB
Python
Raw Normal View History

import asyncio
2018-07-16 18:53:41 +00:00
from datetime import datetime
2018-07-11 22:47:44 +00:00
import logging
2018-07-13 19:41:28 +00:00
import json
2018-07-11 22:47:44 +00:00
import os
2018-07-16 18:53:41 +00:00
import time
2018-07-11 22:47:44 +00:00
import discord
2018-07-15 23:11:51 +00:00
logging.basicConfig(level=logging.INFO, format="[%(asctime)s] [%(name)s] [%(levelname)s] %(message)s")
2018-07-11 22:47:44 +00:00
logging.getLogger("discord").setLevel(logging.INFO)
logging.getLogger("websockets").setLevel(logging.INFO)
logger = logging.getLogger("pluralkit.bot")
2018-07-15 23:04:40 +00:00
logger.setLevel(logging.INFO)
2018-07-11 22:47:44 +00:00
client = discord.Client()
2018-07-11 22:49:02 +00:00
2018-07-11 22:47:44 +00:00
@client.event
async def on_error(evt, *args, **kwargs):
2018-07-11 22:49:02 +00:00
logger.exception(
"Error while handling event {} with arguments {}:".format(evt, args))
2018-07-11 22:47:44 +00:00
@client.event
async def on_ready():
# Print status info
logger.info("Connected to Discord.")
2018-07-11 22:49:02 +00:00
logger.info("Account: {}#{}".format(
client.user.name, client.user.discriminator))
2018-07-11 22:47:44 +00:00
logger.info("User ID: {}".format(client.user.id))
2018-07-11 22:49:02 +00:00
2018-07-11 22:47:44 +00:00
@client.event
async def on_message(message):
# Ignore bot messages
if message.author.bot:
return
2018-07-11 22:49:02 +00:00
2018-07-11 22:47:44 +00:00
# Split into args. shlex sucks so we don't bother with quotes
args = message.content.split(" ")
2018-07-16 18:53:41 +00:00
from pluralkit import proxy, utils, stats
2018-07-13 21:03:35 +00:00
command_items = utils.command_map.items()
command_items = sorted(command_items, key=lambda x: len(x[0]), reverse=True)
2018-07-14 18:02:10 +00:00
2018-07-13 21:03:35 +00:00
prefix = "pk;"
for command, (func, _, _, _) in command_items:
2018-07-14 18:02:10 +00:00
if message.content.lower().startswith(prefix + command):
2018-07-13 21:03:35 +00:00
args_str = message.content[len(prefix + command):].strip()
args = args_str.split(" ")
2018-07-13 21:03:35 +00:00
# Splitting on empty string yields one-element array, remove that
if len(args) == 1 and not args[0]:
args = []
async with client.pool.acquire() as conn:
2018-07-16 18:53:41 +00:00
time_before = time.perf_counter()
2018-07-13 21:03:35 +00:00
await func(conn, message, args)
2018-07-16 18:53:41 +00:00
time_after = time.perf_counter()
# Report command time stats
execution_time = time_after - time_before
response_time = (datetime.now() - message.timestamp).total_seconds()
await stats.report_command(command, execution_time, response_time)
2018-07-13 21:03:35 +00:00
return
2018-07-12 01:21:25 +00:00
# Try doing proxy parsing
async with client.pool.acquire() as conn:
await proxy.handle_proxying(conn, message)
2018-07-11 22:47:44 +00:00
@client.event
2018-07-13 19:41:28 +00:00
async def on_socket_raw_receive(msg):
# Since on_reaction_add is buggy (only works for messages the bot's already cached, ie. no old messages)
# we parse socket data manually for the reaction add event
if isinstance(msg, str):
try:
msg_data = json.loads(msg)
if msg_data.get("t") == "MESSAGE_REACTION_ADD":
evt_data = msg_data.get("d")
if evt_data:
user_id = evt_data["user_id"]
message_id = evt_data["message_id"]
emoji = evt_data["emoji"]["name"]
async with client.pool.acquire() as conn:
from pluralkit import proxy
await proxy.handle_reaction(conn, user_id, message_id, emoji)
except ValueError:
pass
2018-07-11 22:49:02 +00:00
async def periodical_stat_timer(pool):
async with pool.acquire() as conn:
while True:
from pluralkit import stats
await stats.report_periodical_stats(conn)
await asyncio.sleep(30)
2018-07-11 22:47:44 +00:00
async def run():
2018-07-16 18:53:41 +00:00
from pluralkit import db, stats
2018-07-11 22:47:44 +00:00
try:
logger.info("Connecting to database...")
pool = await db.connect()
logger.info("Attempting to create tables...")
async with pool.acquire() as conn:
await db.create_tables(conn)
logger.info("Connecting to InfluxDB...")
2018-07-16 18:53:41 +00:00
await stats.connect()
logger.info("Starting periodical stat reporting...")
asyncio.get_event_loop().create_task(periodical_stat_timer(pool))
2018-07-11 22:47:44 +00:00
client.pool = pool
logger.info("Connecting to Discord...")
await client.start(os.environ["TOKEN"])
finally:
logger.info("Logging out from Discord...")
2018-07-11 22:49:02 +00:00
await client.logout()