commit
15e3383163
3
.gitignore
vendored
3
.gitignore
vendored
@ -2,4 +2,5 @@ config.ini
|
||||
__pycache__
|
||||
count.db
|
||||
notes.txt
|
||||
config.yaml
|
||||
config.yaml
|
||||
track_events.log
|
@ -29,7 +29,7 @@ class MyBot(commands.Bot):
|
||||
|
||||
config.LOG.info("Loading cogs...")
|
||||
config.LOG.info(
|
||||
"YouTube support is enabled"
|
||||
"YouTube support is enabled, make sure to set a poToken"
|
||||
if config.YOUTUBE_SUPPORT
|
||||
else "YouTube support is disabled"
|
||||
)
|
||||
|
@ -6,7 +6,7 @@ from cogs.music import Music
|
||||
from typing import Literal
|
||||
|
||||
from utils.ai_recommendations import add_song_recommendations
|
||||
from utils.config import BOT_COLOR
|
||||
from utils.config import create_embed
|
||||
|
||||
|
||||
class Autoplay(commands.Cog):
|
||||
@ -19,32 +19,29 @@ class Autoplay(commands.Cog):
|
||||
async def autoplay(
|
||||
self, interaction: discord.Interaction, toggle: Literal["ON", "OFF"]
|
||||
):
|
||||
"Keep the music playing forever with music suggestions from OpenAI"
|
||||
"Keep music playing 24/7 with AI-generated song recommendations"
|
||||
if toggle == "OFF":
|
||||
self.bot.autoplay.remove(interaction.guild.id)
|
||||
|
||||
embed = discord.Embed(
|
||||
embed = create_embed(
|
||||
title="Autoplay Off",
|
||||
description=(
|
||||
"Autoplay has been turned off. I will no longer"
|
||||
" automatically add new songs to the queue based on AI"
|
||||
" recommendations."
|
||||
"Autoplay has been turned off. Song recommendations will"
|
||||
" no longer be added to the queue."
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
return await interaction.response.send_message(embed=embed)
|
||||
|
||||
# Otherwise, toggle must be "ON", so enable autoplaying
|
||||
|
||||
if interaction.guild.id in self.bot.autoplay:
|
||||
embed = discord.Embed(
|
||||
embed = create_embed(
|
||||
title="Autoplay Already Enabled",
|
||||
description=(
|
||||
"Autoplay is already enabled. If you would like to turn it"
|
||||
" off, choose the `OFF` option in the"
|
||||
" </autoplay:1228216490386391052> command."
|
||||
" off, run </autoplay:1228216490386391052> and choose the"
|
||||
" `OFF` option."
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
@ -53,21 +50,13 @@ class Autoplay(commands.Cog):
|
||||
player = self.bot.lavalink.player_manager.get(interaction.guild.id)
|
||||
|
||||
if len(player.queue) < 5:
|
||||
embed = discord.Embed(
|
||||
embed = create_embed(
|
||||
title="Not Enough Context",
|
||||
description=(
|
||||
"You must have at least 5 songs in the queue so that I can"
|
||||
" get a good understanding of what music I should continue"
|
||||
" to play. Add some more music to the queue, then try"
|
||||
"Autoplay requires at least 5 songs in the queue in order"
|
||||
" to generate recommendations. Please add more and try"
|
||||
" again."
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
)
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
@ -77,13 +66,12 @@ class Autoplay(commands.Cog):
|
||||
for song in player.queue[:10]:
|
||||
inputs[song.title] = song.author
|
||||
|
||||
embed = discord.Embed(
|
||||
embed = create_embed(
|
||||
title="Getting AI Recommendations",
|
||||
description=(
|
||||
"Attempting to generate recommendations based on the current"
|
||||
" songs in your queue. Just a moment..."
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
@ -91,35 +79,25 @@ class Autoplay(commands.Cog):
|
||||
self.bot.openai, self.bot.user, player, 5, inputs
|
||||
):
|
||||
self.bot.autoplay.append(interaction.guild.id)
|
||||
embed = discord.Embed(
|
||||
embed = create_embed(
|
||||
title=":infinity: Autoplay Enabled :infinity:",
|
||||
description=(
|
||||
"I have added a few similar songs to the queue and will"
|
||||
" continue to do so once the queue gets low again. Now"
|
||||
" just sit back and enjoy the music!\n\nEnabled by:"
|
||||
"Recommendations have been generated and added to the"
|
||||
" queue. Autoplay will automatically search for more"
|
||||
" songs whenever the queue gets low.\n\nEnabled by:"
|
||||
f" {interaction.user.mention}"
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
await interaction.edit_original_response(embed=embed)
|
||||
|
||||
else:
|
||||
embed = discord.Embed(
|
||||
embed = create_embed(
|
||||
title="Autoplay Error",
|
||||
description=(
|
||||
"Autoplay is an experimental feature, meaning sometimes it"
|
||||
" doesn't work as expected. I had an error when attempting"
|
||||
" to get similar songs for you, please try running the"
|
||||
" command again. If the issue persists, fill out a bug"
|
||||
" report with the </bug:1224840889906499626> command."
|
||||
"Unable to get AI recommendations at this time. Please try"
|
||||
" again. If issues continue, please fill out a bug report"
|
||||
" with </bug:1224840889906499626>."
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
)
|
||||
await interaction.edit_original_response(embed=embed)
|
||||
|
||||
|
@ -15,7 +15,9 @@ class BugReport(discord.ui.Modal, title="Report a bug"):
|
||||
placeholder="EX: itsmefreddy01...",
|
||||
)
|
||||
command = discord.ui.TextInput(
|
||||
label="Command with error", placeholder="EX: skip...", required=True
|
||||
label="Command with error",
|
||||
placeholder="EX: autoplay, skip...",
|
||||
required=True,
|
||||
)
|
||||
report = discord.ui.TextInput(
|
||||
label="A detailed report of the bug",
|
||||
@ -27,8 +29,8 @@ class BugReport(discord.ui.Modal, title="Report a bug"):
|
||||
|
||||
async def on_submit(self, interaction: discord.Interaction):
|
||||
await interaction.response.send_message(
|
||||
f"Thanks for your bug report. We will get back to you as soon as"
|
||||
f" possible",
|
||||
f"Thanks for your bug report. We will work on resolving the"
|
||||
f" issue as soon as possible.",
|
||||
ephemeral=True,
|
||||
)
|
||||
channel = self.bot.get_channel(BUG_CHANNEL_ID)
|
||||
|
@ -4,7 +4,7 @@ from discord import app_commands
|
||||
from discord.ext import commands
|
||||
from cogs.music import Music
|
||||
|
||||
from utils.config import BOT_COLOR
|
||||
from utils.config import create_embed
|
||||
|
||||
|
||||
class Clear(commands.Cog):
|
||||
@ -18,19 +18,13 @@ class Clear(commands.Cog):
|
||||
player = self.bot.lavalink.player_manager.get(interaction.guild.id)
|
||||
|
||||
player.queue.clear()
|
||||
embed = discord.Embed(
|
||||
|
||||
embed = create_embed(
|
||||
title="Queue Cleared",
|
||||
description=(
|
||||
"The queue has been cleared of all songs!\n\nIssued by:"
|
||||
f" {interaction.user.mention}"
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
|
@ -169,18 +169,18 @@ class Help(commands.Cog):
|
||||
embed = discord.Embed(
|
||||
title=f":musical_note: Help :musical_note:",
|
||||
description=(
|
||||
"**Check out recent news and updates about the bot with"
|
||||
" the </news:1260842465666007040> command!\n\u200b**"
|
||||
"**Check out recent updates with the"
|
||||
" </news:1260842465666007040> command!\n\u200b**"
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
|
||||
embed.add_field(
|
||||
name="**Use Me**",
|
||||
name="**Get Started**",
|
||||
value=(
|
||||
"> To get started, use the </play:1224840890368000172>"
|
||||
" command and enter the name or link to the song of your"
|
||||
" choice."
|
||||
"> Start playing music with the"
|
||||
" </play:1224840890368000172> command. Enter the name or"
|
||||
" link of the song you want to play."
|
||||
),
|
||||
inline=False,
|
||||
)
|
||||
@ -195,9 +195,9 @@ class Help(commands.Cog):
|
||||
embed.add_field(
|
||||
name="**Help for Specific Commands**",
|
||||
value=(
|
||||
"> If you want more information on how to use a specific"
|
||||
" command, use the </help:1224854217597124610> command and"
|
||||
" include the specific command."
|
||||
"> To get information on a specific command, use"
|
||||
" </help:1224854217597124610> and include the command"
|
||||
" name."
|
||||
),
|
||||
inline=False,
|
||||
)
|
||||
|
@ -4,7 +4,7 @@ from discord import app_commands
|
||||
from discord.ext import commands
|
||||
from cogs.music import Music
|
||||
|
||||
from utils.config import BOT_COLOR
|
||||
from utils.config import create_embed
|
||||
|
||||
|
||||
class Lyrics(commands.Cog):
|
||||
@ -19,19 +19,12 @@ class Lyrics(commands.Cog):
|
||||
|
||||
# If the Genius API client is not setup, send an error message
|
||||
if not self.bot.genius:
|
||||
embed = discord.Embed(
|
||||
embed = create_embed(
|
||||
title="Lyrics Feature Error",
|
||||
description=(
|
||||
"The lyrics feature is currently disabled due to errors"
|
||||
" with the Genius API."
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
)
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
@ -48,20 +41,13 @@ class Lyrics(commands.Cog):
|
||||
|
||||
# If no lyrics are found, send an error message
|
||||
if song is None:
|
||||
embed = discord.Embed(
|
||||
embed = create_embed(
|
||||
title="Lyrics Not Found",
|
||||
description=(
|
||||
"Unfortunately, I wasn't able to find any lyrics for the"
|
||||
" song that is currently playing."
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_thumbnail(url=player.current.artwork_url)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
thumbnail=player.current.artwork_url,
|
||||
)
|
||||
return await interaction.edit_original_response(embed=embed)
|
||||
|
||||
@ -73,40 +59,27 @@ class Lyrics(commands.Cog):
|
||||
|
||||
# If the lyrics are too long, send just a link to the lyrics
|
||||
if len(lyrics) > 2048:
|
||||
embed = discord.Embed(
|
||||
embed = create_embed(
|
||||
title=(
|
||||
f"Lyrics for {player.current.title} by"
|
||||
f" {player.current.author}"
|
||||
),
|
||||
description=(
|
||||
"Song lyrics are too long to display on Discord. [Click"
|
||||
f" here to view the lyrics on Genius]({song.url})."
|
||||
"The lyrics for this song are too long to display on"
|
||||
" Discord. [Click here to view the lyrics on"
|
||||
f" Genius]({song.url})."
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_thumbnail(url=player.current.artwork_url)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
thumbnail=player.current.artwork_url,
|
||||
)
|
||||
return await interaction.edit_original_response(embed=embed)
|
||||
|
||||
# If everything is successful, send the lyrics
|
||||
embed = discord.Embed(
|
||||
embed = create_embed(
|
||||
title=(
|
||||
f"Lyrics for {player.current.title} by {player.current.author}"
|
||||
),
|
||||
description=f"Provided from [Genius]({song.url})\n\n" + lyrics,
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_thumbnail(url=player.current.artwork_url)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
thumbnail=player.current.artwork_url,
|
||||
)
|
||||
await interaction.edit_original_response(embed=embed)
|
||||
|
||||
|
@ -2,13 +2,14 @@ import discord
|
||||
from discord.ext import commands
|
||||
import lavalink
|
||||
from lavalink import errors
|
||||
from discord.ext import tasks
|
||||
import os
|
||||
|
||||
from utils.config import (
|
||||
LAVALINK_HOST,
|
||||
LAVALINK_PASSWORD,
|
||||
LAVALINK_PORT,
|
||||
LOG,
|
||||
LOG_SONGS,
|
||||
)
|
||||
from utils.command_tree import CheckPlayerError
|
||||
from utils.ai_recommendations import add_song_recommendations
|
||||
@ -95,6 +96,7 @@ class LavalinkVoiceClient(discord.VoiceProtocol):
|
||||
class Music(commands.Cog):
|
||||
def __init__(self, bot):
|
||||
self.bot = bot
|
||||
self.log_file = "track_events.log"
|
||||
|
||||
async def cog_load(self):
|
||||
if not hasattr(
|
||||
@ -119,11 +121,16 @@ class Music(commands.Cog):
|
||||
return
|
||||
else:
|
||||
await node.connect()
|
||||
LOG.info(f"Connected to lavalink node {node.name}")
|
||||
|
||||
self.lavalink: lavalink.Client = self.bot.lavalink
|
||||
self.lavalink.add_event_hooks(self)
|
||||
|
||||
if os.path.exists("/.dockerenv"):
|
||||
self.log_file = "/config/track_events.log"
|
||||
|
||||
if LOG_SONGS:
|
||||
LOG.info(f"Logging track events to {self.log_file}")
|
||||
|
||||
def cog_unload(self):
|
||||
"""Cog unload handler. This removes any event hooks that were registered."""
|
||||
self.lavalink._event_hooks.clear()
|
||||
@ -241,6 +248,54 @@ class Music(commands.Cog):
|
||||
self.bot.openai, self.bot.user, event.player, 5, inputs
|
||||
)
|
||||
|
||||
@lavalink.listener(lavalink.events.NodeConnectedEvent)
|
||||
async def node_connected(self, event: lavalink.events.NodeConnectedEvent):
|
||||
LOG.info(f"Lavalink node {event.node.name} has connected")
|
||||
|
||||
@lavalink.listener(lavalink.events.NodeReadyEvent)
|
||||
async def node_ready(self, event: lavalink.events.NodeReadyEvent):
|
||||
LOG.info(f"Lavalink node {event.node.name} is ready")
|
||||
|
||||
@lavalink.listener(lavalink.events.NodeDisconnectedEvent)
|
||||
async def node_disconnected(
|
||||
self, event: lavalink.events.NodeDisconnectedEvent
|
||||
):
|
||||
LOG.error(f"Lavalink node {event.node.name} has disconnected")
|
||||
|
||||
# If we get a track load failed event (like LoadError, but for some reason that
|
||||
# wasn't the eception raised), skip the track
|
||||
@lavalink.listener(lavalink.events.TrackLoadFailedEvent)
|
||||
async def track_load_failed(self, event: lavalink.events.TrackEndEvent):
|
||||
await event.player.skip()
|
||||
|
||||
# Only log track events if enabled
|
||||
if LOG_SONGS:
|
||||
|
||||
@lavalink.listener(lavalink.events.TrackStartEvent)
|
||||
async def track_start(self, event: lavalink.events.TrackStartEvent):
|
||||
with open(self.log_file, "a") as f:
|
||||
f.write(
|
||||
f"STARTED: {event.track.title} by {event.track.author}\n"
|
||||
)
|
||||
|
||||
@lavalink.listener(lavalink.events.TrackStuckEvent)
|
||||
async def track_stuck(self, event: lavalink.events.TrackStuckEvent):
|
||||
with open(self.log_file, "a") as f:
|
||||
f.write(
|
||||
f"STUCK: {event.track.title} by {event.track.author} -"
|
||||
f" {event.track.uri}\n"
|
||||
)
|
||||
|
||||
@lavalink.listener(lavalink.events.TrackExceptionEvent)
|
||||
async def track_exception(
|
||||
self, event: lavalink.events.TrackExceptionEvent
|
||||
):
|
||||
with open(self.log_file, "a") as f:
|
||||
f.write(
|
||||
f"EXCEPTION{event.track.title} by {event.track.author} -"
|
||||
f" {event.track.uri}\n"
|
||||
)
|
||||
|
||||
|
||||
async def setup(bot):
|
||||
await bot.add_cog(Music(bot))
|
||||
|
@ -13,7 +13,7 @@ class News(commands.Cog):
|
||||
async def news(self, interaction: discord.Interaction):
|
||||
"Get recent news and updates about the bot"
|
||||
embed = discord.Embed(
|
||||
title="Recent News :newspaper2:",
|
||||
title="Recent News and Updates",
|
||||
description=(
|
||||
"View recent code commits"
|
||||
" [here](https://github.com/packetparker/guava/commits)\n\u200b"
|
||||
@ -22,30 +22,22 @@ class News(commands.Cog):
|
||||
)
|
||||
|
||||
embed.add_field(
|
||||
name="**Lyrics!**",
|
||||
name="**Limited YouTube Support**",
|
||||
value=(
|
||||
"> You can now get lyrics for the song that is currently"
|
||||
" playing. Just use the `/lyrics` command! Some songs may not"
|
||||
" have lyrics available, but the bot will do its best to find"
|
||||
" them."
|
||||
"Support for YouTube links and searches has been added. This"
|
||||
" is currently in a testing phase and is not guaranteed to"
|
||||
" work. If you encounter any issues, please submit a but"
|
||||
" report."
|
||||
),
|
||||
inline=False,
|
||||
)
|
||||
|
||||
embed.add_field(
|
||||
name="**Apple Music Support!**",
|
||||
name="**General Improvements**",
|
||||
value=(
|
||||
"> After some trial and error, you can now play music through"
|
||||
" Apple Music links. Just paste the link and the bot will do"
|
||||
" the rest!"
|
||||
),
|
||||
)
|
||||
|
||||
embed.add_field(
|
||||
name="**Autoplay Update**",
|
||||
value=(
|
||||
"> Autoplay is now much more stable after a revamp of the"
|
||||
" previous system. If you experienced short outages recently,"
|
||||
" this was due to the update. Thank you for your patience!"
|
||||
"Quality of life updates and general improvements have been"
|
||||
" made. Hopefully there are no new bugs, but please report any"
|
||||
" with </bug:1224840889906499626>."
|
||||
),
|
||||
inline=False,
|
||||
)
|
||||
|
@ -5,7 +5,7 @@ from discord.ext import commands
|
||||
from cogs.music import Music
|
||||
import lavalink
|
||||
|
||||
from utils.config import BOT_COLOR
|
||||
from utils.config import create_embed
|
||||
|
||||
|
||||
class NowPlaying(commands.Cog):
|
||||
@ -15,7 +15,7 @@ class NowPlaying(commands.Cog):
|
||||
@app_commands.command()
|
||||
@app_commands.check(Music.create_player)
|
||||
async def np(self, interaction: discord.Interaction):
|
||||
"Show what song is currently playing"
|
||||
"See what song is currently playing"
|
||||
player = self.bot.lavalink.player_manager.get(interaction.guild.id)
|
||||
|
||||
time_in = str(datetime.timedelta(milliseconds=player.position))[:-7]
|
||||
@ -25,21 +25,14 @@ class NowPlaying(commands.Cog):
|
||||
time_in = time_in[2:]
|
||||
total_duration = total_duration[3:]
|
||||
|
||||
embed = discord.Embed(
|
||||
embed = create_embed(
|
||||
title="Now Playing 🎶",
|
||||
description=(
|
||||
f"**[{player.current.title}]({player.current.uri})** by"
|
||||
f" {player.current.author}\n{f'` {time_in}/{total_duration} `'}\n\nQueued"
|
||||
f" by: {player.current.requester.mention}"
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_thumbnail(url=player.current.artwork_url)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
thumbnail=player.current.artwork_url,
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
|
46
code/cogs/owner/potoken.py
Normal file
46
code/cogs/owner/potoken.py
Normal file
@ -0,0 +1,46 @@
|
||||
from discord.ext import commands
|
||||
import requests
|
||||
|
||||
from utils.config import LAVALINK_HOST, LAVALINK_PORT, LAVALINK_PASSWORD, LOG
|
||||
|
||||
|
||||
class POToken(commands.Cog):
|
||||
def __init__(self, bot):
|
||||
self.bot = bot
|
||||
|
||||
@commands.command()
|
||||
@commands.dm_only()
|
||||
@commands.is_owner()
|
||||
async def potoken(self, ctx, token: str = None, visitor_data: str = None):
|
||||
"""Update the poToken for lavalink youtube support."""
|
||||
if not token or not visitor_data:
|
||||
return await ctx.send(
|
||||
"Missing token and/or visitor data. Format as"
|
||||
f" `{self.bot.command_prefix}potoken <token> <visitor"
|
||||
" data>`\n\nTo generate a poToken, see"
|
||||
" [this](https://github.com/iv-org/youtube-trusted-session-generator)"
|
||||
)
|
||||
|
||||
url = f"http://{LAVALINK_HOST}:{LAVALINK_PORT}/youtube"
|
||||
request = requests.post(
|
||||
url,
|
||||
json={"poToken": token, "visitorData": visitor_data},
|
||||
headers={"Authorization": LAVALINK_PASSWORD},
|
||||
)
|
||||
|
||||
if request.status_code != 204:
|
||||
LOG.error("Error updating poToken")
|
||||
return await ctx.send(
|
||||
"Error setting poToken, YouTube source plugin is likely not"
|
||||
" enabled. Read the Guava docs and look"
|
||||
" [here](https://github.com/lavalink-devs/youtube-source)."
|
||||
)
|
||||
|
||||
LOG.info("poToken successfully updated")
|
||||
await ctx.send(
|
||||
"Successfully posted the token and visitor data to lavalink."
|
||||
)
|
||||
|
||||
|
||||
async def setup(bot):
|
||||
await bot.add_cog(POToken(bot))
|
@ -1,10 +1,11 @@
|
||||
import discord
|
||||
import datetime
|
||||
from typing import Literal
|
||||
from discord import app_commands
|
||||
from discord.ext import commands
|
||||
from cogs.music import Music
|
||||
|
||||
from utils.config import BOT_COLOR
|
||||
from utils.config import create_embed
|
||||
|
||||
|
||||
class Pause(commands.Cog):
|
||||
@ -12,28 +13,39 @@ class Pause(commands.Cog):
|
||||
self.bot = bot
|
||||
|
||||
@app_commands.command()
|
||||
@app_commands.describe(pause="TRUE to pause, FALSE to unpause")
|
||||
@app_commands.check(Music.create_player)
|
||||
async def pause(self, interaction: discord.Interaction):
|
||||
"Pauses the song that is currently playing"
|
||||
async def pause(
|
||||
self, interaction: discord.Interaction, pause: Literal["TRUE", "FALSE"]
|
||||
):
|
||||
"Pause or unpause the current song"
|
||||
player = self.bot.lavalink.player_manager.get(interaction.guild.id)
|
||||
|
||||
await player.set_pause(pause=True)
|
||||
embed = discord.Embed(
|
||||
title=f"Music Now Paused ⏸️",
|
||||
description=(
|
||||
f"**[{player.current.title}]({player.current.uri})**\n\nQueued"
|
||||
f" by: {player.current.requester.mention}"
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_thumbnail(url=player.current.artwork_url)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
if pause:
|
||||
await player.set_pause(pause=True)
|
||||
embed = create_embed(
|
||||
title=f"Music Paused ⏸️",
|
||||
description=(
|
||||
f"**[{player.current.title}]({player.current.uri})** by"
|
||||
f" {player.current.author}\n\nQueued by:"
|
||||
f" {player.current.requester.mention}"
|
||||
),
|
||||
thumbnail=player.current.artwork_url,
|
||||
)
|
||||
+ " UTC"
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
return await interaction.response.send_message(embed=embed)
|
||||
|
||||
else:
|
||||
await player.set_pause(pause=False)
|
||||
embed = create_embed(
|
||||
title=f"Music Unpaused ▶️",
|
||||
description=(
|
||||
f"**[{player.current.title}]({player.current.uri})** by"
|
||||
f" {player.current.author}\n\nQueued by:"
|
||||
f" {player.current.requester.mention}"
|
||||
),
|
||||
thumbnail=player.current.artwork_url,
|
||||
)
|
||||
return await interaction.response.send_message(embed=embed)
|
||||
|
||||
|
||||
async def setup(bot):
|
||||
|
@ -4,11 +4,14 @@ from discord import app_commands
|
||||
from discord.ext import commands
|
||||
from lavalink import LoadType
|
||||
import re
|
||||
import requests
|
||||
|
||||
from cogs.music import Music, LavalinkVoiceClient
|
||||
from utils.config import BOT_COLOR, YOUTUBE_SUPPORT
|
||||
from utils.custom_sources import SpotifySource, AppleSource
|
||||
from utils.config import YOUTUBE_SUPPORT, create_embed
|
||||
from utils.custom_sources import (
|
||||
LoadError,
|
||||
CustomAudioTrack,
|
||||
)
|
||||
from utils.source_helpers.parse import parse_custom_source
|
||||
|
||||
|
||||
url_rx = re.compile(r"https?://(?:www\.)?.+")
|
||||
@ -28,7 +31,7 @@ class Play(commands.Cog):
|
||||
# Notify users that YouTube links are not allowed if YouTube support is disabled
|
||||
if "youtube.com" in query or "youtu.be" in query:
|
||||
if not YOUTUBE_SUPPORT:
|
||||
embed = discord.Embed(
|
||||
embed = create_embed(
|
||||
title="YouTube Not Supported",
|
||||
description=(
|
||||
"Unfortunately, YouTube does not allow bots to stream"
|
||||
@ -37,372 +40,134 @@ class Play(commands.Cog):
|
||||
" song and I will automatically find it on a supported"
|
||||
" platform."
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
)
|
||||
|
||||
###
|
||||
### APPLE MUSIC links, perform API requests and load all tracks from the playlist/album/track
|
||||
###
|
||||
|
||||
# Check for custom sources (Apple Music/Spotify)
|
||||
if "music.apple.com" in query:
|
||||
if not self.bot.apple_headers:
|
||||
embed = discord.Embed(
|
||||
title="Apple Music Error",
|
||||
description=(
|
||||
"Apple Music support seems to be broken at the moment."
|
||||
" Please try again and fill out a bug report with"
|
||||
" </bug:1224840889906499626> if this continues to"
|
||||
" happen."
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
)
|
||||
|
||||
embed = discord.Embed(color=BOT_COLOR)
|
||||
|
||||
if "/playlist/" in query and "?i=" not in query:
|
||||
playlist_id = query.split("/playlist/")[1].split("/")[1]
|
||||
# Get all of the tracks in the playlist (limit at 250)
|
||||
playlist_url = f"https://api.music.apple.com/v1/catalog/us/playlists/{playlist_id}/tracks?limit=100"
|
||||
response = requests.get(
|
||||
playlist_url, headers=self.bot.apple_headers
|
||||
)
|
||||
if response.status_code == 200:
|
||||
playlist = response.json()
|
||||
# Get the general playlist info (name, artwork)
|
||||
playlist_info_url = f"https://api.music.apple.com/v1/catalog/us/playlists/{playlist_id}"
|
||||
playlist_info = requests.get(
|
||||
playlist_info_url, headers=self.bot.apple_headers
|
||||
)
|
||||
playlist_info = playlist_info.json()
|
||||
try:
|
||||
artwork_url = playlist_info["data"][0]["attributes"][
|
||||
"artwork"
|
||||
]["url"].replace("{w}x{h}", "300x300")
|
||||
except KeyError:
|
||||
artwork_url = None
|
||||
|
||||
embed.title = "Playlist Queued"
|
||||
embed.description = (
|
||||
f"**{playlist_info['data'][0]['attributes']['name']}**\n`"
|
||||
f" {len(playlist['data'])} ` tracks\n\nQueued by:"
|
||||
f" {interaction.user.mention}"
|
||||
)
|
||||
embed.set_thumbnail(url=artwork_url)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(
|
||||
datetime.timezone.utc
|
||||
).strftime("%Y-%m-%d %H:%M:%S")
|
||||
+ " UTC"
|
||||
)
|
||||
# Add small alert if the playlist is the max size
|
||||
if len(playlist["data"]) == 100:
|
||||
embed.description += (
|
||||
"\n\n*This playlist is longer than the 100 song"
|
||||
" maximum. Only the first 100 songs will be"
|
||||
" queued.*"
|
||||
)
|
||||
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
tracks = await AppleSource.load_playlist(
|
||||
self, interaction.user, playlist
|
||||
)
|
||||
for track in tracks["tracks"]:
|
||||
player.add(requester=interaction.user, track=track)
|
||||
|
||||
# If there is an album, not a specific song within the album
|
||||
if "/album/" in query and "?i=" not in query:
|
||||
album_id = query.split("/album/")[1].split("/")[1]
|
||||
album_url = f"https://api.music.apple.com/v1/catalog/us/albums/{album_id}"
|
||||
response = requests.get(
|
||||
album_url, headers=self.bot.apple_headers
|
||||
)
|
||||
if response.status_code == 200:
|
||||
album = response.json()
|
||||
|
||||
embed.title = "Album Queued"
|
||||
embed.description = (
|
||||
f"**{album['data'][0]['attributes']['name']}** by"
|
||||
f" **{album['data'][0]['attributes']['artistName']}**\n`"
|
||||
f" {len(album['data'][0]['relationships']['tracks']['data'])} `"
|
||||
f" tracks\n\nQueued by: {interaction.user.mention}"
|
||||
)
|
||||
embed.set_thumbnail(
|
||||
url=album["data"][0]["attributes"]["artwork"][
|
||||
"url"
|
||||
].replace("{w}x{h}", "300x300")
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(
|
||||
datetime.timezone.utc
|
||||
).strftime("%Y-%m-%d %H:%M:%S")
|
||||
+ " UTC"
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
tracks = await AppleSource.load_album(
|
||||
self, interaction.user, album
|
||||
)
|
||||
for track in tracks["tracks"]:
|
||||
player.add(requester=interaction.user, track=track)
|
||||
|
||||
# If there is a specific song
|
||||
if "/album/" in query and "?i=" in query:
|
||||
song_id = query.split("/album/")[1].split("?i=")[1]
|
||||
song_url = f"https://api.music.apple.com/v1/catalog/us/songs/{song_id}"
|
||||
response = requests.get(
|
||||
song_url, headers=self.bot.apple_headers
|
||||
)
|
||||
if response.status_code == 200:
|
||||
song = response.json()
|
||||
|
||||
embed.title = "Song Queued"
|
||||
embed.description = (
|
||||
f"**{song['data'][0]['attributes']['name']}** by"
|
||||
f" **{song['data'][0]['attributes']['artistName']}**\n\nQueued"
|
||||
f" by: {interaction.user.mention}"
|
||||
)
|
||||
embed.set_thumbnail(
|
||||
url=song["data"][0]["attributes"]["artwork"][
|
||||
"url"
|
||||
].replace("{w}x{h}", "300x300")
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(
|
||||
datetime.timezone.utc
|
||||
).strftime("%Y-%m-%d %H:%M:%S")
|
||||
+ " UTC"
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
results = await AppleSource.load_item(
|
||||
self, interaction.user, song
|
||||
)
|
||||
player.add(
|
||||
requester=interaction.user, track=results.tracks[0]
|
||||
)
|
||||
|
||||
###
|
||||
### SPOTIFY links, perform API requests and load all tracks from the playlist/album/track
|
||||
###
|
||||
results, embed = await parse_custom_source(
|
||||
self, "apple", query, interaction.user
|
||||
)
|
||||
|
||||
elif "open.spotify.com" in query:
|
||||
if not self.bot.spotify_headers:
|
||||
embed = discord.Embed(
|
||||
title="Spotify Error",
|
||||
description=(
|
||||
"Spotify support seems to be broken at the moment."
|
||||
" Please try again and fill out a bug report with"
|
||||
" </bug:1224840889906499626> if this continues to"
|
||||
" happen."
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
)
|
||||
|
||||
embed = discord.Embed(color=BOT_COLOR)
|
||||
|
||||
if "open.spotify.com/playlist" in query:
|
||||
playlist_id = query.split("playlist/")[1].split("?si=")[0]
|
||||
playlist_url = (
|
||||
f"https://api.spotify.com/v1/playlists/{playlist_id}"
|
||||
)
|
||||
response = requests.get(
|
||||
playlist_url, headers=self.bot.spotify_headers
|
||||
)
|
||||
if response.status_code == 200:
|
||||
playlist = response.json()
|
||||
|
||||
embed.title = "Playlist Queued"
|
||||
embed.description = (
|
||||
f"**{playlist['name']}** from"
|
||||
f" **{playlist['owner']['display_name']}**\n`"
|
||||
f" {len(playlist['tracks']['items'])} `"
|
||||
f" tracks\n\nQueued by: {interaction.user.mention}"
|
||||
)
|
||||
embed.set_thumbnail(url=playlist["images"][0]["url"])
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(
|
||||
datetime.timezone.utc
|
||||
).strftime("%Y-%m-%d %H:%M:%S")
|
||||
+ " UTC"
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
tracks = await SpotifySource.load_playlist(
|
||||
self, interaction.user, playlist
|
||||
)
|
||||
for track in tracks["tracks"]:
|
||||
player.add(requester=interaction.user, track=track)
|
||||
|
||||
if "open.spotify.com/album" in query:
|
||||
album_id = query.split("album/")[1]
|
||||
album_url = f"https://api.spotify.com/v1/albums/{album_id}"
|
||||
response = requests.get(
|
||||
album_url, headers=self.bot.spotify_headers
|
||||
)
|
||||
if response.status_code == 200:
|
||||
album = response.json()
|
||||
|
||||
embed.title = "Album Queued"
|
||||
embed.description = (
|
||||
f"**{album['name']}** by"
|
||||
f" **{album['artists'][0]['name']}**\n`"
|
||||
f" {len(album['tracks']['items'])} ` tracks\n\nQueued"
|
||||
f" by: {interaction.user.mention}"
|
||||
)
|
||||
embed.set_thumbnail(url=album["images"][0]["url"])
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(
|
||||
datetime.timezone.utc
|
||||
).strftime("%Y-%m-%d %H:%M:%S")
|
||||
+ " UTC"
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
tracks = await SpotifySource.load_album(
|
||||
self, interaction.user, album
|
||||
)
|
||||
for track in tracks["tracks"]:
|
||||
player.add(requester=interaction.user, track=track)
|
||||
|
||||
if "open.spotify.com/track" in query:
|
||||
track_id = query.split("track/")[1]
|
||||
track_url = f"https://api.spotify.com/v1/tracks/{track_id}"
|
||||
response = requests.get(
|
||||
track_url, headers=self.bot.spotify_headers
|
||||
)
|
||||
if response.status_code == 200:
|
||||
track = response.json()
|
||||
|
||||
embed.title = "Track Queued"
|
||||
embed.description = (
|
||||
f"**{track['name']}** by"
|
||||
f" **{track['artists'][0]['name']}**\n\nQueued by:"
|
||||
f" {interaction.user.mention}"
|
||||
)
|
||||
embed.set_thumbnail(url=track["album"]["images"][0]["url"])
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(
|
||||
datetime.timezone.utc
|
||||
).strftime("%Y-%m-%d %H:%M:%S")
|
||||
+ " UTC"
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
results = await SpotifySource.load_item(
|
||||
self, interaction.user, track
|
||||
)
|
||||
player.add(
|
||||
requester=interaction.user, track=results.tracks[0]
|
||||
)
|
||||
|
||||
if "open.spotify.com/artists" in query:
|
||||
embed.title = "Artists Cannot Be Played"
|
||||
embed.description = (
|
||||
"I cannot play just artists, you must provide a"
|
||||
" song/album/playlist. Please try again."
|
||||
)
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
)
|
||||
|
||||
###
|
||||
### For anything else, use default Lavalink providers to search the query
|
||||
###
|
||||
results, embed = await parse_custom_source(
|
||||
self, "spotify", query, interaction.user
|
||||
)
|
||||
|
||||
# For anything else, use default Lavalink providers to search the query
|
||||
else:
|
||||
# If the query is not a URL, begin searching
|
||||
if not url_rx.match(query):
|
||||
dzsearch = f"dzsearch:{query}"
|
||||
results = await player.node.get_tracks(dzsearch)
|
||||
|
||||
# If Deezer returned nothing
|
||||
if not results.tracks or results.load_type in (
|
||||
LoadType.EMPTY,
|
||||
LoadType.ERROR,
|
||||
):
|
||||
ytmsearch = f"ytmsearch:{query}"
|
||||
results = await player.node.get_tracks(ytmsearch)
|
||||
|
||||
if not results.tracks or results.load_type in (
|
||||
LoadType.EMPTY,
|
||||
LoadType.ERROR,
|
||||
):
|
||||
ytsearch = f"ytsearch:{query}"
|
||||
results = await player.node.get_tracks(ytsearch)
|
||||
if YOUTUBE_SUPPORT:
|
||||
ytmsearch = f"ytmsearch:{query}"
|
||||
results = await player.node.get_tracks(ytmsearch)
|
||||
# If YouTube Music returned nothing
|
||||
if not results.tracks or results.load_type in (
|
||||
LoadType.EMPTY,
|
||||
LoadType.ERROR,
|
||||
):
|
||||
# Final search attempt with YouTube
|
||||
ytsearch = f"ytsearch:{query} audio"
|
||||
results = await player.node.get_tracks(ytsearch)
|
||||
else:
|
||||
results = await player.node.get_tracks(query)
|
||||
|
||||
embed = discord.Embed(color=BOT_COLOR)
|
||||
|
||||
# If there are no results found, set results/embed to None, handled further down
|
||||
if not results.tracks or results.load_type in (
|
||||
LoadType.EMPTY,
|
||||
LoadType.ERROR,
|
||||
):
|
||||
embed.title = "Nothing Found"
|
||||
embed.description = (
|
||||
"Nothing for that query could be found. If this continues"
|
||||
" happening for other songs, please run"
|
||||
" </bug:1224840889906499626> to let the developer know."
|
||||
)
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
)
|
||||
results, embed = None, None
|
||||
|
||||
elif results.load_type == LoadType.PLAYLIST:
|
||||
tracks = results.tracks
|
||||
|
||||
for track in tracks:
|
||||
player.add(requester=interaction.user, track=track)
|
||||
|
||||
embed.title = "Songs Queued!"
|
||||
embed.description = (
|
||||
f"**{results.playlist_info.name}**\n` {len(tracks)} `"
|
||||
f" tracks\n\nQueued by: {interaction.user.mention}"
|
||||
# Create the embed if the results are a playlist
|
||||
if results.load_type == LoadType.PLAYLIST:
|
||||
embed = create_embed(
|
||||
title="Songs Queued",
|
||||
description=(
|
||||
f"**{results.playlist_info.name}**\n"
|
||||
f"` {len(results.tracks)} ` tracks\n\n"
|
||||
f"Queued by: {interaction.user.mention}"
|
||||
),
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
# Otherwise, the result is just a single track, create that embed
|
||||
else:
|
||||
track = results.tracks[0]
|
||||
# Remove all but first track (most relevant result)
|
||||
results.tracks = results.tracks[:1]
|
||||
embed = create_embed(
|
||||
title="Song Queued",
|
||||
description=(
|
||||
f"**{results.tracks[0].title}** by"
|
||||
f" **{results.tracks[0].author}**\n\nQueued by:"
|
||||
f" {interaction.user.mention}"
|
||||
),
|
||||
thumbnail=results.tracks[0].artwork_url,
|
||||
)
|
||||
|
||||
# If there are no results, and no embed
|
||||
if not results and not embed:
|
||||
embed = create_embed(
|
||||
title="Nothing Found",
|
||||
description=(
|
||||
"No songs were found for that query. Please try again and"
|
||||
" fill out a bug report with </bug:1224840889906499626> if"
|
||||
" this continues to happen."
|
||||
),
|
||||
)
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
)
|
||||
# If there are no results, but there is an embed (error msg)
|
||||
elif embed and not results:
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
)
|
||||
# If there are results, add them to the player
|
||||
else:
|
||||
for track in results.tracks:
|
||||
player.add(requester=interaction.user, track=track)
|
||||
|
||||
embed.title = "Track Queued"
|
||||
embed.description = (
|
||||
f"**{track.title}** by **{track.author}**\n\nQueued by:"
|
||||
f" {interaction.user.mention}"
|
||||
)
|
||||
embed.set_thumbnail(url=track.artwork_url)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
# If the track is CustomAudioTrack (Apple Music/Spotify)
|
||||
if type(results.tracks[0]) == CustomAudioTrack:
|
||||
# Attempt to load an actual track from a provider
|
||||
try:
|
||||
await results.tracks[0].load(player.node)
|
||||
# If it fails, remove it from the queue and alert the user
|
||||
except LoadError as e:
|
||||
player.queue.remove(results.tracks[0])
|
||||
embed = create_embed(
|
||||
title="Load Error",
|
||||
description=(
|
||||
"Apple Music and Spotify do not allow direct"
|
||||
" playing from their websites, and I was unable to"
|
||||
" load a track on a supported platform. Please try"
|
||||
" again."
|
||||
),
|
||||
)
|
||||
+ " UTC"
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
)
|
||||
|
||||
# Join the voice channel if not already connected
|
||||
if not interaction.guild.voice_client:
|
||||
await interaction.user.voice.channel.connect(
|
||||
cls=LavalinkVoiceClient, self_deaf=True
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
# Only join the voice channel now, so that the bot doesn't join if nothing is found
|
||||
if not interaction.guild.voice_client:
|
||||
await interaction.user.voice.channel.connect(
|
||||
cls=LavalinkVoiceClient, self_deaf=True
|
||||
)
|
||||
# Only call player.play if it is not already playing, otherwise it will
|
||||
# effectively skip the current track
|
||||
if not player.is_playing:
|
||||
await player.play()
|
||||
|
||||
# We don't want to call .play() if the player is playing as that will
|
||||
# effectively skip the current track
|
||||
if not player.is_playing:
|
||||
await player.play()
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
|
||||
async def setup(bot):
|
||||
|
@ -6,7 +6,37 @@ from cogs.music import Music
|
||||
import math
|
||||
import lavalink
|
||||
|
||||
from utils.config import BOT_COLOR
|
||||
from utils.config import create_embed
|
||||
|
||||
"""
|
||||
Create an embed for the queue given the current queue and desired page number/pages
|
||||
"""
|
||||
|
||||
|
||||
def create_queue_embed(queue: list, page: int, pages: int):
|
||||
items_per_page = 10
|
||||
start = (page - 1) * items_per_page
|
||||
end = start + items_per_page
|
||||
|
||||
queue_list = ""
|
||||
for index, track in enumerate(queue[start:end], start=start):
|
||||
# Change ms duration to hour, min, sec in the format of 00:00:00
|
||||
track_duration = lavalink.utils.format_time(track.duration)
|
||||
# If the track is less than an hour, remove the hour from the duration
|
||||
if track_duration.split(":")[0] == "00":
|
||||
track_duration = track_duration[3:]
|
||||
|
||||
queue_list += (
|
||||
f"`{index+1}. ` [{track.title}]({track.uri}) -"
|
||||
f" {track.author} `({track_duration})`\n"
|
||||
)
|
||||
|
||||
embed = create_embed(
|
||||
title=f"Current Song Queue",
|
||||
description=f"**{len(queue)} total tracks**\n\n{queue_list}",
|
||||
footer=f"Page {page}/{pages}",
|
||||
)
|
||||
return embed
|
||||
|
||||
|
||||
class Queue(commands.Cog):
|
||||
@ -22,54 +52,68 @@ class Queue(commands.Cog):
|
||||
"See the current queue of songs"
|
||||
player = self.bot.lavalink.player_manager.get(interaction.guild.id)
|
||||
|
||||
pages = math.ceil(len(player.queue) / 10)
|
||||
# Force page to 1 if an invalid page is provided
|
||||
if page < 1 or page > pages:
|
||||
page = 1
|
||||
|
||||
if not player.queue:
|
||||
embed = discord.Embed(
|
||||
embed = create_embed(
|
||||
title="Nothing Queued",
|
||||
description=(
|
||||
"Nothing is currently in the queue, add a song with the"
|
||||
" </play:1224840890368000172> command."
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
)
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
)
|
||||
|
||||
items_per_page = 10
|
||||
pages = math.ceil(len(player.queue) / items_per_page)
|
||||
|
||||
start = (page - 1) * items_per_page
|
||||
end = start + items_per_page
|
||||
|
||||
queue_list = ""
|
||||
for index, track in enumerate(player.queue[start:end], start=start):
|
||||
# Change ms duration to hour, min, sec in the format of 00:00:00
|
||||
track_duration = lavalink.utils.format_time(track.duration)
|
||||
# If the track is less than an hour, remove the hour from the duration
|
||||
if track_duration.split(":")[0] == "00":
|
||||
track_duration = track_duration[3:]
|
||||
|
||||
queue_list += (
|
||||
f"`{index+1}. ` [{track.title}]({track.uri}) -"
|
||||
f" {track.author} `({track_duration})`\n"
|
||||
)
|
||||
|
||||
embed = discord.Embed(
|
||||
title=f"Queue for {interaction.guild.name}",
|
||||
description=(
|
||||
f"**{len(player.queue)} tracks total**\n\n{queue_list}"
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_footer(text=f"Viewing page {page}/{pages}")
|
||||
await interaction.response.send_message(embed=embed)
|
||||
embed = create_queue_embed(player.queue, page, pages)
|
||||
view = QueueView(page, pages, player.queue)
|
||||
await interaction.response.send_message(embed=embed, view=view)
|
||||
|
||||
|
||||
async def setup(bot):
|
||||
await bot.add_cog(Queue(bot))
|
||||
|
||||
|
||||
class QueueView(discord.ui.View):
|
||||
def __init__(self, page: int, pages: int, queue: list):
|
||||
super().__init__()
|
||||
self.page = page
|
||||
self.pages = pages
|
||||
self.queue = queue
|
||||
# Create the previous and next buttons
|
||||
self.previous_button = discord.ui.Button(
|
||||
label="Previous", style=discord.ButtonStyle.gray
|
||||
)
|
||||
# Determine if the button should be disabled, add callback, add to view
|
||||
self.previous_button.disabled = self.page <= 1
|
||||
self.previous_button.callback = self.previous_page
|
||||
self.add_item(self.previous_button)
|
||||
|
||||
self.next_button = discord.ui.Button(
|
||||
label="Next", style=discord.ButtonStyle.gray
|
||||
)
|
||||
self.next_button.disabled = self.page >= self.pages
|
||||
self.next_button.callback = self.next_page
|
||||
self.add_item(self.next_button)
|
||||
|
||||
async def previous_page(self, interaction: discord.Interaction):
|
||||
# Decrement the page number, recreate the embed, determine if the
|
||||
# button should be disabled, and update the message
|
||||
self.page -= 1
|
||||
embed = create_queue_embed(self.queue, self.page, self.pages)
|
||||
self.previous_button.disabled = self.page <= 1
|
||||
self.next_button.disabled = self.page >= self.pages
|
||||
await interaction.response.edit_message(embed=embed, view=self)
|
||||
|
||||
async def next_page(self, interaction: discord.Interaction):
|
||||
# Increment the page number, recreate the embed, determine if the
|
||||
# button should be disabled, and update the message
|
||||
self.page += 1
|
||||
embed = create_queue_embed(self.queue, self.page, self.pages)
|
||||
self.previous_button.disabled = self.page <= 1
|
||||
self.next_button.disabled = self.page >= self.pages
|
||||
await interaction.response.edit_message(embed=embed, view=self)
|
||||
|
@ -4,7 +4,7 @@ from discord import app_commands
|
||||
from discord.ext import commands
|
||||
from cogs.music import Music
|
||||
|
||||
from utils.config import BOT_COLOR
|
||||
from utils.config import create_embed
|
||||
|
||||
|
||||
class Remove(commands.Cog):
|
||||
@ -19,27 +19,24 @@ class Remove(commands.Cog):
|
||||
player = self.bot.lavalink.player_manager.get(interaction.guild.id)
|
||||
|
||||
if not player.queue:
|
||||
embed = discord.Embed(
|
||||
embed = create_embed(
|
||||
title="Nothing Queued",
|
||||
description=(
|
||||
"Nothing is currently in the queue, so there is nothing"
|
||||
" for me to remove."
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
description="There are no songs in the queue to remove.",
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
)
|
||||
return await interaction.response.send_message(embed=embed)
|
||||
|
||||
if number > len(player.queue) or number < 1:
|
||||
embed = create_embed(
|
||||
title="Number Out of Range",
|
||||
description=(
|
||||
"The number you entered is outside of the allowed range."
|
||||
" Please try again with a valid song number."
|
||||
),
|
||||
)
|
||||
return await interaction.response.send_message(
|
||||
"The number entered is not a number within the queue - please"
|
||||
" try again!",
|
||||
ephemeral=True,
|
||||
embed=embed, ephemeral=True
|
||||
)
|
||||
|
||||
index = number - 1
|
||||
@ -48,21 +45,13 @@ class Remove(commands.Cog):
|
||||
removed_artwork = player.queue[index].artwork_url
|
||||
player.queue.pop(index)
|
||||
|
||||
embed = discord.Embed(
|
||||
embed = create_embed(
|
||||
title="Song Removed from Queue",
|
||||
description=(
|
||||
"**Song Removed -"
|
||||
f" [{removed_title}]({removed_url})**\n\nIssued by:"
|
||||
f" {interaction.user.mention}"
|
||||
f"**[{removed_title}]({removed_url})** has been unqueued.\n\n"
|
||||
f"Issued by: {interaction.user.mention}"
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_thumbnail(url=removed_artwork)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
thumbnail=removed_artwork,
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
|
@ -4,7 +4,7 @@ from discord import app_commands
|
||||
from discord.ext import commands
|
||||
from cogs.music import Music
|
||||
|
||||
from utils.config import BOT_COLOR
|
||||
from utils.config import create_embed
|
||||
|
||||
|
||||
class Repeat(commands.GroupCog, name="repeat"):
|
||||
@ -17,34 +17,11 @@ class Repeat(commands.GroupCog, name="repeat"):
|
||||
"Turn song/queue repetition off"
|
||||
player = self.bot.lavalink.player_manager.get(interaction.guild.id)
|
||||
|
||||
if player.loop == 0:
|
||||
embed = discord.Embed(
|
||||
title=f"Repeating Already Off",
|
||||
description=f"Music repetition is already turned off.",
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
)
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
)
|
||||
|
||||
player.loop = 0
|
||||
|
||||
embed = discord.Embed(
|
||||
title=f"Repeating Off",
|
||||
description=f"Music will no longer be repeated.",
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
embed = create_embed(
|
||||
title="Repeating Off",
|
||||
description="Music will not be repeated.",
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
@ -54,37 +31,14 @@ class Repeat(commands.GroupCog, name="repeat"):
|
||||
"Forever repeat that song that is currently playing"
|
||||
player = self.bot.lavalink.player_manager.get(interaction.guild.id)
|
||||
|
||||
if player.loop == 1:
|
||||
embed = discord.Embed(
|
||||
title=f"Repeating Already On",
|
||||
description=f"The current song is already being repeated.",
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
)
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
)
|
||||
|
||||
player.loop = 1
|
||||
|
||||
embed = discord.Embed(
|
||||
title=f"Repeating Current Song 🔁",
|
||||
embed = create_embed(
|
||||
title="Repeating Current Song 🔁",
|
||||
description=(
|
||||
f"The song that is currently playing will be repeated until"
|
||||
f" the </repeat off:1224840891395608737> command is run"
|
||||
"The song that is currently playing will be repeated until"
|
||||
" the </repeat off:1224840891395608737> command is run"
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
@ -94,37 +48,14 @@ class Repeat(commands.GroupCog, name="repeat"):
|
||||
"Continuously repeat the queue once it reaches the end"
|
||||
player = self.bot.lavalink.player_manager.get(interaction.guild.id)
|
||||
|
||||
if player.loop == 2:
|
||||
embed = discord.Embed(
|
||||
title=f"Repeating Already On",
|
||||
description=f"The queue is already being repeated.",
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
)
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
)
|
||||
|
||||
player.loop = 2
|
||||
|
||||
embed = discord.Embed(
|
||||
title=f"Repeating Current Song 🔂",
|
||||
embed = create_embed(
|
||||
title="Repeating Queue 🔂",
|
||||
description=(
|
||||
f"All songs in the queue will continue to repeat until the"
|
||||
f" </repeat off:1224840891395608737> command is run."
|
||||
"The queue will continuously repeat until the"
|
||||
" </repeat off:1224840891395608737> command is run."
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
|
@ -1,40 +0,0 @@
|
||||
import discord
|
||||
import datetime
|
||||
from discord import app_commands
|
||||
from discord.ext import commands
|
||||
from cogs.music import Music
|
||||
|
||||
from utils.config import BOT_COLOR
|
||||
|
||||
|
||||
class Resume(commands.Cog):
|
||||
def __init__(self, bot):
|
||||
self.bot = bot
|
||||
|
||||
@app_commands.command()
|
||||
@app_commands.check(Music.create_player)
|
||||
async def resume(self, interaction: discord.Interaction):
|
||||
"Resumes the paused song"
|
||||
player = self.bot.lavalink.player_manager.get(interaction.guild.id)
|
||||
|
||||
await player.set_pause(pause=False)
|
||||
embed = discord.Embed(
|
||||
title=f"Music Now Resumed ⏯️",
|
||||
description=(
|
||||
f"**[{player.current.title}]({player.current.uri})**\n\nQueued"
|
||||
f" by: {player.current.requester.mention}"
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_thumbnail(url=player.current.artwork_url)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
|
||||
async def setup(bot):
|
||||
await bot.add_cog(Resume(bot))
|
@ -4,7 +4,7 @@ from discord import app_commands
|
||||
from discord.ext import commands
|
||||
from cogs.music import Music
|
||||
|
||||
from utils.config import BOT_COLOR
|
||||
from utils.config import create_embed
|
||||
|
||||
|
||||
class Shuffle(commands.GroupCog, name="shuffle"):
|
||||
@ -19,16 +19,9 @@ class Shuffle(commands.GroupCog, name="shuffle"):
|
||||
|
||||
player.shuffle = True
|
||||
|
||||
embed = discord.Embed(
|
||||
title=f"Shuffle Enabled 🔀",
|
||||
description=f"All music will now be shuffled.",
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
embed = create_embed(
|
||||
title="Shuffle Enabled 🔀",
|
||||
description="All music will now be shuffled.",
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
@ -40,16 +33,9 @@ class Shuffle(commands.GroupCog, name="shuffle"):
|
||||
|
||||
player.shuffle = False
|
||||
|
||||
embed = discord.Embed(
|
||||
title=f"Disabled 🔀",
|
||||
description=f"Music will no longer be shuffled.",
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
embed = create_embed(
|
||||
title="Shuffle Disabled 🔀",
|
||||
description="Music will no longer be shuffled.",
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
|
@ -5,7 +5,7 @@ from discord.ext import commands
|
||||
from cogs.music import Music
|
||||
import asyncio
|
||||
|
||||
from utils.config import BOT_COLOR
|
||||
from utils.config import create_embed
|
||||
from utils.custom_sources import LoadError
|
||||
|
||||
|
||||
@ -22,22 +22,25 @@ class Skip(commands.Cog):
|
||||
"Skips the song that is currently playing"
|
||||
player = self.bot.lavalink.player_manager.get(interaction.guild.id)
|
||||
|
||||
embed = discord.Embed(color=BOT_COLOR)
|
||||
|
||||
if number != 1:
|
||||
if number < 1:
|
||||
embed.title = "Invalid Number"
|
||||
embed.description = "The number option cannot be less than 1"
|
||||
embed = create_embed(
|
||||
title="Invalid Number",
|
||||
description="The number option cannot be less than 1",
|
||||
)
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
)
|
||||
|
||||
elif number > len(player.queue):
|
||||
embed.title = "Number too Large"
|
||||
embed.description = (
|
||||
"The number you entered is larger than the number of songs"
|
||||
" in queue. If you want to stop playing music entirely,"
|
||||
" try the </stop:1224840890866991305> command."
|
||||
embed = create_embed(
|
||||
title="Number too Large",
|
||||
description=(
|
||||
"The number you entered is larger than the number of"
|
||||
" songs in queue. If you want to stop playing music"
|
||||
" entirely, try the </stop:1224840890866991305>"
|
||||
" command."
|
||||
),
|
||||
)
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
@ -46,65 +49,54 @@ class Skip(commands.Cog):
|
||||
for i in range(number - 2, -1, -1):
|
||||
player.queue.pop(i)
|
||||
|
||||
# If there is a next song, get it
|
||||
try:
|
||||
next_song = player.queue[0]
|
||||
except IndexError:
|
||||
# If the song is on repeat, catch the IndexError and get the current song
|
||||
# Otherwise, pass
|
||||
if player.loop == 1:
|
||||
embed = discord.Embed(
|
||||
title="Song on Repeat",
|
||||
description=(
|
||||
"There is nothing in queue, but the current song is on"
|
||||
" repeat. Use </stop:1224840890866991305> to stop"
|
||||
" playing music."
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
)
|
||||
else:
|
||||
pass
|
||||
|
||||
# Sometimes when a playlist/album of custom source tracks are loaded, one is not able to be found
|
||||
# so, when a user attempts to skip to that track, we get a LoadError. In this case, just pass it.
|
||||
try:
|
||||
await player.skip()
|
||||
except LoadError:
|
||||
# If the queue is empty, but the current song is on repeat
|
||||
if player.loop == 1 and not player.queue:
|
||||
embed = create_embed(
|
||||
title="Song on Repeat",
|
||||
description=(
|
||||
"There is nothing in queue, but the current song is on"
|
||||
" repeat. Use </stop:1224840890866991305> to stop"
|
||||
" playing music."
|
||||
),
|
||||
)
|
||||
return await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
)
|
||||
else:
|
||||
pass
|
||||
await player.skip()
|
||||
|
||||
# Skip current track, continue skipping on LoadError
|
||||
while True:
|
||||
try:
|
||||
await player.skip()
|
||||
break
|
||||
except LoadError as e:
|
||||
continue
|
||||
|
||||
if not player.current:
|
||||
embed = discord.Embed(
|
||||
embed = create_embed(
|
||||
title="End of Queue",
|
||||
description=(
|
||||
"All songs in queue have been played. Thank you for using"
|
||||
f" me :wave:\n\nIssued by: {interaction.user.mention}"
|
||||
"I have left the voice channel as all songs in the queue"
|
||||
" have been played.\n\n"
|
||||
f"Issued by: {interaction.user.mention}"
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
return await interaction.response.send_message(embed=embed)
|
||||
|
||||
# It takes a sec for the new track to be grabbed and played
|
||||
# So just wait a sec before sending the message
|
||||
await asyncio.sleep(0.5)
|
||||
embed = discord.Embed(
|
||||
title="Track Skipped",
|
||||
description=(
|
||||
f"**Now Playing: [{next_song.title}]({next_song.uri})** by"
|
||||
f" {next_song.author}\n\nQueued by:"
|
||||
f" {next_song.requester.mention}"
|
||||
embed = create_embed(
|
||||
title=(
|
||||
f"{'Track Skipped' if number == 1 else f'{number} Tracks Skipped'}"
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_thumbnail(url=next_song.artwork_url)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
description=(
|
||||
f"**[{player.current.title}]({player.current.uri})**"
|
||||
f" by **{player.current.author}** is now playing\n\n"
|
||||
f"Issued by: {interaction.user.mention}"
|
||||
),
|
||||
thumbnail=player.current.artwork_url,
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
|
@ -4,7 +4,7 @@ from discord import app_commands
|
||||
from discord.ext import commands
|
||||
from cogs.music import Music
|
||||
|
||||
from utils.config import BOT_COLOR
|
||||
from utils.config import create_embed
|
||||
|
||||
|
||||
class Stop(commands.Cog):
|
||||
@ -25,19 +25,12 @@ class Stop(commands.Cog):
|
||||
await player.stop()
|
||||
await interaction.guild.voice_client.disconnect(force=True)
|
||||
|
||||
embed = discord.Embed(
|
||||
embed = create_embed(
|
||||
title="Queue Cleared and Music Stopped",
|
||||
description=(
|
||||
"Thank you for using me :wave:\n\nIssued by:"
|
||||
f" {interaction.user.mention}"
|
||||
f"Thank you for using {self.bot.user.mention}\n\n"
|
||||
f"Issued by: {interaction.user.mention}"
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
|
@ -1,9 +1,8 @@
|
||||
import discord
|
||||
from discord import app_commands
|
||||
from discord.ext.commands.errors import *
|
||||
import datetime
|
||||
|
||||
from utils.config import BOT_COLOR
|
||||
from utils.config import create_embed
|
||||
from utils.custom_sources import LoadError
|
||||
|
||||
|
||||
@ -38,16 +37,9 @@ class Tree(app_commands.CommandTree):
|
||||
# Custom Error class for the `create_player` function
|
||||
# Issues that arise may be user not in vc, user not in correct vc, missing perms, etc.
|
||||
elif isinstance(error, CheckPlayerError):
|
||||
embed = discord.Embed(
|
||||
embed = create_embed(
|
||||
title=error.info["title"],
|
||||
description=error.info["description"],
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
)
|
||||
try:
|
||||
await interaction.response.send_message(
|
||||
@ -62,46 +54,13 @@ class Tree(app_commands.CommandTree):
|
||||
isinstance(error, app_commands.CheckFailure)
|
||||
and interaction.command.name in music_commands
|
||||
):
|
||||
embed = discord.Embed(
|
||||
embed = create_embed(
|
||||
title="Player Creation Error",
|
||||
description=(
|
||||
"An error occured when trying to create a player. Please"
|
||||
" submit a bug report with </bug:1224840889906499626> if"
|
||||
" this issue persists."
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
)
|
||||
try:
|
||||
await interaction.response.send_message(
|
||||
embed=embed, ephemeral=True
|
||||
)
|
||||
except discord.errors.InteractionResponded:
|
||||
await interaction.followup.send(embed=embed, ephemeral=True)
|
||||
|
||||
# If a Spotify song is linked but cannot be found on a provider (e.g. YouTube)
|
||||
elif isinstance(error, LoadError):
|
||||
embed = discord.Embed(
|
||||
title="Nothing Found",
|
||||
description=(
|
||||
"Spotify does not allow direct play, meaning songs have to"
|
||||
" be found on a supported provider. In this case, the song"
|
||||
" couldn't be found. Please try again with a different"
|
||||
" song, or try searching for just the name and artist"
|
||||
" manually rather than sending a link."
|
||||
),
|
||||
color=BOT_COLOR,
|
||||
)
|
||||
embed.set_footer(
|
||||
text=datetime.datetime.now(datetime.timezone.utc).strftime(
|
||||
"%Y-%m-%d %H:%M:%S"
|
||||
)
|
||||
+ " UTC"
|
||||
)
|
||||
try:
|
||||
await interaction.response.send_message(
|
||||
|
@ -8,6 +8,7 @@ import sys
|
||||
import discord
|
||||
import logging
|
||||
import requests
|
||||
from datetime import datetime
|
||||
from colorlog import ColoredFormatter
|
||||
|
||||
log_level = logging.DEBUG
|
||||
@ -32,7 +33,8 @@ BOT_COLOR = None
|
||||
BOT_INVITE_LINK = None
|
||||
FEEDBACK_CHANNEL_ID = None
|
||||
BUG_CHANNEL_ID = None
|
||||
YOUTUBE_SUPPORT = None
|
||||
LOG_SONGS = False
|
||||
YOUTUBE_SUPPORT = False
|
||||
SPOTIFY_CLIENT_ID = None
|
||||
SPOTIFY_CLIENT_SECRET = None
|
||||
GENIUS_CLIENT_ID = None
|
||||
@ -53,10 +55,17 @@ schema = {
|
||||
"bot_invite_link": {"type": "string"},
|
||||
"feedback_channel_id": {"type": "integer"},
|
||||
"bug_channel_id": {"type": "integer"},
|
||||
"youtube_support": {"type": "boolean"},
|
||||
"log_songs": {"type": "boolean"},
|
||||
},
|
||||
"required": ["token"],
|
||||
},
|
||||
"youtube": {
|
||||
"type": "object",
|
||||
"properties": {
|
||||
"enabled": {"type": "boolean"},
|
||||
},
|
||||
"required": ["enabled"],
|
||||
},
|
||||
"spotify": {
|
||||
"type": "object",
|
||||
"properties": {
|
||||
@ -117,7 +126,10 @@ bot_info:
|
||||
bot_invite_link: ""
|
||||
feedback_channel_id: ""
|
||||
bug_channel_id: ""
|
||||
youtube_support: false
|
||||
log_songs: true
|
||||
|
||||
youtube:
|
||||
enabled: false
|
||||
|
||||
spotify:
|
||||
spotify_client_id: ""
|
||||
@ -148,7 +160,7 @@ lavalink:
|
||||
|
||||
# Thouroughly validate all of the options in the config.yaml file
|
||||
def validate_config(file_contents):
|
||||
global TOKEN, BOT_COLOR, BOT_INVITE_LINK, FEEDBACK_CHANNEL_ID, BUG_CHANNEL_ID, YOUTUBE_SUPPORT, SPOTIFY_CLIENT_ID, SPOTIFY_CLIENT_SECRET, GENIUS_CLIENT_ID, GENIUS_CLIENT_SECRET, OPENAI_API_KEY, LAVALINK_HOST, LAVALINK_PORT, LAVALINK_PASSWORD
|
||||
global TOKEN, BOT_COLOR, BOT_INVITE_LINK, FEEDBACK_CHANNEL_ID, BUG_CHANNEL_ID, LOG_SONGS, YOUTUBE_SUPPORT, SPOTIFY_CLIENT_ID, SPOTIFY_CLIENT_SECRET, GENIUS_CLIENT_ID, GENIUS_CLIENT_SECRET, OPENAI_API_KEY, LAVALINK_HOST, LAVALINK_PORT, LAVALINK_PASSWORD
|
||||
config = yaml.safe_load(file_contents)
|
||||
|
||||
try:
|
||||
@ -208,10 +220,12 @@ def validate_config(file_contents):
|
||||
else:
|
||||
BUG_CHANNEL_ID = config["bot_info"]["bug_channel_id"]
|
||||
|
||||
if "youtube_support" in config["bot_info"]:
|
||||
YOUTUBE_SUPPORT = bool(config["bot_info"]["youtube_support"])
|
||||
else:
|
||||
YOUTUBE_SUPPORT = False
|
||||
if "log_songs" in config["bot_info"]:
|
||||
LOG_SONGS = bool(config["bot_info"]["log_songs"])
|
||||
|
||||
# Check for YouTube support
|
||||
if "youtube" in config:
|
||||
YOUTUBE_SUPPORT = bool(config["youtube"]["enabled"])
|
||||
|
||||
#
|
||||
# If the SPOTIFY section is present, make sure the client ID and secret are valid
|
||||
@ -274,3 +288,34 @@ def validate_config(file_contents):
|
||||
LAVALINK_HOST = config["lavalink"]["host"]
|
||||
LAVALINK_PORT = config["lavalink"]["port"]
|
||||
LAVALINK_PASSWORD = config["lavalink"]["password"]
|
||||
|
||||
|
||||
"""
|
||||
Template for embeds
|
||||
"""
|
||||
|
||||
|
||||
def create_embed(
|
||||
title: str = None,
|
||||
description: str = None,
|
||||
color=None,
|
||||
footer=None,
|
||||
thumbnail=None,
|
||||
):
|
||||
embed = discord.Embed(
|
||||
title=title,
|
||||
description=description,
|
||||
color=color if color else BOT_COLOR,
|
||||
)
|
||||
|
||||
if footer:
|
||||
embed.set_footer(text=footer)
|
||||
# else:
|
||||
# embed.set_footer(
|
||||
# text=datetime.utcnow().strftime("%Y-%m-%d %H:%M:%S") + " UTC"
|
||||
# )
|
||||
|
||||
if thumbnail:
|
||||
embed.set_thumbnail(url=thumbnail)
|
||||
|
||||
return embed
|
||||
|
@ -6,6 +6,8 @@ from lavalink import (
|
||||
PlaylistInfo,
|
||||
)
|
||||
|
||||
from utils.config import YOUTUBE_SUPPORT
|
||||
|
||||
|
||||
class LoadError(
|
||||
Exception
|
||||
@ -32,21 +34,22 @@ class CustomAudioTrack(DeferredAudioTrack):
|
||||
LoadType.EMPTY,
|
||||
LoadType.ERROR,
|
||||
):
|
||||
ytmsearch = f"ytmsearch:{self.title} {self.author}"
|
||||
results = await client.get_tracks(ytmsearch)
|
||||
|
||||
if not results.tracks or results.load_type in (
|
||||
LoadType.EMPTY,
|
||||
LoadType.ERROR,
|
||||
):
|
||||
ytsearch = f"ytsearch:{self.title} {self.author} audio"
|
||||
results = await client.get_tracks(ytsearch)
|
||||
if YOUTUBE_SUPPORT:
|
||||
ytmsearch = f"ytmsearch:{self.title} {self.author}"
|
||||
results = await client.get_tracks(ytmsearch)
|
||||
|
||||
if not results.tracks or results.load_type in (
|
||||
LoadType.EMPTY,
|
||||
LoadType.ERROR,
|
||||
):
|
||||
raise LoadError
|
||||
ytsearch = f"ytsearch:{self.title} {self.author} audio"
|
||||
results = await client.get_tracks(ytsearch)
|
||||
|
||||
if not results.tracks or results.load_type in (
|
||||
LoadType.EMPTY,
|
||||
LoadType.ERROR,
|
||||
):
|
||||
raise LoadError
|
||||
|
||||
first_track = results.tracks[
|
||||
0
|
||||
@ -70,6 +73,10 @@ class SpotifySource(Source):
|
||||
) # Initialising our custom source with the name 'custom'.
|
||||
|
||||
async def load_item(self, user, metadata):
|
||||
try:
|
||||
artwork_url = metadata["album"]["images"][0]["url"]
|
||||
except IndexError:
|
||||
artwork_url = None
|
||||
track = CustomAudioTrack(
|
||||
{ # Create an instance of our CustomAudioTrack.
|
||||
"identifier": metadata[
|
||||
@ -82,7 +89,7 @@ class SpotifySource(Source):
|
||||
"title": metadata["name"],
|
||||
"uri": metadata["external_urls"]["spotify"],
|
||||
"duration": metadata["duration_ms"],
|
||||
"artworkUrl": metadata["album"]["images"][0]["url"],
|
||||
"artworkUrl": artwork_url,
|
||||
},
|
||||
requester=user,
|
||||
)
|
||||
@ -91,6 +98,11 @@ class SpotifySource(Source):
|
||||
)
|
||||
|
||||
async def load_album(self, user, metadata):
|
||||
try:
|
||||
artwork_url = metadata["images"][0]["url"]
|
||||
except IndexError:
|
||||
artwork_url = None
|
||||
|
||||
tracks = []
|
||||
for track in metadata["tracks"][
|
||||
"items"
|
||||
@ -108,7 +120,7 @@ class SpotifySource(Source):
|
||||
"title": track["name"],
|
||||
"uri": track["external_urls"]["spotify"],
|
||||
"duration": track["duration_ms"],
|
||||
"artworkUrl": metadata["images"][0]["url"],
|
||||
"artworkUrl": artwork_url,
|
||||
},
|
||||
requester=user,
|
||||
)
|
||||
@ -123,6 +135,10 @@ class SpotifySource(Source):
|
||||
for track in metadata["tracks"][
|
||||
"items"
|
||||
]: # Loop through each track in the playlist.
|
||||
try:
|
||||
artwork_url = track["track"]["album"]["images"][0]["url"]
|
||||
except IndexError:
|
||||
artwork_url = None
|
||||
tracks.append(
|
||||
CustomAudioTrack(
|
||||
{ # Create an instance of our CustomAudioTrack.
|
||||
@ -136,9 +152,7 @@ class SpotifySource(Source):
|
||||
"title": track["track"]["name"],
|
||||
"uri": track["track"]["external_urls"]["spotify"],
|
||||
"duration": track["track"]["duration_ms"],
|
||||
"artworkUrl": track["track"]["album"]["images"][0][
|
||||
"url"
|
||||
],
|
||||
"artworkUrl": artwork_url,
|
||||
},
|
||||
requster=user,
|
||||
)
|
||||
@ -148,6 +162,34 @@ class SpotifySource(Source):
|
||||
LoadType.PLAYLIST, tracks, playlist_info=PlaylistInfo.none()
|
||||
)
|
||||
|
||||
async def load_artist(self, user, metadata):
|
||||
tracks = []
|
||||
for track in metadata["tracks"]:
|
||||
try:
|
||||
artwork_url = track["album"]["images"][0]["url"]
|
||||
except IndexError:
|
||||
artwork_url = None
|
||||
tracks.append(
|
||||
CustomAudioTrack(
|
||||
{
|
||||
"identifier": track["id"],
|
||||
"isSeekable": True,
|
||||
"author": track["artists"][0]["name"],
|
||||
"length": track["duration_ms"],
|
||||
"isStream": False,
|
||||
"title": track["name"],
|
||||
"uri": track["external_urls"]["spotify"],
|
||||
"duration": track["duration_ms"],
|
||||
"artworkUrl": artwork_url,
|
||||
},
|
||||
requester=user,
|
||||
)
|
||||
)
|
||||
|
||||
return LoadResult(
|
||||
LoadType.PLAYLIST, tracks, playlist_info=PlaylistInfo.none()
|
||||
)
|
||||
|
||||
|
||||
"""
|
||||
Custom Source for Apple Music links
|
||||
|
74
code/utils/source_helpers/apple/album.py
Normal file
74
code/utils/source_helpers/apple/album.py
Normal file
@ -0,0 +1,74 @@
|
||||
import datetime
|
||||
import discord
|
||||
import requests
|
||||
from typing import Tuple, Optional
|
||||
from requests.exceptions import JSONDecodeError
|
||||
|
||||
from utils.config import create_embed, LOG
|
||||
|
||||
|
||||
async def load(
|
||||
headers: dict,
|
||||
query: str,
|
||||
user: discord.User,
|
||||
) -> Tuple[Optional[dict], Optional[discord.Embed]]:
|
||||
"""
|
||||
Get the album info from the Apple Music API
|
||||
"""
|
||||
album_id = query.split("/album/")[1].split("/")[1]
|
||||
|
||||
try:
|
||||
# Get the album info
|
||||
response = requests.get(
|
||||
f"https://api.music.apple.com/v1/catalog/us/albums/{album_id}",
|
||||
headers=headers,
|
||||
)
|
||||
|
||||
if response.status_code == 404:
|
||||
embed = create_embed(
|
||||
title="Album Not Found",
|
||||
description=(
|
||||
"The album could not be found as the provided link is"
|
||||
" invalid. Please try again."
|
||||
),
|
||||
)
|
||||
return None, embed
|
||||
|
||||
if response.status_code == 401:
|
||||
LOG.error(
|
||||
"Could not authorize with Apple Music API. Likely need to"
|
||||
" restart the bot."
|
||||
)
|
||||
return None, None
|
||||
|
||||
response.raise_for_status()
|
||||
# Unpack the album info
|
||||
album = response.json()
|
||||
name = album["data"][0]["attributes"]["name"]
|
||||
artist = album["data"][0]["attributes"]["artistName"]
|
||||
num_tracks = len(album["data"][0]["relationships"]["tracks"]["data"])
|
||||
except IndexError:
|
||||
LOG.error("Failed unpacking Apple Music album info")
|
||||
return None, None
|
||||
except (JSONDecodeError, requests.HTTPError):
|
||||
LOG.error("Failed making request to Apple Music API")
|
||||
return None, None
|
||||
|
||||
# Extract artwork URL, if available
|
||||
artwork_url = (
|
||||
album["data"][0]["attributes"].get("artwork", {}).get("url", None)
|
||||
)
|
||||
if artwork_url:
|
||||
artwork_url = artwork_url.replace("{w}x{h}", "300x300")
|
||||
|
||||
embed = create_embed(
|
||||
title="Album Queued",
|
||||
description=(
|
||||
f"**{name}** by **{artist}**\n"
|
||||
f"` {num_tracks} ` tracks\n\n"
|
||||
f"Queued by: {user.mention}"
|
||||
),
|
||||
thumbnail=artwork_url,
|
||||
)
|
||||
|
||||
return album, embed
|
88
code/utils/source_helpers/apple/playlist.py
Normal file
88
code/utils/source_helpers/apple/playlist.py
Normal file
@ -0,0 +1,88 @@
|
||||
import discord
|
||||
import requests
|
||||
from typing import Tuple, Optional
|
||||
from requests.exceptions import JSONDecodeError
|
||||
|
||||
from utils.config import create_embed, LOG
|
||||
|
||||
|
||||
async def load(
|
||||
headers: dict,
|
||||
query: str,
|
||||
user: discord.User,
|
||||
) -> Tuple[Optional[dict], Optional[discord.Embed]]:
|
||||
"""
|
||||
Get the playlist info from the Apple Music API
|
||||
"""
|
||||
playlist_id = query.split("/playlist/")[1].split("/")[1]
|
||||
try:
|
||||
# Get all of the tracks in the playlist (limit at 100)
|
||||
response = requests.get(
|
||||
f"https://api.music.apple.com/v1/catalog/us/playlists/{playlist_id}/tracks?limit=100",
|
||||
headers=headers,
|
||||
)
|
||||
|
||||
if response.status_code == 404:
|
||||
embed = create_embed(
|
||||
title="Playlist Not Found",
|
||||
description=(
|
||||
"The playlist could not be found as the provided link is"
|
||||
" invalid. Please try again."
|
||||
),
|
||||
)
|
||||
return None, embed
|
||||
|
||||
if response.status_code == 401:
|
||||
LOG.error(
|
||||
"Could not authorize with Apple Music API. Likely need to"
|
||||
" restart the bot."
|
||||
)
|
||||
return None, None
|
||||
|
||||
response.raise_for_status()
|
||||
playlist = response.json()
|
||||
|
||||
# Get the general playlist info (name, artwork)
|
||||
response = requests.get(
|
||||
f"https://api.music.apple.com/v1/catalog/us/playlists/{playlist_id}",
|
||||
headers=headers,
|
||||
)
|
||||
|
||||
response.raise_for_status()
|
||||
# Unpack the playlist info
|
||||
playlist_info = response.json()
|
||||
name = playlist_info["data"][0]["attributes"]["name"]
|
||||
num_tracks = len(playlist["data"])
|
||||
except IndexError:
|
||||
LOG.error("Failed unpacking Apple Music playlist info")
|
||||
return None, None
|
||||
except (JSONDecodeError, requests.HTTPError):
|
||||
LOG.error("Failed making request to Apple Music API")
|
||||
return None, None
|
||||
|
||||
# Extract artwork URL, if available
|
||||
artwork_url = (
|
||||
playlist_info["data"][0]["attributes"]
|
||||
.get("artwork", {})
|
||||
.get("url", None)
|
||||
)
|
||||
if artwork_url:
|
||||
artwork_url = artwork_url.replace("{w}x{h}", "300x300")
|
||||
|
||||
embed = create_embed(
|
||||
title="Playlist Queued",
|
||||
description=(
|
||||
f"**{name}**\n` {num_tracks} ` tracks\n\nQueued by: {user.mention}"
|
||||
),
|
||||
thumbnail=artwork_url,
|
||||
)
|
||||
|
||||
# Add small alert if the playlist is the max size
|
||||
if len(playlist["data"]) == 100:
|
||||
embed.description += (
|
||||
"\n\n*This playlist is longer than the 100 song"
|
||||
" maximum. Only the first 100 songs will be"
|
||||
" queued.*"
|
||||
)
|
||||
|
||||
return playlist, embed
|
68
code/utils/source_helpers/apple/song.py
Normal file
68
code/utils/source_helpers/apple/song.py
Normal file
@ -0,0 +1,68 @@
|
||||
import discord
|
||||
import requests
|
||||
from typing import Tuple, Optional
|
||||
from requests.exceptions import JSONDecodeError
|
||||
|
||||
from utils.config import create_embed, LOG
|
||||
|
||||
|
||||
async def load(
|
||||
headers: dict,
|
||||
query: str,
|
||||
user: discord.User,
|
||||
) -> Tuple[Optional[dict], Optional[discord.Embed]]:
|
||||
"""
|
||||
Get the song info from the Apple Music API
|
||||
"""
|
||||
song_id = query.split("/album/")[1].split("?i=")[1]
|
||||
|
||||
try:
|
||||
# Get the song info
|
||||
response = requests.get(
|
||||
f"https://api.music.apple.com/v1/catalog/us/songs/{song_id}",
|
||||
headers=headers,
|
||||
)
|
||||
|
||||
if response.status_code == 404:
|
||||
embed = create_embed(
|
||||
title="Song Not Found",
|
||||
description=(
|
||||
"The song could not be found as the provided link is"
|
||||
" invalid. Please try again."
|
||||
),
|
||||
)
|
||||
return None, embed
|
||||
|
||||
if response.status_code == 401:
|
||||
LOG.error(
|
||||
"Could not authorize with Apple Music API. Likely need to"
|
||||
" restart the bot."
|
||||
)
|
||||
return None, None
|
||||
|
||||
response.raise_for_status()
|
||||
# Unpack the song info
|
||||
song = response.json()
|
||||
name = song["data"][0]["attributes"]["name"]
|
||||
artist = song["data"][0]["attributes"]["artistName"]
|
||||
except IndexError:
|
||||
LOG.error("Failed unpacking Apple Music song info")
|
||||
return None, None
|
||||
except (JSONDecodeError, requests.HTTPError):
|
||||
LOG.error("Failed making request to Apple Music API")
|
||||
return None, None
|
||||
|
||||
# Extract artwork URL, if available
|
||||
artwork_url = (
|
||||
song["data"][0]["attributes"].get("artwork", {}).get("url", None)
|
||||
)
|
||||
if artwork_url:
|
||||
artwork_url = artwork_url.replace("{w}x{h}", "300x300")
|
||||
|
||||
embed = create_embed(
|
||||
title="Song Queued",
|
||||
description=f"**{name}** by **{artist}**\n\nQueued by {user.mention}",
|
||||
thumbnail=artwork_url,
|
||||
)
|
||||
|
||||
return song, embed
|
91
code/utils/source_helpers/parse.py
Normal file
91
code/utils/source_helpers/parse.py
Normal file
@ -0,0 +1,91 @@
|
||||
import discord
|
||||
|
||||
from utils.source_helpers.apple import (
|
||||
album as apple_album,
|
||||
playlist as apple_playlist,
|
||||
song as apple_song,
|
||||
)
|
||||
from utils.source_helpers.spotify import (
|
||||
album as spotify_album,
|
||||
artist as spotify_artist,
|
||||
playlist as spotify_playlist,
|
||||
song as spotify_song,
|
||||
)
|
||||
from utils.custom_sources import AppleSource, SpotifySource
|
||||
|
||||
|
||||
async def parse_custom_source(
|
||||
self, provider: str, query: str, user: discord.User
|
||||
):
|
||||
"""
|
||||
Parse the query and run the appropriate functions to get the results/info
|
||||
|
||||
Return the results and an embed or None, None
|
||||
"""
|
||||
load_funcs = {
|
||||
"apple": {
|
||||
"album": apple_album.load,
|
||||
"playlist": apple_playlist.load,
|
||||
"song": apple_song.load,
|
||||
},
|
||||
"spotify": {
|
||||
"album": spotify_album.load,
|
||||
"artist": spotify_artist.load,
|
||||
"playlist": spotify_playlist.load,
|
||||
"song": spotify_song.load,
|
||||
},
|
||||
}
|
||||
|
||||
headers = {
|
||||
"apple": self.bot.apple_headers,
|
||||
"spotify": self.bot.spotify_headers,
|
||||
}
|
||||
|
||||
sources = {
|
||||
"apple": AppleSource,
|
||||
"spotify": SpotifySource,
|
||||
}
|
||||
# Catch all songs
|
||||
if "?i=" in query or "/track/" in query:
|
||||
song, embed = await load_funcs[provider]["song"](
|
||||
headers[provider], query, user
|
||||
)
|
||||
|
||||
if song:
|
||||
results = await sources[provider].load_item(self, user, song)
|
||||
else:
|
||||
return None, embed
|
||||
# Catch all playlists
|
||||
elif "/playlist/" in query:
|
||||
playlist, embed = await load_funcs[provider]["playlist"](
|
||||
headers[provider], query, user
|
||||
)
|
||||
|
||||
if playlist:
|
||||
results = await sources[provider].load_playlist(
|
||||
self, user, playlist
|
||||
)
|
||||
else:
|
||||
return None, embed
|
||||
# Catch all albums
|
||||
elif "/album/" in query:
|
||||
album, embed = await load_funcs[provider]["album"](
|
||||
headers[provider], query, user
|
||||
)
|
||||
|
||||
if album:
|
||||
results = await sources[provider].load_album(self, user, album)
|
||||
else:
|
||||
return None, embed
|
||||
# Catch Spotify artists
|
||||
elif "/artist/" in query:
|
||||
artist, embed = await load_funcs[provider]["artist"](
|
||||
headers[provider], query, user
|
||||
)
|
||||
|
||||
if artist:
|
||||
results = await sources[provider].load_artist(self, user, artist)
|
||||
else:
|
||||
return None, embed
|
||||
|
||||
return results, embed
|
68
code/utils/source_helpers/spotify/album.py
Normal file
68
code/utils/source_helpers/spotify/album.py
Normal file
@ -0,0 +1,68 @@
|
||||
import datetime
|
||||
import discord
|
||||
import requests
|
||||
from typing import Tuple, Optional
|
||||
from requests.exceptions import JSONDecodeError
|
||||
|
||||
from utils.config import create_embed, LOG
|
||||
|
||||
|
||||
async def load(
|
||||
headers: dict,
|
||||
query: str,
|
||||
user: discord.User,
|
||||
) -> Tuple[Optional[dict], Optional[discord.Embed]]:
|
||||
"""
|
||||
Get the album info from the Spotify API
|
||||
"""
|
||||
album_id = query.split("/album/")[1]
|
||||
|
||||
try:
|
||||
# Get the album info
|
||||
response = requests.get(
|
||||
f"https://api.spotify.com/v1/albums/{album_id}",
|
||||
headers=headers,
|
||||
)
|
||||
|
||||
if response.status_code == 404:
|
||||
embed = create_embed(
|
||||
title="Album Not Found",
|
||||
description=(
|
||||
"The album could not be found as the provided link is"
|
||||
" invalid. Please try again."
|
||||
),
|
||||
)
|
||||
return None, embed
|
||||
|
||||
if response.status_code == 401:
|
||||
LOG.error(
|
||||
"Could not authorize with Spotify API. Likely need to"
|
||||
" restart the bot."
|
||||
)
|
||||
return None, None
|
||||
|
||||
response.raise_for_status()
|
||||
# Unpack the album info
|
||||
album = response.json()
|
||||
name = album["name"]
|
||||
artist = album["artists"][0]["name"]
|
||||
num_tracks = len(album["tracks"]["items"])
|
||||
artwork_url = album["images"][0]["url"]
|
||||
except IndexError:
|
||||
LOG.error("Failed unpacking Spotify album info")
|
||||
return None, None
|
||||
except (JSONDecodeError, requests.HTTPError):
|
||||
LOG.error("Failed making request to Spotify API")
|
||||
return None, None
|
||||
|
||||
embed = create_embed(
|
||||
title="Album Queued",
|
||||
description=(
|
||||
f"**{name}** by **{artist}**\n"
|
||||
f"` {num_tracks} ` tracks\n\n"
|
||||
f"Queued by: {user.mention}"
|
||||
),
|
||||
thumbnail=artwork_url,
|
||||
)
|
||||
|
||||
return album, embed
|
77
code/utils/source_helpers/spotify/artist.py
Normal file
77
code/utils/source_helpers/spotify/artist.py
Normal file
@ -0,0 +1,77 @@
|
||||
import datetime
|
||||
import discord
|
||||
import requests
|
||||
from typing import Tuple, Optional
|
||||
from requests.exceptions import JSONDecodeError
|
||||
|
||||
from utils.config import create_embed, LOG
|
||||
|
||||
|
||||
async def load(
|
||||
headers: dict,
|
||||
query: str,
|
||||
user: discord.User,
|
||||
) -> Tuple[Optional[dict], Optional[discord.Embed]]:
|
||||
"""
|
||||
Get the artists top tracks from the Spotify API
|
||||
"""
|
||||
artist_id = query.split("/artist/")[1]
|
||||
|
||||
try:
|
||||
# Get the artists songs
|
||||
response = requests.get(
|
||||
f"https://api.spotify.com/v1/artists/{artist_id}/top-tracks",
|
||||
headers=headers,
|
||||
)
|
||||
|
||||
if response.status_code == 404:
|
||||
embed = create_embed(
|
||||
title="Artist Not Found",
|
||||
description=(
|
||||
"Either the provided link is malformed, the artist does"
|
||||
" not exist, or the artist does not have any songs."
|
||||
),
|
||||
)
|
||||
return None, embed
|
||||
|
||||
if response.status_code == 401:
|
||||
LOG.error(
|
||||
"Could not authorize with Spotify API. Likely need to"
|
||||
" restart the bot."
|
||||
)
|
||||
return None, None
|
||||
|
||||
response.raise_for_status()
|
||||
# Unpack the artists songs
|
||||
artist = response.json()
|
||||
name = artist["tracks"][0]["artists"][0]["name"]
|
||||
num_tracks = len(artist["tracks"])
|
||||
|
||||
# Get the artist info (for the thumbnail)
|
||||
response = requests.get(
|
||||
f"https://api.spotify.com/v1/artists/{artist_id}",
|
||||
headers=headers,
|
||||
)
|
||||
|
||||
response.raise_for_status()
|
||||
try:
|
||||
artwork_url = response.json()["images"][0]["url"]
|
||||
except IndexError:
|
||||
artwork_url = None
|
||||
|
||||
except IndexError:
|
||||
LOG.error("Failed unpacking Spotify artist info")
|
||||
return None, None
|
||||
except (JSONDecodeError, requests.HTTPError):
|
||||
LOG.error("Failed making request to Spotify API")
|
||||
return None, None
|
||||
|
||||
embed = create_embed(
|
||||
title="Artist Queued",
|
||||
description=(
|
||||
f"Top `{num_tracks}` track by **{name}**\n\n"
|
||||
f"Queued by {user.mention}"
|
||||
),
|
||||
thumbnail=artwork_url,
|
||||
)
|
||||
return artist, embed
|
68
code/utils/source_helpers/spotify/playlist.py
Normal file
68
code/utils/source_helpers/spotify/playlist.py
Normal file
@ -0,0 +1,68 @@
|
||||
import datetime
|
||||
import discord
|
||||
import requests
|
||||
from typing import Tuple, Optional
|
||||
from requests.exceptions import JSONDecodeError
|
||||
|
||||
from utils.config import create_embed, LOG
|
||||
|
||||
|
||||
async def load(
|
||||
headers: dict,
|
||||
query: str,
|
||||
user: discord.User,
|
||||
) -> Tuple[Optional[dict], Optional[discord.Embed]]:
|
||||
"""
|
||||
Get the playlist info from the Spotify API
|
||||
"""
|
||||
playlist_id = query.split("/playlist/")[1]
|
||||
|
||||
try:
|
||||
# Get the playlist info
|
||||
response = requests.get(
|
||||
f"https://api.spotify.com/v1/playlists/{playlist_id}",
|
||||
headers=headers,
|
||||
)
|
||||
|
||||
if response.status_code == 404:
|
||||
embed = create_embed(
|
||||
title="Playlist Not Found",
|
||||
description=(
|
||||
"The playlist could not be found as the provided link is"
|
||||
" invalid. Please try again."
|
||||
),
|
||||
)
|
||||
return None, embed
|
||||
|
||||
if response.status_code == 401:
|
||||
LOG.error(
|
||||
"Could not authorize with Spotify API. Likely need to"
|
||||
" restart the bot."
|
||||
)
|
||||
return None, None
|
||||
|
||||
response.raise_for_status()
|
||||
# Unpack the playlist info
|
||||
playlist = response.json()
|
||||
name = playlist["name"]
|
||||
owner = playlist["owner"]["display_name"]
|
||||
num_tracks = len(playlist["tracks"]["items"])
|
||||
artwork_url = playlist["images"][0]["url"]
|
||||
except IndexError:
|
||||
LOG.error("Failed unpacking Spotify playlist info")
|
||||
return None, None
|
||||
except (JSONDecodeError, requests.HTTPError):
|
||||
LOG.error("Failed making request to Spotify API")
|
||||
return None, None
|
||||
|
||||
embed = create_embed(
|
||||
title="Playlist Queued",
|
||||
description=(
|
||||
f"**{name}** from **{owner}**\n"
|
||||
f"` {num_tracks} ` tracks\n\n"
|
||||
f"Queued by {user.mention}"
|
||||
),
|
||||
thumbnail=artwork_url,
|
||||
)
|
||||
|
||||
return playlist, embed
|
63
code/utils/source_helpers/spotify/song.py
Normal file
63
code/utils/source_helpers/spotify/song.py
Normal file
@ -0,0 +1,63 @@
|
||||
import datetime
|
||||
import discord
|
||||
import requests
|
||||
from typing import Tuple, Optional
|
||||
from requests.exceptions import JSONDecodeError
|
||||
|
||||
from utils.config import create_embed, LOG
|
||||
|
||||
|
||||
async def load(
|
||||
headers: dict,
|
||||
query: str,
|
||||
user: discord.User,
|
||||
) -> Tuple[Optional[dict], Optional[discord.Embed]]:
|
||||
"""
|
||||
Get the song info from the Spotify API
|
||||
"""
|
||||
song_id = query.split("/track/")[1]
|
||||
|
||||
try:
|
||||
# Get the song info
|
||||
response = requests.get(
|
||||
f"https://api.spotify.com/v1/tracks/{song_id}",
|
||||
headers=headers,
|
||||
)
|
||||
|
||||
if response.status_code == 404:
|
||||
embed = create_embed(
|
||||
title="Song Not Found",
|
||||
description=(
|
||||
"The song could not be found as the provided link is"
|
||||
" invalid. Please try again."
|
||||
),
|
||||
)
|
||||
return None, embed
|
||||
|
||||
if response.status_code == 401:
|
||||
LOG.error(
|
||||
"Could not authorize with Spotify API. Likely need to"
|
||||
" restart the bot."
|
||||
)
|
||||
return None, None
|
||||
|
||||
response.raise_for_status()
|
||||
# Unpack the song info
|
||||
song = response.json()
|
||||
name = song["name"]
|
||||
artist = song["artists"][0]["name"]
|
||||
artwork_url = song["album"]["images"][0]["url"]
|
||||
except IndexError:
|
||||
LOG.error("Failed unpacking Spotify song info")
|
||||
return None, None
|
||||
except (JSONDecodeError, requests.HTTPError):
|
||||
LOG.error("Failed making request to Spotify API")
|
||||
return None, None
|
||||
|
||||
embed = create_embed(
|
||||
title="Song Queued",
|
||||
description=f"**{name}** by **{artist}**\n\nQueued by {user.mention}",
|
||||
thumbnail=artwork_url,
|
||||
)
|
||||
|
||||
return song, embed
|
Loading…
x
Reference in New Issue
Block a user