Refactor code, remove commands and use guild commands instead of global

This commit is contained in:
2024-02-26 10:09:46 +01:00
parent 33616baa8b
commit bdef01ff8b
2 changed files with 133 additions and 99 deletions

213
main.go
View File

@ -16,6 +16,10 @@ import (
var config Config var config Config
func init() {
log.SetFlags(log.LstdFlags | log.Lshortfile)
}
func init() { func init() {
loadedConfig, err := Load() loadedConfig, err := Load()
if err != nil { if err != nil {
@ -60,23 +64,26 @@ func GetPostsFromReddit(subreddit string) (string, error) {
func handleRedditCommand(s *discordgo.Session, i *discordgo.InteractionCreate, subreddit string) { func handleRedditCommand(s *discordgo.Session, i *discordgo.InteractionCreate, subreddit string) {
post, err := GetPostsFromReddit(subreddit) post, err := GetPostsFromReddit(subreddit)
if err != nil { if err = s.InteractionRespond(i.Interaction, &discordgo.InteractionResponse{
s.InteractionRespond(i.Interaction, &discordgo.InteractionResponse{ Type: discordgo.InteractionResponseChannelMessageWithSource,
Type: discordgo.InteractionResponseChannelMessageWithSource, Data: &discordgo.InteractionResponseData{
Data: &discordgo.InteractionResponseData{ Content: fmt.Sprintf("Cannot get a random post: %v", err),
Content: fmt.Sprintf("Cannot get a random post: %v", err), Flags: discordgo.MessageFlagsEphemeral,
Flags: discordgo.MessageFlagsEphemeral, },
}, }); err != nil {
}) log.Println("Failed to respond to interaction:", err)
return return
} }
s.InteractionRespond(i.Interaction, &discordgo.InteractionResponse{ if err := s.InteractionRespond(i.Interaction, &discordgo.InteractionResponse{
Type: discordgo.InteractionResponseChannelMessageWithSource, Type: discordgo.InteractionResponseChannelMessageWithSource,
Data: &discordgo.InteractionResponseData{ Data: &discordgo.InteractionResponseData{
Content: post, Content: post,
}, },
}) }); err != nil {
log.Println("Failed to respond to interaction:", err)
return
}
} }
var ( var (
@ -97,22 +104,6 @@ var (
Name: "thighs", Name: "thighs",
Description: "Sends thighs from /r/ZettaiRyouiki", Description: "Sends thighs from /r/ZettaiRyouiki",
}, },
{
Name: "help",
Description: "Sends help message",
},
{
Name: "echo",
Description: "Echoes your message",
Options: []*discordgo.ApplicationCommandOption{
{
Type: discordgo.ApplicationCommandOptionString,
Name: "message",
Description: "The message to echo",
Required: true,
},
},
},
} }
commandHandlers = map[string]func(s *discordgo.Session, i *discordgo.InteractionCreate){ commandHandlers = map[string]func(s *discordgo.Session, i *discordgo.InteractionCreate){
@ -135,68 +126,45 @@ var (
"thighs": func(s *discordgo.Session, i *discordgo.InteractionCreate) { "thighs": func(s *discordgo.Session, i *discordgo.InteractionCreate) {
handleRedditCommand(s, i, "ZettaiRyouiki") handleRedditCommand(s, i, "ZettaiRyouiki")
}, },
// Help command
"help": func(s *discordgo.Session, i *discordgo.InteractionCreate) {
// Send the help message to the channel where the command was used
s.InteractionRespond(i.Interaction, &discordgo.InteractionResponse{
Type: discordgo.InteractionResponseChannelMessageWithSource,
Data: &discordgo.InteractionResponseData{
Content: "**Commands**\n\n/dank_memes - Sends dank meme from /r/GoodAnimemes\n/waifus - Sends waifu from /r/WatchItForThePlot\n/milkers - Sends milkers from /r/RetrousseTits\n/thighs - Sends thighs from /r/ZettaiRyouiki\n/help - Sends help message\n/echo - Echoes your message",
},
})
},
// Echo command // Echo command
"echo": func(s *discordgo.Session, i *discordgo.InteractionCreate) { "echo": func(s *discordgo.Session, i *discordgo.InteractionCreate) {
// Check if the user provided a message // Check if the user provided a message
if len(i.ApplicationCommandData().Options) == 0 { if len(i.ApplicationCommandData().Options) == 0 {
// If not, send an ephemeral message to the user // If not, send an ephemeral message to the user
s.InteractionRespond(i.Interaction, &discordgo.InteractionResponse{ err := s.InteractionRespond(i.Interaction, &discordgo.InteractionResponse{
Type: discordgo.InteractionResponseChannelMessageWithSource, Type: discordgo.InteractionResponseChannelMessageWithSource,
Data: &discordgo.InteractionResponseData{ Data: &discordgo.InteractionResponseData{
Content: "You need to provide a message!", Content: "You need to provide a message!",
Flags: discordgo.MessageFlagsEphemeral, Flags: discordgo.MessageFlagsEphemeral,
}, },
}) })
return if err != nil {
} log.Println("Failed to respond to interaction:", err)
return
// Check that the option contains text }
if i.ApplicationCommandData().Options[0].Type != discordgo.ApplicationCommandOptionString {
// If not, send an ephemeral message to the user
s.InteractionRespond(i.Interaction, &discordgo.InteractionResponse{
Type: discordgo.InteractionResponseChannelMessageWithSource,
Data: &discordgo.InteractionResponseData{
Content: "The message needs to be text!",
Flags: discordgo.MessageFlagsEphemeral,
},
})
return
} }
// Check that the option is not empty // Check that the option is not empty
if i.ApplicationCommandData().Options[0].StringValue() == "" { if i.ApplicationCommandData().Options[0].StringValue() == "" {
// If not, send an ephemeral message to the user // If not, send an ephemeral message to the user
s.InteractionRespond(i.Interaction, &discordgo.InteractionResponse{ err := s.InteractionRespond(i.Interaction, &discordgo.InteractionResponse{
Type: discordgo.InteractionResponseChannelMessageWithSource, Type: discordgo.InteractionResponseChannelMessageWithSource,
Data: &discordgo.InteractionResponseData{ Data: &discordgo.InteractionResponseData{
Content: "The message cannot be empty!", Content: "The message cannot be empty!",
Flags: discordgo.MessageFlagsEphemeral, Flags: discordgo.MessageFlagsEphemeral,
}, },
}) })
return if err != nil {
log.Println("Failed to respond to interaction:", err)
return
}
} }
// Respond to the original message so we don't get "This interaction failed" error // Respond to the original message so we don't get "This interaction failed" error
s.InteractionRespond(i.Interaction, &discordgo.InteractionResponse{ if _, err := s.ChannelMessageSend(i.ChannelID, i.ApplicationCommandData().Options[0].StringValue()); err != nil {
Type: discordgo.InteractionResponseChannelMessageWithSource, log.Println("Failed to send message to channel:", err)
Data: &discordgo.InteractionResponseData{ return
Content: "love u", }
Flags: discordgo.MessageFlagsEphemeral,
},
})
// Send the message to the channel where the command was used
s.ChannelMessageSend(i.ChannelID, i.ApplicationCommandData().Options[0].StringValue())
}, },
} }
) )
@ -224,67 +192,100 @@ func onMessageCreate(s *discordgo.Session, m *discordgo.MessageCreate) {
log.Println("User:", user) log.Println("User:", user)
if m.Author.Username == user { if m.Author.Username == user {
log.Println("User is in allowedUsers") log.Println("User is in allowedUsers")
r, err := GetGPT3Response(m.Content, m.Author.Username) r, err := GenerateGPT4Response(m.Content, m.Author.Username)
if err != nil { if err != nil {
log.Println("Failed to get GPT-3 response:", err) log.Println("Failed to get OpenAI response:", err)
return return
} }
log.Println("GPT-3 response:", r) log.Println("OpenAI response:", r)
log.Println("Channel ID:", m.ChannelID) log.Println("Channel ID:", m.ChannelID)
s.ChannelMessageSend(m.ChannelID, r) _, err = s.ChannelMessageSend(m.ChannelID, r)
if err != nil {
log.Println("Failed to send message to channel:", err)
return
}
} }
} }
} }
if m.Mentions != nil { if m.Mentions != nil {
for _, mention := range m.Mentions { for _, mention := range m.Mentions {
if mention.ID == s.State.User.ID { if mention.ID == s.State.User.ID {
r, err := GetGPT3Response(m.Content, m.Author.Username) r, err := GenerateGPT4Response(m.Content, m.Author.Username)
if err != nil { if err != nil {
if strings.Contains(err.Error(), "prompt is too long") { if strings.Contains(err.Error(), "prompt is too long") {
s.ChannelMessageSend(m.ChannelID, "Message is too long!") if _, err := s.ChannelMessageSend(m.ChannelID, "Message is too long!"); err != nil {
log.Println("Failed to send message to channel:", err)
return
}
return return
} }
if strings.Contains(err.Error(), "prompt is too short") { if strings.Contains(err.Error(), "prompt is too short") {
s.ChannelMessageSend(m.ChannelID, "Message is too short!") if _, err := s.ChannelMessageSend(m.ChannelID, "Message is too short!"); err != nil {
log.Println("Failed to send message to channel:", err)
return
}
return return
} }
s.ChannelMessageSend(m.ChannelID, fmt.Sprintf("Brain broke :flushed: %v", err)) message, err := s.ChannelMessageSend(m.ChannelID, fmt.Sprintf("Brain broke :flushed: %v", err))
if err != nil {
log.Println("Failed to send message to channel:", err)
return
}
_ = message
return
}
_, err = s.ChannelMessageSend(m.ChannelID, r)
if err != nil {
log.Println("Failed to send message to channel:", err)
return return
} }
s.ChannelMessageSend(m.ChannelID, r)
} }
} }
} }
if strings.HasPrefix(strings.ToLower(m.Content), "lovibot") { if strings.HasPrefix(strings.ToLower(m.Content), "lovibot") {
r, err := GetGPT3Response(m.Content, m.Author.Username) r, err := GenerateGPT4Response(m.Content, m.Author.Username)
if err != nil { if err != nil {
if strings.Contains(err.Error(), "prompt is too long") { if strings.Contains(err.Error(), "prompt is too long") {
s.ChannelMessageSend(m.ChannelID, "Message is too long!") _, err := s.ChannelMessageSend(m.ChannelID, "Message is too long!")
if err != nil {
log.Println("Failed to send message to channel:", err)
return
}
return return
} }
if strings.Contains(err.Error(), "prompt is too short") { if strings.Contains(err.Error(), "prompt is too short") {
s.ChannelMessageSend(m.ChannelID, "Message is too short!") _, err := s.ChannelMessageSend(m.ChannelID, "Message is too short!")
if err != nil {
log.Println("Failed to send message to channel:", err)
return
}
return return
} }
s.ChannelMessageSend(m.ChannelID, fmt.Sprintf("Brain broke :flushed: %v", err)) message, err := s.ChannelMessageSend(m.ChannelID, fmt.Sprintf("Brain broke :flushed: %v", err))
if err != nil {
log.Println("Failed to send message to channel:", err)
return
}
_ = message
return
}
_, err = s.ChannelMessageSend(m.ChannelID, r)
if err != nil {
log.Println("Failed to send message to channel:", err)
return return
} }
s.ChannelMessageSend(m.ChannelID, r)
} }
} }
func main() { func main() {
// Print the token for debugging purposes.
discordToken := config.DiscordToken discordToken := config.DiscordToken
fmt.Println("Discord Token:", discordToken)
// Create a new Discord session using the provided bot token. // Create a new Discord session using the provided bot token.
session, err := discordgo.New("Bot " + discordToken) session, err := discordgo.New("Bot " + discordToken)
@ -295,7 +296,7 @@ func main() {
// Add a handler function to the discordgo.Session that is triggered when a slash command is received. // Add a handler function to the discordgo.Session that is triggered when a slash command is received.
session.AddHandler(func(s *discordgo.Session, i *discordgo.InteractionCreate) { session.AddHandler(func(s *discordgo.Session, i *discordgo.InteractionCreate) {
if h, ok := commandHandlers[i.ApplicationCommandData().Name]; ok { if h, ok := commandHandlers[i.ApplicationCommandData().Name]; ok {
log.Printf("Handling '%v' command.", i.ApplicationCommandData().Name) log.Printf("Handling '%v' command. %+v", i.ApplicationCommandData().Name, i.ApplicationCommandData())
h(s, i) h(s, i)
} }
}) })
@ -314,11 +315,55 @@ func main() {
log.Fatalf("Cannot open the session: %v", err) log.Fatalf("Cannot open the session: %v", err)
} }
// Register the commands. // Remove all existing commands.
log.Println("Adding commands...") appID := session.State.User.ID
log.Println("Removing existing commands from all servers for the bot", appID)
// Remove the commands for guild 98905546077241344
log.Println("Removing commands for Killyoy's server...")
commands, err := session.ApplicationCommands(appID, "98905546077241344")
if err != nil {
log.Panicf("Cannot get commands for guild 98905546077241344: %v", err)
}
for _, v := range commands {
err := session.ApplicationCommandDelete(session.State.User.ID, "98905546077241344", v.ID)
if err != nil {
log.Panicf("Cannot delete '%v' command: %v", v.Name, err)
}
log.Printf("Deleted '%v' command.", v.Name)
}
// Remove the commands for guild 341001473661992962
log.Println("Removing commands for TheLovinator's server...")
commands, err = session.ApplicationCommands(appID, "341001473661992962")
if err != nil {
log.Panicf("Cannot get commands for guild 341001473661992962: %v", err)
}
for _, v := range commands {
err := session.ApplicationCommandDelete(session.State.User.ID, "341001473661992962", v.ID)
if err != nil {
log.Panicf("Cannot delete '%v' command: %v", v.Name, err)
}
log.Printf("Deleted '%v' command.", v.Name)
}
// Register the commands for guild 98905546077241344
log.Println("Registering commands for Killyoy's server...")
registeredCommands := make([]*discordgo.ApplicationCommand, len(commands)) registeredCommands := make([]*discordgo.ApplicationCommand, len(commands))
for i, v := range commands { for i, v := range commands {
cmd, err := session.ApplicationCommandCreate(session.State.User.ID, "", v) cmd, err := session.ApplicationCommandCreate(session.State.User.ID, "98905546077241344", v)
if err != nil {
log.Panicf("Cannot create '%v' command: %v", v.Name, err)
}
registeredCommands[i] = cmd
log.Printf("Registered '%v' command.", cmd.Name)
}
// Register the commands for guild 341001473661992962
log.Println("Registering commands for TheLovinator's server...")
registeredCommands = make([]*discordgo.ApplicationCommand, len(commands))
for i, v := range commands {
cmd, err := session.ApplicationCommandCreate(session.State.User.ID, "341001473661992962", v)
if err != nil { if err != nil {
log.Panicf("Cannot create '%v' command: %v", v.Name, err) log.Panicf("Cannot create '%v' command: %v", v.Name, err)
} }

View File

@ -3,7 +3,6 @@ package main
import ( import (
"context" "context"
"fmt" "fmt"
"strings"
openai "github.com/sashabaranov/go-openai" openai "github.com/sashabaranov/go-openai"
) )
@ -11,21 +10,12 @@ import (
/* /*
Prompt is the Discord message that the user sent to the bot. Prompt is the Discord message that the user sent to the bot.
*/ */
func GetGPT3Response(prompt string, author string) (string, error) { func GenerateGPT4Response(prompt string, author string) (string, error) {
openAIToken := config.OpenAIToken openAIToken := config.OpenAIToken
if openAIToken == "" { if openAIToken == "" {
return "", fmt.Errorf("OPENAI_API_KEY is not set") return "", fmt.Errorf("OPENAI_API_KEY is not set")
} }
// Remove the mention from the prompt
prompt = strings.Replace(prompt, "@LoviBot ", "", -1)
// Remove the mention from the prompt
prompt = strings.Replace(prompt, "LoviBot ", "", -1)
// Remove the mention from the prompt
prompt = strings.Replace(prompt, "<@345000831499894795> ", "", -1)
// Print the prompt // Print the prompt
fmt.Println("Prompt:", author, ":", prompt) fmt.Println("Prompt:", author, ":", prompt)
@ -35,7 +25,7 @@ func GetGPT3Response(prompt string, author string) (string, error) {
} }
// Check if the prompt is too short // Check if the prompt is too short
if len(prompt) < 1 { if len(prompt) < 10 {
return "", fmt.Errorf("prompt is too short") return "", fmt.Errorf("prompt is too short")
} }
@ -71,7 +61,7 @@ func GetGPT3Response(prompt string, author string) (string, error) {
// Add additional information to the system message // Add additional information to the system message
if additionalInfo != "" { if additionalInfo != "" {
systemMessage = fmt.Sprintf("%s %s", systemMessage, additionalInfo) systemMessage = fmt.Sprintf("%s\n%s", systemMessage, additionalInfo)
} }
// Print the system message // Print the system message
@ -96,11 +86,10 @@ func GetGPT3Response(prompt string, author string) (string, error) {
) )
if err != nil { if err != nil {
return "", fmt.Errorf("failed to get response from GPT-3: %v", err) return "", fmt.Errorf("failed to get response from OpenAI: %w", err)
} }
ourResponse := resp.Choices[0].Message.Content ourResponse := resp.Choices[0].Message.Content
ourResponse = strings.Replace(ourResponse, "As a space communist, ", "", -1)
fmt.Println("Response:", ourResponse) fmt.Println("Response:", ourResponse)
return ourResponse, nil return ourResponse, nil