2024-06-22 08:55:26 -04:00
|
|
|
import asyncio
|
|
|
|
import logging
|
|
|
|
|
2024-06-21 18:38:15 -04:00
|
|
|
import discord
|
|
|
|
from discord import app_commands
|
|
|
|
from twitchAPI.helper import first
|
2024-06-22 08:55:26 -04:00
|
|
|
from twitchAPI.twitch import Twitch
|
|
|
|
|
2024-06-21 18:38:15 -04:00
|
|
|
import config
|
2024-06-21 20:30:41 -04:00
|
|
|
from modules.data.db import get_connection
|
2024-06-21 18:38:15 -04:00
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
class TwitchModule:
|
|
|
|
def __init__(self, bot):
|
|
|
|
self.bot = bot
|
2024-06-22 08:55:26 -04:00
|
|
|
self.twitch = Twitch(config.TWITCH_CLIENT_ID, config.TWITCH_CLIENT_SECRET)
|
2024-06-21 18:38:15 -04:00
|
|
|
self.bot.loop.create_task(self.authenticate_twitch())
|
2024-06-21 20:30:41 -04:00
|
|
|
self.bot.loop.create_task(self.check_live_streams())
|
2024-06-21 18:38:15 -04:00
|
|
|
self.add_commands()
|
|
|
|
|
|
|
|
async def authenticate_twitch(self):
|
2024-06-21 20:30:41 -04:00
|
|
|
await self.twitch.authenticate_app([]) # Authenticate without scopes
|
|
|
|
|
|
|
|
async def check_live_streams(self):
|
|
|
|
while True:
|
|
|
|
conn = get_connection()
|
|
|
|
c = conn.cursor()
|
2024-06-22 08:55:26 -04:00
|
|
|
c.execute(
|
|
|
|
"""
|
2024-06-21 20:30:41 -04:00
|
|
|
CREATE TABLE IF NOT EXISTS live_status (
|
|
|
|
twitch_name TEXT PRIMARY KEY,
|
|
|
|
is_live BOOLEAN
|
|
|
|
)
|
2024-06-22 08:55:26 -04:00
|
|
|
"""
|
|
|
|
)
|
2024-06-21 20:30:41 -04:00
|
|
|
conn.commit()
|
|
|
|
|
2024-06-22 08:55:26 -04:00
|
|
|
c.execute("SELECT twitch_name, discord_channel_id FROM followed_channels")
|
2024-06-21 20:30:41 -04:00
|
|
|
followed_channels = c.fetchall()
|
|
|
|
|
|
|
|
for twitch_name, discord_channel_id in followed_channels:
|
|
|
|
try:
|
|
|
|
user_info = await first(self.twitch.get_users(logins=[twitch_name]))
|
|
|
|
if not user_info:
|
|
|
|
continue
|
|
|
|
|
|
|
|
user_id = user_info.id
|
|
|
|
streams = await first(self.twitch.get_streams(user_id=[user_id]))
|
|
|
|
is_live = streams is not None
|
|
|
|
|
2024-06-22 08:55:26 -04:00
|
|
|
c.execute(
|
|
|
|
"SELECT is_live FROM live_status WHERE twitch_name = ?",
|
|
|
|
(twitch_name,),
|
|
|
|
)
|
2024-06-21 20:30:41 -04:00
|
|
|
row = c.fetchone()
|
|
|
|
was_live = row[0] if row else False
|
|
|
|
|
|
|
|
if is_live and not was_live:
|
|
|
|
channel = self.bot.get_channel(discord_channel_id)
|
|
|
|
if channel:
|
|
|
|
embed = discord.Embed(
|
|
|
|
title=f"{twitch_name} is Live!",
|
|
|
|
description=(
|
|
|
|
f"**Title:** {streams.title}\n"
|
|
|
|
f"**Game:** {streams.game_name}\n"
|
|
|
|
f"**Viewers:** {streams.viewer_count}"
|
|
|
|
),
|
2024-06-22 08:55:26 -04:00
|
|
|
color=discord.Color.green(),
|
2024-06-21 20:30:41 -04:00
|
|
|
)
|
|
|
|
embed.set_thumbnail(
|
2024-06-22 08:55:26 -04:00
|
|
|
url=streams.thumbnail_url.replace(
|
|
|
|
"{width}", "320"
|
|
|
|
).replace("{height}", "180")
|
2024-06-21 20:30:41 -04:00
|
|
|
)
|
|
|
|
await channel.send(embed=embed)
|
|
|
|
|
2024-06-22 08:55:26 -04:00
|
|
|
c.execute(
|
|
|
|
"INSERT OR REPLACE INTO live_status (twitch_name, is_live) VALUES (?, ?)",
|
|
|
|
(twitch_name, is_live),
|
|
|
|
)
|
2024-06-21 20:30:41 -04:00
|
|
|
conn.commit()
|
|
|
|
|
|
|
|
except Exception as e:
|
2024-06-22 08:55:26 -04:00
|
|
|
logger.error(
|
|
|
|
f"Error checking live status for {twitch_name}: {e}",
|
|
|
|
exc_info=True,
|
|
|
|
)
|
2024-06-21 20:30:41 -04:00
|
|
|
|
|
|
|
conn.close()
|
2024-06-21 22:14:26 -04:00
|
|
|
await asyncio.sleep(300) # Check every 5 minutes
|
2024-06-21 18:38:15 -04:00
|
|
|
|
|
|
|
def add_commands(self):
|
2024-06-21 20:30:41 -04:00
|
|
|
@app_commands.command(
|
|
|
|
name="follow_twitch",
|
2024-06-22 08:55:26 -04:00
|
|
|
description="Follow a Twitch channel to get live alerts",
|
2024-06-21 20:30:41 -04:00
|
|
|
)
|
2024-06-22 08:55:26 -04:00
|
|
|
async def follow_twitch(
|
|
|
|
interaction: discord.Interaction,
|
|
|
|
twitch_name: str,
|
|
|
|
channel: discord.TextChannel = None,
|
|
|
|
):
|
2024-06-21 20:30:41 -04:00
|
|
|
channel = channel or interaction.channel
|
|
|
|
conn = get_connection()
|
|
|
|
c = conn.cursor()
|
|
|
|
c.execute(
|
2024-06-22 08:55:26 -04:00
|
|
|
"INSERT OR REPLACE INTO followed_channels (twitch_name, discord_channel_id) VALUES (?, ?)",
|
|
|
|
(twitch_name, channel.id),
|
2024-06-21 20:30:41 -04:00
|
|
|
)
|
|
|
|
conn.commit()
|
|
|
|
conn.close()
|
|
|
|
await interaction.response.send_message(
|
|
|
|
embed=discord.Embed(
|
|
|
|
title="Followed Twitch Channel",
|
|
|
|
description=f"Now following {twitch_name}. Alerts will be sent to {channel.mention}.",
|
2024-06-22 08:55:26 -04:00
|
|
|
color=discord.Color.green(),
|
2024-06-21 20:30:41 -04:00
|
|
|
)
|
|
|
|
)
|
|
|
|
logger.info(f"Now following {twitch_name} for alerts in {channel.name}")
|
|
|
|
|
|
|
|
@app_commands.command(
|
2024-06-22 08:55:26 -04:00
|
|
|
name="unfollow_twitch", description="Unfollow a Twitch channel"
|
2024-06-21 20:30:41 -04:00
|
|
|
)
|
|
|
|
async def unfollow_twitch(interaction: discord.Interaction, twitch_name: str):
|
|
|
|
conn = get_connection()
|
|
|
|
c = conn.cursor()
|
|
|
|
c.execute(
|
2024-06-22 08:55:26 -04:00
|
|
|
"DELETE FROM followed_channels WHERE twitch_name = ?", (twitch_name,)
|
2024-06-21 20:30:41 -04:00
|
|
|
)
|
2024-06-22 08:55:26 -04:00
|
|
|
c.execute("DELETE FROM live_status WHERE twitch_name = ?", (twitch_name,))
|
2024-06-21 20:30:41 -04:00
|
|
|
conn.commit()
|
|
|
|
conn.close()
|
|
|
|
await interaction.response.send_message(
|
|
|
|
embed=discord.Embed(
|
|
|
|
title="Unfollowed Twitch Channel",
|
|
|
|
description=f"No longer following {twitch_name}.",
|
2024-06-22 08:55:26 -04:00
|
|
|
color=discord.Color.red(),
|
2024-06-21 20:30:41 -04:00
|
|
|
)
|
|
|
|
)
|
|
|
|
logger.info(f"No longer following {twitch_name}")
|
|
|
|
|
2024-06-21 18:38:15 -04:00
|
|
|
@app_commands.command(
|
2024-06-22 08:55:26 -04:00
|
|
|
name="twitch_live", description="Check if a Twitch streamer is live"
|
2024-06-21 18:38:15 -04:00
|
|
|
)
|
|
|
|
async def twitch_live(interaction: discord.Interaction, streamer: str):
|
|
|
|
await interaction.response.defer()
|
|
|
|
try:
|
|
|
|
logger.debug(f"Fetching user info for streamer: {streamer}")
|
|
|
|
user_info = await first(self.twitch.get_users(logins=[streamer]))
|
|
|
|
if not user_info:
|
|
|
|
await interaction.followup.send(
|
|
|
|
embed=discord.Embed(
|
|
|
|
title="Twitch Live Check",
|
|
|
|
description=f"Streamer {streamer} not found.",
|
2024-06-22 08:55:26 -04:00
|
|
|
color=discord.Color.red(),
|
2024-06-21 18:38:15 -04:00
|
|
|
)
|
|
|
|
)
|
|
|
|
logger.info(f"Streamer {streamer} not found.")
|
|
|
|
return
|
|
|
|
|
|
|
|
user_id = user_info.id
|
|
|
|
logger.debug(f"Fetching stream info for user ID: {user_id}")
|
|
|
|
streams = await first(self.twitch.get_streams(user_id=[user_id]))
|
|
|
|
if streams:
|
|
|
|
embed = discord.Embed(
|
|
|
|
title=f"{streamer} is Live!",
|
|
|
|
description=(
|
|
|
|
f"**Title:** {streams.title}\n"
|
|
|
|
f"**Game:** {streams.game_name}\n"
|
|
|
|
f"**Viewers:** {streams.viewer_count}"
|
|
|
|
),
|
2024-06-22 08:55:26 -04:00
|
|
|
color=discord.Color.green(),
|
2024-06-21 18:38:15 -04:00
|
|
|
)
|
|
|
|
embed.set_thumbnail(
|
2024-06-22 08:55:26 -04:00
|
|
|
url=streams.thumbnail_url.replace("{width}", "320").replace(
|
|
|
|
"{height}", "180"
|
|
|
|
)
|
2024-06-21 18:38:15 -04:00
|
|
|
)
|
|
|
|
await interaction.followup.send(embed=embed)
|
|
|
|
logger.info(f"Streamer {streamer} is live.")
|
|
|
|
else:
|
|
|
|
await interaction.followup.send(
|
|
|
|
embed=discord.Embed(
|
|
|
|
title=f"{streamer} is not live",
|
|
|
|
description=f"{streamer} is currently offline.",
|
2024-06-22 08:55:26 -04:00
|
|
|
color=discord.Color.red(),
|
2024-06-21 18:38:15 -04:00
|
|
|
)
|
|
|
|
)
|
|
|
|
logger.info(f"Streamer {streamer} is offline.")
|
|
|
|
except Exception as e:
|
|
|
|
logger.error(f"Error in twitch_live command: {e}", exc_info=True)
|
|
|
|
await interaction.followup.send(f"An error occurred: {e}")
|
|
|
|
|
2024-06-21 20:30:41 -04:00
|
|
|
self.bot.tree.add_command(follow_twitch)
|
|
|
|
self.bot.tree.add_command(unfollow_twitch)
|
2024-06-21 18:38:15 -04:00
|
|
|
self.bot.tree.add_command(twitch_live)
|
|
|
|
|
|
|
|
|
|
|
|
async def setup(bot):
|
|
|
|
TwitchModule(bot)
|