Support OpenAI AND Groq
Some checks failed
Create and publish a Docker image / build-and-push-image (push) Has been cancelled
Some checks failed
Create and publish a Docker image / build-and-push-image (push) Has been cancelled
This commit is contained in:
parent
b335e82699
commit
06034d0b37
@ -106,10 +106,11 @@ Field | Description | Requirement
|
||||
GENIUS_CLIENT_ID | `CLIENT ID`: ID from Genius API Dashboard | **OPTIONAL** - *Used for the /lyrics command*
|
||||
GENIUS_CLIENT_SECRET | `CLIENT SECRET`: Secret string from Genius API Dashboard | **OPTIONAL** - *Used for the /lyrics command*
|
||||
|
||||
## OPENAI | OPTIONAL
|
||||
## AI | OPTIONAL
|
||||
Field | Description | Requirement
|
||||
--- | --- | ---
|
||||
OPENAI_API_KEY | API Key from OpenAI for autoplay recommendations | **OPTIONAL** - *Used to support the /autoplay feature*
|
||||
SERVICE | Which providers API you will use. Supports `groq` or `openai` | **OPTIONAL** - *Used to support the /autoplay feature*
|
||||
API_KEY | API key for the provider you are using | **OPTIONAL** - *Used to support the /autoplay feature*
|
||||
|
||||
<br>
|
||||
|
||||
|
10
code/bot.py
10
code/bot.py
@ -2,7 +2,6 @@ import discord
|
||||
from discord.ext import commands, tasks
|
||||
import os
|
||||
import requests
|
||||
import openai
|
||||
import lyricsgenius
|
||||
|
||||
import utils.config as config
|
||||
@ -20,13 +19,10 @@ class MyBot(commands.Bot):
|
||||
)
|
||||
|
||||
async def setup_hook(self):
|
||||
# Get Spotify, Apple Music, Genius, and OpenAI access tokens/clients
|
||||
# Get Spotify, Apple Music, and Genius access tokens/clients
|
||||
get_access_token.start()
|
||||
refresh_media_api_key.start()
|
||||
login_genius.start()
|
||||
if config.OPENAI_API_KEY:
|
||||
bot.openai = openai.OpenAI(api_key=config.OPENAI_API_KEY)
|
||||
|
||||
config.LOG.info("Loading cogs...")
|
||||
config.LOG.info(
|
||||
"YouTube support is enabled, make sure to set a poToken"
|
||||
@ -59,9 +55,9 @@ class MyBot(commands.Bot):
|
||||
)
|
||||
continue
|
||||
# Load the OPTIONAL autoplay cog
|
||||
if ext[:-3] == "autoplay" and config.OPENAI_API_KEY == None:
|
||||
if ext[:-3] == "autoplay" and config.AI_CLIENT == None:
|
||||
config.LOG.info(
|
||||
"Skipped loading autoplay cog - OpenAI API credentials"
|
||||
"Skipped loading autoplay cog - AI API credentials"
|
||||
" not provided"
|
||||
)
|
||||
continue
|
||||
|
@ -75,9 +75,7 @@ class Autoplay(commands.Cog):
|
||||
)
|
||||
await interaction.response.send_message(embed=embed)
|
||||
|
||||
if await add_song_recommendations(
|
||||
self.bot.openai, self.bot.user, player, 5, inputs
|
||||
):
|
||||
if await add_song_recommendations(self.bot.user, player, 5, inputs):
|
||||
self.bot.autoplay.append(interaction.guild.id)
|
||||
embed = create_embed(
|
||||
title=":infinity: Autoplay Enabled :infinity:",
|
||||
|
@ -56,7 +56,8 @@ commands_and_descriptions = {
|
||||
},
|
||||
"autoplay": {
|
||||
"description": (
|
||||
"Keep the music playing forever with music suggestions from OpenAI"
|
||||
"Keep the music playing forever with automatic song"
|
||||
" recommendations"
|
||||
),
|
||||
"arguments": {
|
||||
"on": "Turn autoplay feature on",
|
||||
|
@ -245,7 +245,7 @@ class Music(commands.Cog):
|
||||
for song in event.player.queue[:10]:
|
||||
inputs[song.title] = song.author
|
||||
await add_song_recommendations(
|
||||
self.bot.openai, self.bot.user, event.player, 5, inputs
|
||||
self.bot.user, event.player, 5, inputs
|
||||
)
|
||||
|
||||
@lavalink.listener(lavalink.events.NodeConnectedEvent)
|
||||
|
@ -1,40 +1,52 @@
|
||||
from lavalink import LoadType
|
||||
import re
|
||||
|
||||
from utils.config import AI_CLIENT, AI_MODEL
|
||||
|
||||
|
||||
async def add_song_recommendations(
|
||||
openai_client, bot_user, player, number, inputs, retries: int = 1
|
||||
bot_user, player, number, inputs, retries: int = 1
|
||||
):
|
||||
input_list = [f'"{song} by {artist}"' for song, artist in inputs.items()]
|
||||
|
||||
completion = (
|
||||
openai_client.chat.completions.create(
|
||||
AI_CLIENT.chat.completions.create(
|
||||
messages=[
|
||||
{
|
||||
"role": "system",
|
||||
"content": f"""
|
||||
Given an input list of songs formatted as ["song_name
|
||||
by artist_name", "song_name by artist_name", ...], generate
|
||||
a list of 5 new songs that the user may enjoy based on
|
||||
the input.
|
||||
|
||||
Thoroughly analyze each song in the input list, considering
|
||||
factors such as tempo, beat, mood, genre, lyrical themes,
|
||||
instrumentation, and overall meaning. Use this analysis to
|
||||
recommend 5 songs that closely align with the user's musical
|
||||
preferences.
|
||||
|
||||
The output must be formatted in the exact same way:
|
||||
["song_name by artist_name", "song_name by artist_name", ...].
|
||||
|
||||
If you are unable to find 5 new songs or encounter any issues,
|
||||
return the following list instead: ["NOTHING_FOUND"]. Do
|
||||
not return partial results—either provide 5 songs or return
|
||||
["NOTHING_FOUND"]. Ensure accuracy in song and artist names.
|
||||
|
||||
DO NOT include any additional information or text in the
|
||||
output, it should STRICTLY be either a list of the songs
|
||||
or ["NOTHING_FOUND"].
|
||||
""",
|
||||
},
|
||||
{
|
||||
"role": "user",
|
||||
"content": f"""
|
||||
BACKGROUND: You're an AI music recommendation system with a knack for understanding
|
||||
user preferences based on provided input. Your task is to generate a list
|
||||
of {number} songs that the user might enjoy, derived from a given list of {number} songs.
|
||||
The input will be in the format of
|
||||
["Song-1-Name by Song-1-Artist", "Song-2-Name by Song-2-Artist", ...]
|
||||
and you need to return a list formatted in the same way.
|
||||
|
||||
When recommending songs, consider the genre, tempo, and mood of the input
|
||||
songs to suggest similar ones that align with the user's tastes. Also, it
|
||||
is important to mix up the artists, don't only give the same artists that
|
||||
are already in the queue. If you cannot find {number} songs that match the
|
||||
criteria or encounter any issues, return the list ["NOTHING FOUND"].
|
||||
|
||||
Please be sure to also only use characters A-Z, a-z, 0-9, and spaces in the
|
||||
song and artist names. Do not include escape/special characters, emojis, or
|
||||
quotes in the output.
|
||||
|
||||
INPUT: {input_list}
|
||||
{input_list}
|
||||
""",
|
||||
}
|
||||
},
|
||||
],
|
||||
model="gpt-4o-mini",
|
||||
model=AI_MODEL,
|
||||
)
|
||||
.choices[0]
|
||||
.message.content.strip()
|
||||
@ -47,7 +59,7 @@ async def add_song_recommendations(
|
||||
if completion == '["NOTHING FOUND"]':
|
||||
if retries <= 3:
|
||||
await add_song_recommendations(
|
||||
openai_client, bot_user, player, number, inputs, retries + 1
|
||||
bot_user, player, number, inputs, retries + 1
|
||||
)
|
||||
else:
|
||||
return False
|
||||
|
@ -8,7 +8,7 @@ import sys
|
||||
import discord
|
||||
import logging
|
||||
import requests
|
||||
from datetime import datetime
|
||||
from groq import Groq
|
||||
from colorlog import ColoredFormatter
|
||||
|
||||
log_level = logging.DEBUG
|
||||
@ -39,7 +39,8 @@ SPOTIFY_CLIENT_ID = None
|
||||
SPOTIFY_CLIENT_SECRET = None
|
||||
GENIUS_CLIENT_ID = None
|
||||
GENIUS_CLIENT_SECRET = None
|
||||
OPENAI_API_KEY = None
|
||||
AI_CLIENT = None
|
||||
AI_MODEL = None
|
||||
LAVALINK_HOST = None
|
||||
LAVALINK_PORT = None
|
||||
LAVALINK_PASSWORD = None
|
||||
@ -82,12 +83,13 @@ schema = {
|
||||
},
|
||||
"required": ["genius_client_id", "genius_client_secret"],
|
||||
},
|
||||
"openai": {
|
||||
"ai": {
|
||||
"type": "object",
|
||||
"properties": {
|
||||
"openai_api_key": {"type": "string"},
|
||||
"service": {"enum": ["openai", "groq"]},
|
||||
"api_key": {"type": "string"},
|
||||
},
|
||||
"required": ["openai_api_key"],
|
||||
"required": ["service"],
|
||||
},
|
||||
"lavalink": {
|
||||
"type": "object",
|
||||
@ -144,9 +146,9 @@ genius:
|
||||
genius_client_id:
|
||||
genius_client_secret:
|
||||
|
||||
openai:
|
||||
openai_api_key:
|
||||
"""
|
||||
ai:
|
||||
service:
|
||||
api_key: """
|
||||
)
|
||||
|
||||
sys.exit(
|
||||
@ -160,7 +162,7 @@ openai:
|
||||
|
||||
# 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, 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
|
||||
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, AI_CLIENT, AI_MODEL, LAVALINK_HOST, LAVALINK_PORT, LAVALINK_PASSWORD
|
||||
config = yaml.safe_load(file_contents)
|
||||
|
||||
try:
|
||||
@ -270,17 +272,24 @@ def validate_config(file_contents):
|
||||
)
|
||||
|
||||
#
|
||||
# If the OPENAI section is present, make sure the API key is valid
|
||||
# If the AI section is present, make sure the API key is valid
|
||||
#
|
||||
|
||||
if "openai" in config:
|
||||
client = openai.OpenAI(api_key=config["openai"]["openai_api_key"])
|
||||
if "ai" in config:
|
||||
if config["ai"]["service"] == "openai":
|
||||
client = openai.OpenAI(api_key=config["ai"]["api_key"])
|
||||
model = "gpt-4o-mini"
|
||||
elif config["ai"]["service"] == "groq":
|
||||
client = Groq(api_key=config["ai"]["api_key"])
|
||||
model = "llama-3.3-70b-specdec"
|
||||
|
||||
try:
|
||||
client.models.list()
|
||||
OPENAI_API_KEY = config["openai"]["openai_api_key"]
|
||||
AI_CLIENT = client
|
||||
AI_MODEL = model
|
||||
except openai.AuthenticationError:
|
||||
LOG.critical(
|
||||
"Error in config.yaml file: OpenAI API key is invalid"
|
||||
"Error in config.yaml file: OpenAI/Groq API key is invalid"
|
||||
)
|
||||
|
||||
# Set appropriate values for all non-optional variables
|
||||
|
@ -22,5 +22,6 @@ genius:
|
||||
genius_client_id:
|
||||
genius_client_secret:
|
||||
|
||||
openai:
|
||||
openai_api_key:
|
||||
ai:
|
||||
service:
|
||||
api_key:
|
@ -6,4 +6,5 @@ openai==1.56.0
|
||||
requests==2.32.3
|
||||
lyricsgenius==3.2.0
|
||||
PyYAML==6.0.1
|
||||
jsonschema==4.23.0
|
||||
jsonschema==4.23.0
|
||||
groq==0.18.0
|
Loading…
x
Reference in New Issue
Block a user