From 226a5cd118dfcdef2e0feacf6375ecf03b0b6ba3 Mon Sep 17 00:00:00 2001 From: Parker Date: Sat, 16 Nov 2024 21:01:08 -0600 Subject: [PATCH 1/5] Update stats embed --- code/cogs/owner/stats.py | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/code/cogs/owner/stats.py b/code/cogs/owner/stats.py index 6ab9043..4bf41f8 100644 --- a/code/cogs/owner/stats.py +++ b/code/cogs/owner/stats.py @@ -74,9 +74,10 @@ class Stats(commands.Cog): embed = discord.Embed( title="Statistics", description=( - f"Total Guilds: `{len(self.bot.guilds):,}`\nTotal Commands:" - f" `{total_commands:,}`\n\nTotal Players:" - f" `{self.bot.lavalink.nodes[0].stats.playing_players}`\nLoad:" + f"Total Guilds: `{len(self.bot.guilds):,}`\n" + f"Total Commands: `{total_commands:,}`\n\n" + f"Total Players: `{len(self.bot.lavalink.get_players())}`\n" + "Load:" f" `{round(self.bot.lavalink.nodes[0].stats.lavalink_load * 100, 2)}%`" ), color=BOT_COLOR, From d97ab8b34e5bed801fe9fcbeaa35d82a6c99f575 Mon Sep 17 00:00:00 2001 From: Parker Date: Tue, 19 Nov 2024 21:40:23 -0600 Subject: [PATCH 2/5] Print error/warning if database connection cannot be made --- code/cogs/owner/stats.py | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/code/cogs/owner/stats.py b/code/cogs/owner/stats.py index 4bf41f8..258b102 100644 --- a/code/cogs/owner/stats.py +++ b/code/cogs/owner/stats.py @@ -3,7 +3,7 @@ import sqlite3 import discord import os -from utils.config import BOT_COLOR +from utils.config import BOT_COLOR, LOG class Stats(commands.Cog): @@ -15,6 +15,12 @@ class Stats(commands.Cog): os.makedirs("data") connection = sqlite3.connect("data/count.db") + if not connection: + LOG.error( + "Could not create connection to database. Likely permissions" + " issue." + ) + cursor = connection.cursor() cursor.execute( "CREATE TABLE IF NOT EXISTS count (command_name, count, PRIMARY" @@ -28,6 +34,9 @@ class Stats(commands.Cog): @tasks.loop(seconds=30) async def dump_count(self): connection = sqlite3.connect("data/count.db") + if not connection: + LOG.warning("No database connection. Skipping dump.") + cursor = connection.cursor() for command_name, count in self.bot.temp_command_count.items(): From f25d05ec4228713e6cdd68bb393a71d93078028d Mon Sep 17 00:00:00 2001 From: Parker Date: Tue, 19 Nov 2024 21:53:38 -0600 Subject: [PATCH 3/5] Properly check database writability --- code/cogs/owner/stats.py | 13 ++++--------- 1 file changed, 4 insertions(+), 9 deletions(-) diff --git a/code/cogs/owner/stats.py b/code/cogs/owner/stats.py index 258b102..5409bfb 100644 --- a/code/cogs/owner/stats.py +++ b/code/cogs/owner/stats.py @@ -14,13 +14,11 @@ class Stats(commands.Cog): if not os.path.exists("data"): os.makedirs("data") - connection = sqlite3.connect("data/count.db") - if not connection: - LOG.error( - "Could not create connection to database. Likely permissions" - " issue." - ) + if not os.access("data/count.db", os.W_OK): + LOG.error("Cannot write to data/count.db - check permissions") + return + connection = sqlite3.connect("data/count.db") cursor = connection.cursor() cursor.execute( "CREATE TABLE IF NOT EXISTS count (command_name, count, PRIMARY" @@ -34,9 +32,6 @@ class Stats(commands.Cog): @tasks.loop(seconds=30) async def dump_count(self): connection = sqlite3.connect("data/count.db") - if not connection: - LOG.warning("No database connection. Skipping dump.") - cursor = connection.cursor() for command_name, count in self.bot.temp_command_count.items(): From 082ca714a146ac956a82611755264e3268e05233 Mon Sep 17 00:00:00 2001 From: Parker Date: Tue, 19 Nov 2024 22:36:06 -0600 Subject: [PATCH 4/5] `bot.lavalink = None` if no connection is made --- code/cogs/music.py | 8 +++++++- code/cogs/owner/stats.py | 18 +++++++++++++++--- 2 files changed, 22 insertions(+), 4 deletions(-) diff --git a/code/cogs/music.py b/code/cogs/music.py index f3e98b4..0bed94b 100644 --- a/code/cogs/music.py +++ b/code/cogs/music.py @@ -1,8 +1,8 @@ import discord from discord.ext import commands -from discord import app_commands import lavalink from lavalink import errors +from discord.ext import tasks from utils.config import ( LAVALINK_HOST, @@ -111,10 +111,12 @@ class Music(commands.Cog): try: await node.get_version() except lavalink.errors.ClientError: + self.bot.lavalink = None LOG.error( "Authentication to lavalink node failed. Check your login" " credentials." ) + return else: await node.connect() LOG.info(f"Connected to lavalink node {node.name}") @@ -128,6 +130,10 @@ class Music(commands.Cog): async def create_player(interaction: discord.Interaction): """Create a player for the guild associated with the interaction, or raise an error""" + if not interaction.client.lavalink: + LOG.error("Lavalink is not connected.") + return + try: player = interaction.client.lavalink.player_manager.create( interaction.guild.id diff --git a/code/cogs/owner/stats.py b/code/cogs/owner/stats.py index 5409bfb..04da7ca 100644 --- a/code/cogs/owner/stats.py +++ b/code/cogs/owner/stats.py @@ -2,6 +2,7 @@ from discord.ext import commands, tasks import sqlite3 import discord import os +import lavalink from utils.config import BOT_COLOR, LOG @@ -80,13 +81,24 @@ class Stats(commands.Cog): description=( f"Total Guilds: `{len(self.bot.guilds):,}`\n" f"Total Commands: `{total_commands:,}`\n\n" - f"Total Players: `{len(self.bot.lavalink.get_players())}`\n" - "Load:" - f" `{round(self.bot.lavalink.nodes[0].stats.lavalink_load * 100, 2)}%`" ), color=BOT_COLOR, ) + # Determine the content of the Lavalink description + if self.bot.lavalink: + embed.description += ( + "Total Players:" + f" `{len(self.bot.lavalink.get_players())}`\n" + "Load:" + f" `{round(self.bot.lavalink.nodes[0].stats.lavalink_load * 100, 2)}%`" + ) + else: + embed.description += ( + "Total Players: `NO LAVALINK CONNECTION`\n" + "Load: `NO LAVALINK CONNECTION`" + ) + for entry in data: embed.add_field( name=entry[0], value=f"` {entry[1]:,} `", inline=True From fcbfe460701316ded25e29356ed1fda42386e5c0 Mon Sep 17 00:00:00 2001 From: Parker Date: Tue, 19 Nov 2024 23:39:59 -0600 Subject: [PATCH 5/5] Update stats - split into command group --- code/cogs/owner/stats.py | 69 +++++++++++++++++++++++++++++----------- 1 file changed, 51 insertions(+), 18 deletions(-) diff --git a/code/cogs/owner/stats.py b/code/cogs/owner/stats.py index 04da7ca..00a2c2f 100644 --- a/code/cogs/owner/stats.py +++ b/code/cogs/owner/stats.py @@ -2,7 +2,6 @@ from discord.ext import commands, tasks import sqlite3 import discord import os -import lavalink from utils.config import BOT_COLOR, LOG @@ -30,6 +29,16 @@ class Stats(commands.Cog): self.dump_count.start() + def millis_to_readable(self, ms): + hours = ms // 3600000 + ms %= 3600000 + minutes = ms // 60000 + ms %= 60000 + seconds = ms // 1000 + milliseconds = ms % 1000 + + return f"{hours:02}:{minutes:02}:{seconds:02}.{milliseconds:03}" + @tasks.loop(seconds=30) async def dump_count(self): connection = sqlite3.connect("data/count.db") @@ -59,10 +68,19 @@ class Stats(commands.Cog): except KeyError: self.bot.temp_command_count[interaction.command.name] = 1 - @commands.command() + @commands.group(invoke_without_command=True) @commands.dm_only() @commands.is_owner() async def stats(self, ctx: commands.Context): + await ctx.author.send( + f"This is a group command. Use `{self.bot.command_prefix}stats" + " bot/lavalink` to get specific statistics." + ) + + @stats.command() + @commands.dm_only() + @commands.is_owner() + async def bot(self, ctx: commands.Context): connection = sqlite3.connect("data/count.db") cursor = connection.cursor() @@ -85,20 +103,6 @@ class Stats(commands.Cog): color=BOT_COLOR, ) - # Determine the content of the Lavalink description - if self.bot.lavalink: - embed.description += ( - "Total Players:" - f" `{len(self.bot.lavalink.get_players())}`\n" - "Load:" - f" `{round(self.bot.lavalink.nodes[0].stats.lavalink_load * 100, 2)}%`" - ) - else: - embed.description += ( - "Total Players: `NO LAVALINK CONNECTION`\n" - "Load: `NO LAVALINK CONNECTION`" - ) - for entry in data: embed.add_field( name=entry[0], value=f"` {entry[1]:,} `", inline=True @@ -107,8 +111,37 @@ class Stats(commands.Cog): connection.close() await ctx.send(embed=embed) - @stats.error - async def stats_error(self, ctx, error): + @bot.error + async def bot_error(self, ctx, error): + return + + @stats.command() + @commands.dm_only() + @commands.is_owner() + async def lavalink(self, ctx: commands.Context): + if not self.bot.lavalink: + return await ctx.send("No connection with Lavalink.") + + embed = discord.Embed( + title="Lavalink Statistics", + color=BOT_COLOR, + ) + + for node in self.bot.lavalink.nodes: + embed.add_field( + name=node.name, + value=( + f"\tPlayers: `{node.stats.players}`\n\tUptime:" + f" `{self.millis_to_readable(node.stats.uptime)}`\n\tMemory" + f" Used: `{node.stats.memory_used / 1024 / 1024:.2f}MB`\n" + ), + inline=True, + ) + + await ctx.send(embed=embed) + + @lavalink.error + async def lavalink_error(self, ctx, error): return