2024-06-21 09:26:19 -04:00
|
|
|
import logging
|
2024-06-20 23:00:20 -04:00
|
|
|
import discord
|
|
|
|
from discord import app_commands
|
|
|
|
from plexapi.server import PlexServer
|
2024-06-21 09:26:19 -04:00
|
|
|
from plexapi.client import PlexClient
|
2024-06-20 23:00:20 -04:00
|
|
|
import config
|
|
|
|
|
2024-06-21 09:26:19 -04:00
|
|
|
# Set up logging
|
|
|
|
logging.basicConfig(level=logging.DEBUG)
|
|
|
|
|
2024-06-20 23:00:20 -04:00
|
|
|
|
|
|
|
class PlexModule:
|
|
|
|
def __init__(self, bot):
|
|
|
|
self.bot = bot
|
|
|
|
self.plex = PlexServer(config.PLEX_URL, config.PLEX_TOKEN)
|
|
|
|
self.add_commands()
|
|
|
|
|
|
|
|
def add_commands(self):
|
2024-06-20 23:50:44 -04:00
|
|
|
@app_commands.command(
|
|
|
|
name="list_clients", description="List all available Plex clients"
|
|
|
|
)
|
|
|
|
async def list_clients(interaction: discord.Interaction):
|
|
|
|
await interaction.response.defer()
|
|
|
|
try:
|
2024-06-21 09:26:19 -04:00
|
|
|
logging.debug("Attempting to fetch Plex clients from /clients")
|
2024-06-20 23:50:44 -04:00
|
|
|
clients = self.plex.clients()
|
2024-06-21 09:26:19 -04:00
|
|
|
if not clients:
|
|
|
|
logging.debug("No clients found using /clients,"
|
|
|
|
"attempting /status/sessions")
|
|
|
|
sessions = self.plex.sessions()
|
|
|
|
clients = [player for session in sessions
|
|
|
|
for player in session.players]
|
|
|
|
logging.debug(f"Clients found: {clients}")
|
2024-06-20 23:50:44 -04:00
|
|
|
if not clients:
|
|
|
|
await interaction.followup.send(
|
|
|
|
embed=discord.Embed(
|
|
|
|
title="Plex Clients",
|
|
|
|
description="No clients found",
|
|
|
|
color=discord.Color.red()
|
|
|
|
)
|
|
|
|
)
|
|
|
|
return
|
|
|
|
|
|
|
|
client_info = "\n".join(
|
|
|
|
[f"{client.title} (ID: {client.machineIdentifier})" for client in clients] # noqa: E501
|
|
|
|
)
|
|
|
|
embed = discord.Embed(
|
|
|
|
title="Plex Clients",
|
|
|
|
description=client_info,
|
|
|
|
color=discord.Color.blue()
|
|
|
|
)
|
|
|
|
await interaction.followup.send(embed=embed)
|
|
|
|
except Exception as e:
|
2024-06-21 09:26:19 -04:00
|
|
|
logging.error(f"Error fetching clients: {e}")
|
2024-06-20 23:50:44 -04:00
|
|
|
await interaction.followup.send(f"An error occurred: {e}")
|
|
|
|
|
2024-06-20 23:00:20 -04:00
|
|
|
@app_commands.command(
|
|
|
|
name="list_libraries", description="List all Plex libraries"
|
|
|
|
)
|
|
|
|
async def list_libraries(interaction: discord.Interaction):
|
|
|
|
await interaction.response.defer()
|
|
|
|
try:
|
|
|
|
libraries = self.plex.library.sections()
|
|
|
|
library_names = [lib.title for lib in libraries]
|
2024-06-20 23:50:04 -04:00
|
|
|
embed = discord.Embed(
|
|
|
|
title="Plex Libraries",
|
|
|
|
description=", ".join(library_names),
|
|
|
|
color=discord.Color.blue()
|
2024-06-20 23:00:20 -04:00
|
|
|
)
|
2024-06-20 23:50:04 -04:00
|
|
|
await interaction.followup.send(embed=embed)
|
2024-06-20 23:00:20 -04:00
|
|
|
except Exception as e:
|
2024-06-21 09:26:19 -04:00
|
|
|
logging.error(f"Error fetching libraries: {e}")
|
2024-06-20 23:00:20 -04:00
|
|
|
await interaction.followup.send(f"An error occurred: {e}")
|
|
|
|
|
|
|
|
@app_commands.command(
|
|
|
|
name="search_library",
|
2024-06-20 23:50:04 -04:00
|
|
|
description="Search for a movie or TV show in a library"
|
2024-06-20 23:00:20 -04:00
|
|
|
)
|
|
|
|
async def search_library(
|
|
|
|
interaction: discord.Interaction, library: str, query: str
|
|
|
|
):
|
|
|
|
await interaction.response.defer()
|
|
|
|
try:
|
|
|
|
lib = self.plex.library.section(library)
|
|
|
|
results = lib.search(query)
|
|
|
|
if not results:
|
|
|
|
await interaction.followup.send(
|
2024-06-20 23:50:04 -04:00
|
|
|
embed=discord.Embed(
|
|
|
|
title="Search Library",
|
|
|
|
description=f"No results found for '{query}' in '{library}'", # noqa: E501
|
|
|
|
color=discord.Color.red()
|
|
|
|
)
|
2024-06-20 23:00:20 -04:00
|
|
|
)
|
|
|
|
return
|
|
|
|
|
|
|
|
result_titles = [result.title for result in results]
|
2024-06-20 23:50:04 -04:00
|
|
|
embed = discord.Embed(
|
|
|
|
title=f"Search results in {library}",
|
|
|
|
description=", ".join(result_titles),
|
|
|
|
color=discord.Color.blue()
|
2024-06-20 23:00:20 -04:00
|
|
|
)
|
2024-06-20 23:50:04 -04:00
|
|
|
await interaction.followup.send(embed=embed)
|
2024-06-20 23:00:20 -04:00
|
|
|
except Exception as e:
|
2024-06-21 09:26:19 -04:00
|
|
|
logging.error(f"Error searching library: {e}")
|
2024-06-20 23:00:20 -04:00
|
|
|
await interaction.followup.send(f"An error occurred: {e}")
|
|
|
|
|
|
|
|
@app_commands.command(
|
2024-06-20 23:50:04 -04:00
|
|
|
name="play_movie",
|
|
|
|
description="Play a movie on a specified Plex client"
|
2024-06-20 23:00:20 -04:00
|
|
|
)
|
|
|
|
async def play_movie(
|
2024-06-20 23:50:44 -04:00
|
|
|
interaction: discord.Interaction, movie_name: str,
|
|
|
|
client_name: str = None
|
2024-06-20 23:00:20 -04:00
|
|
|
):
|
|
|
|
await interaction.response.defer()
|
|
|
|
try:
|
|
|
|
client = next(
|
|
|
|
(c for c in self.plex.clients() if c.title == client_name),
|
2024-06-21 09:26:19 -04:00
|
|
|
None
|
2024-06-20 23:00:20 -04:00
|
|
|
)
|
2024-06-21 09:26:19 -04:00
|
|
|
if not client:
|
|
|
|
logging.debug("No clients found using /clients,"
|
|
|
|
"attempting /status/sessions")
|
|
|
|
sessions = self.plex.sessions()
|
|
|
|
clients = [player for session in sessions
|
|
|
|
for player in session.players]
|
|
|
|
client = clients[0] if clients else None
|
|
|
|
|
2024-06-20 23:00:20 -04:00
|
|
|
if not client:
|
|
|
|
await interaction.followup.send(
|
2024-06-20 23:50:04 -04:00
|
|
|
embed=discord.Embed(
|
|
|
|
title="Play Movie",
|
2024-06-20 23:50:44 -04:00
|
|
|
description="No clients available to play the movie", # noqa: E501
|
2024-06-20 23:50:04 -04:00
|
|
|
color=discord.Color.red()
|
|
|
|
)
|
2024-06-20 23:00:20 -04:00
|
|
|
)
|
|
|
|
return
|
|
|
|
|
2024-06-21 09:26:19 -04:00
|
|
|
# Ensure the client object is properly initialized
|
2024-06-20 23:00:20 -04:00
|
|
|
movie = self.plex.library.section('Movies').get(movie_name)
|
2024-06-21 09:26:19 -04:00
|
|
|
client_url = f"http://{client.address}:{client.port}"
|
|
|
|
plex_client = PlexClient(client_url, self.plex._token)
|
|
|
|
plex_client.playMedia(movie)
|
2024-06-20 23:50:04 -04:00
|
|
|
embed = discord.Embed(
|
|
|
|
title="Playing Movie",
|
2024-06-20 23:50:44 -04:00
|
|
|
description=f"Playing '{movie_name}' on '{client.title}'",
|
2024-06-20 23:50:04 -04:00
|
|
|
color=discord.Color.green()
|
2024-06-20 23:00:20 -04:00
|
|
|
)
|
2024-06-20 23:50:04 -04:00
|
|
|
await interaction.followup.send(embed=embed)
|
|
|
|
except Exception as e:
|
2024-06-21 09:26:19 -04:00
|
|
|
logging.error(f"Error playing movie: {e}")
|
2024-06-20 23:50:04 -04:00
|
|
|
await interaction.followup.send(f"An error occurred: {e}")
|
|
|
|
|
|
|
|
@app_commands.command(
|
|
|
|
name="play_tv_show",
|
|
|
|
description="Play a TV show on a specified Plex client"
|
|
|
|
)
|
|
|
|
async def play_tv_show(
|
2024-06-20 23:50:44 -04:00
|
|
|
interaction: discord.Interaction, library: str, show_name: str,
|
|
|
|
season: int, episode: int, client_name: str = None
|
2024-06-20 23:50:04 -04:00
|
|
|
):
|
|
|
|
await interaction.response.defer()
|
|
|
|
try:
|
|
|
|
client = next(
|
|
|
|
(c for c in self.plex.clients() if c.title == client_name),
|
2024-06-21 09:26:19 -04:00
|
|
|
None
|
2024-06-20 23:50:04 -04:00
|
|
|
)
|
2024-06-21 09:26:19 -04:00
|
|
|
if not client:
|
|
|
|
logging.debug("No clients found using /clients,"
|
|
|
|
"attempting /status/sessions")
|
|
|
|
sessions = self.plex.sessions()
|
|
|
|
clients = [player for session in sessions
|
|
|
|
for player in session.players]
|
|
|
|
client = clients[0] if clients else None
|
|
|
|
|
2024-06-20 23:50:04 -04:00
|
|
|
if not client:
|
|
|
|
await interaction.followup.send(
|
|
|
|
embed=discord.Embed(
|
|
|
|
title="Play TV Show",
|
2024-06-20 23:50:44 -04:00
|
|
|
description="No clients available to play the TV show", # noqa: E501
|
2024-06-20 23:50:04 -04:00
|
|
|
color=discord.Color.red()
|
|
|
|
)
|
|
|
|
)
|
|
|
|
return
|
|
|
|
|
2024-06-21 09:26:19 -04:00
|
|
|
# Ensure the client object is properly initialized
|
2024-06-20 23:50:04 -04:00
|
|
|
show = self.plex.library.section(library).get(show_name)
|
|
|
|
episode = show.season(season).episode(episode)
|
2024-06-21 09:26:19 -04:00
|
|
|
client_url = f"http://{client.address}:{client.port}"
|
|
|
|
plex_client = PlexClient(client_url, self.plex._token)
|
|
|
|
plex_client.playMedia(episode)
|
2024-06-20 23:50:04 -04:00
|
|
|
embed = discord.Embed(
|
|
|
|
title="Playing TV Show",
|
2024-06-20 23:50:44 -04:00
|
|
|
description=f"Playing '{show_name}' S{season}E{episode} on '{client.title}'", # noqa: E501
|
2024-06-20 23:50:04 -04:00
|
|
|
color=discord.Color.green()
|
|
|
|
)
|
|
|
|
await interaction.followup.send(embed=embed)
|
2024-06-20 23:00:20 -04:00
|
|
|
except Exception as e:
|
2024-06-21 09:26:19 -04:00
|
|
|
logging.error(f"Error playing TV show: {e}")
|
2024-06-20 23:00:20 -04:00
|
|
|
await interaction.followup.send(f"An error occurred: {e}")
|
|
|
|
|
2024-06-20 23:50:44 -04:00
|
|
|
self.bot.tree.add_command(list_clients)
|
2024-06-20 23:00:20 -04:00
|
|
|
self.bot.tree.add_command(list_libraries)
|
|
|
|
self.bot.tree.add_command(search_library)
|
|
|
|
self.bot.tree.add_command(play_movie)
|
2024-06-20 23:50:04 -04:00
|
|
|
self.bot.tree.add_command(play_tv_show)
|
2024-06-20 23:00:20 -04:00
|
|
|
|
|
|
|
|
|
|
|
async def setup(bot):
|
|
|
|
PlexModule(bot)
|