2022-10-29 05:08:17 +02:00
|
|
|
package discord
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"os"
|
|
|
|
"os/signal"
|
|
|
|
|
|
|
|
"github.com/bwmarrin/discordgo"
|
|
|
|
"github.com/stretchr/testify/mock"
|
2023-06-19 09:08:24 +02:00
|
|
|
"github.com/yeslayla/birdbot-common/common"
|
2023-06-18 01:38:47 +02:00
|
|
|
"github.com/yeslayla/birdbot/persistence"
|
2022-10-29 05:08:17 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type Discord struct {
|
|
|
|
mock.Mock
|
|
|
|
|
2023-06-16 03:25:46 +02:00
|
|
|
guildID string
|
|
|
|
applicationID string
|
|
|
|
session *discordgo.Session
|
|
|
|
|
|
|
|
commands map[string]*discordgo.ApplicationCommand
|
|
|
|
commandHandlers map[string]func(session *discordgo.Session, i *discordgo.InteractionCreate)
|
2022-10-29 05:08:17 +02:00
|
|
|
|
2023-06-18 01:38:47 +02:00
|
|
|
db persistence.Database
|
|
|
|
|
2022-10-29 05:08:17 +02:00
|
|
|
// Signal for shutdown
|
|
|
|
stop chan os.Signal
|
|
|
|
}
|
|
|
|
|
|
|
|
// New creates a new Discord session
|
2023-06-18 01:38:47 +02:00
|
|
|
func New(applicationID string, guildID string, token string, db persistence.Database) *Discord {
|
2022-10-29 05:08:17 +02:00
|
|
|
|
|
|
|
// Create Discord Session
|
|
|
|
session, err := discordgo.New(fmt.Sprint("Bot ", token))
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Failed to create Discord session: %v", err)
|
|
|
|
}
|
2023-06-16 03:25:46 +02:00
|
|
|
session.ShouldReconnectOnError = true
|
2022-10-29 05:08:17 +02:00
|
|
|
return &Discord{
|
2023-06-18 01:38:47 +02:00
|
|
|
db: db,
|
|
|
|
|
2023-06-16 03:25:46 +02:00
|
|
|
session: session,
|
|
|
|
applicationID: applicationID,
|
|
|
|
guildID: guildID,
|
|
|
|
commands: make(map[string]*discordgo.ApplicationCommand),
|
|
|
|
commandHandlers: make(map[string]func(*discordgo.Session, *discordgo.InteractionCreate)),
|
2022-10-29 05:08:17 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Run opens the Discod session until exit
|
|
|
|
func (discord *Discord) Run() error {
|
|
|
|
|
|
|
|
if err := discord.session.Open(); err != nil {
|
|
|
|
return fmt.Errorf("failed to open Discord session: %v", err)
|
|
|
|
}
|
|
|
|
defer discord.session.Close()
|
|
|
|
|
2023-06-16 03:25:46 +02:00
|
|
|
// Register command handler
|
|
|
|
discord.session.AddHandler(func(session *discordgo.Session, i *discordgo.InteractionCreate) {
|
2023-06-22 01:44:49 +02:00
|
|
|
if i.GuildID != discord.guildID || i.Interaction.Type != discordgo.InteractionApplicationCommand {
|
2023-06-16 03:25:46 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if handler, ok := discord.commandHandlers[i.ApplicationCommandData().Name]; ok {
|
|
|
|
handler(session, i)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2022-10-29 05:08:17 +02:00
|
|
|
// Keep alive
|
|
|
|
discord.stop = make(chan os.Signal, 1)
|
|
|
|
signal.Notify(discord.stop, os.Interrupt)
|
|
|
|
<-discord.stop
|
2023-06-16 03:25:46 +02:00
|
|
|
|
|
|
|
discord.ClearCommands()
|
|
|
|
|
2022-10-29 05:08:17 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Stop tells the Discord session to exit
|
|
|
|
func (discord *Discord) Stop() {
|
|
|
|
discord.stop <- os.Kill
|
|
|
|
}
|
|
|
|
|
|
|
|
// OnReady registers a handler for when the Discord session is ready
|
|
|
|
func (discord *Discord) OnReady(handler func(*Discord)) {
|
|
|
|
discord.session.AddHandler(func(s *discordgo.Session, r *discordgo.Ready) {
|
|
|
|
handler(discord)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// OnEventCreate registers a handler when a guild scheduled event is created
|
2023-03-31 05:51:05 +02:00
|
|
|
func (discord *Discord) OnEventCreate(handler func(*Discord, common.Event)) {
|
2022-10-29 05:08:17 +02:00
|
|
|
discord.session.AddHandler(func(s *discordgo.Session, r *discordgo.GuildScheduledEventCreate) {
|
|
|
|
if r.GuildID != discord.guildID {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
event := NewEvent(r.GuildScheduledEvent)
|
|
|
|
handler(discord, event)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// OnEventDelete registers a handler when a guild scheduled event is deleted
|
2023-03-31 05:51:05 +02:00
|
|
|
func (discord *Discord) OnEventDelete(handler func(*Discord, common.Event)) {
|
2022-10-29 05:08:17 +02:00
|
|
|
discord.session.AddHandler(func(s *discordgo.Session, r *discordgo.GuildScheduledEventDelete) {
|
|
|
|
if r.GuildID != discord.guildID {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
event := NewEvent(r.GuildScheduledEvent)
|
|
|
|
handler(discord, event)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// OnEventUpdate registers a handler when a guild scheduled event is updated
|
2023-03-31 05:51:05 +02:00
|
|
|
func (discord *Discord) OnEventUpdate(handler func(*Discord, common.Event)) {
|
2022-10-29 05:08:17 +02:00
|
|
|
discord.session.AddHandler(func(s *discordgo.Session, r *discordgo.GuildScheduledEventUpdate) {
|
|
|
|
if r.GuildID != discord.guildID {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
event := NewEvent(r.GuildScheduledEvent)
|
|
|
|
handler(discord, event)
|
|
|
|
})
|
|
|
|
}
|
2023-03-05 19:08:18 +01:00
|
|
|
|
2023-06-18 01:38:47 +02:00
|
|
|
// OnMessageRecieved registers a handler when a message is recieved
|
|
|
|
func (discord *Discord) OnMessageRecieved(handler func(*Discord, string, common.User, string)) {
|
|
|
|
discord.session.AddHandler(func(s *discordgo.Session, r *discordgo.MessageCreate) {
|
2023-06-19 09:08:24 +02:00
|
|
|
if r.GuildID != discord.guildID || r.Author.Bot {
|
2023-06-18 01:38:47 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
handler(discord, r.ChannelID, NewUser(r.Author), r.Content)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-03-05 19:08:18 +01:00
|
|
|
func (discord *Discord) SetStatus(status string) {
|
|
|
|
if err := discord.session.UpdateGameStatus(0, status); err != nil {
|
|
|
|
log.Fatal("Failed to update status: ", err)
|
|
|
|
}
|
|
|
|
}
|