mastermind-bot/main.go
2026-01-19 10:38:13 +01:00

215 lines
5.9 KiB
Go

package main
import (
"encoding/json"
"log"
"os"
"os/signal"
"strconv"
"strings"
dg "github.com/bwmarrin/discordgo"
)
type Settings struct {
Token string `json:"token"`
Server string `json:"server"`
Leaderboard [][]string `json:"leaderboard"`
Color int `json:"color"`
students map[string]int
rankings []map[string]struct{}
}
var stg Settings
func init() {
file, err := os.ReadFile("settings.json")
if err != nil {
log.Fatalf("Error while reading settings.json : %s", err)
}
if err := json.Unmarshal(file, &stg); err != nil {
log.Fatalf("Error while reading settings.json : %s", err.Error())
}
stg.students = make(map[string]int, len(stg.Leaderboard))
for i, r := range stg.Leaderboard {
stg.rankings = append(stg.rankings, make(map[string]struct{}, len(r)))
for _, s := range r {
stg.rankings[i][s] = struct{}{}
stg.students[s] = i
}
}
}
var descs = map[string]*dg.ApplicationCommand{
"ping": &dg.ApplicationCommand{
Name: "ping",
Description: "Replies with pong",
DescriptionLocalizations: &map[dg.Locale]string{
dg.French: "Répond pong",
},
},
"sub": &dg.ApplicationCommand{
Name: "sub",
Description: "Submit a guess",
DescriptionLocalizations: &map[dg.Locale]string{
dg.French: "Envoie un essais pour deviner",
},
// /!\ Make sure to update handler if other options are added
// Guess options are added in init function
// It is assumed in handler that there aren't other options
Options: []*dg.ApplicationCommandOption{},
},
}
// Just in case...
var duplicateNames = []string{
"bis", "ter", "quater", "quinquies", "sexies", "septies", "octies", "nonies", "decies", "undecies", "duodecies", "terdecies", "quaterdecies", "quindecies", "sexdecies", "septdecies", "octodecies", "novodecies", "vicies",
}
func init() {
for rank, students := range stg.rankings {
rankName := strconv.Itoa(rank + 1)
for i := range len(students) {
opt := dg.ApplicationCommandOption{
Type: dg.ApplicationCommandOptionUser,
Description: "Guess for student " + rankName,
DescriptionLocalizations: map[dg.Locale]string{
dg.French: "Supposition pour l'élève " + rankName,
},
Required: true,
}
if i == 0 {
opt.Name = rankName
} else {
opt.Name = rankName + "_" + duplicateNames[i-1]
}
descs["sub"].Options = append(descs["sub"].Options, &opt)
}
}
}
func send_error(s *dg.Session, i *dg.Interaction, reason string) {
s.InteractionRespond(i, &dg.InteractionResponse{
Type: dg.InteractionResponseChannelMessageWithSource,
Data: &dg.InteractionResponseData{
Embeds: []*dg.MessageEmbed{
{
Title: "⚠️ Error",
Description: reason,
Color: 0xFF0000,
},
},
},
})
}
// Handlers are separated because we need to be able to access descs
var handlers = map[string]func(*dg.Session, *dg.InteractionCreate){
"ping": func(s *dg.Session, i *dg.InteractionCreate) {
s.InteractionRespond(i.Interaction, &dg.InteractionResponse{
Type: dg.InteractionResponseChannelMessageWithSource,
Data: &dg.InteractionResponseData{
Content: ":ping_pong: Pong !",
},
})
},
"sub": func(s *dg.Session, i *dg.InteractionCreate) {
red := 0
green := 0
guesses := []string{}
ids := make(map[string]struct{}, len(descs["sub"].Options))
for _, opt := range descs["sub"].Options {
val := i.ApplicationCommandData().GetOption(opt.Name)
if val == nil {
send_error(s, i.Interaction, "Missing option")
return
}
if val.Type != dg.ApplicationCommandOptionUser {
send_error(s, i.Interaction, "Invalid option type")
return
}
guess := val.Value.(string)
if _, ok := ids[guess]; ok {
send_error(s, i.Interaction, "Les doublons ne sont pas autorisés !")
return
} else {
ids[guess] = struct{}{}
}
grstr, _, _ := strings.Cut(opt.Name, "_")
guessRank, err := strconv.Atoi(grstr)
guessRank -= 1
if err != nil || guessRank < 0 || stg.rankings[guessRank] == nil {
send_error(s, i.Interaction, "Invalid option name")
}
if num, ok := stg.students[guess]; ok {
if num == guessRank {
green += 1
} else {
red += 1
}
}
guesses = append(guesses, strconv.Itoa(guessRank+1)+": <@"+guess+">")
}
var cmdAuthor string
if i.Member != nil {
cmdAuthor = i.Member.DisplayName()
} else {
cmdAuthor = i.User.DisplayName()
}
var title string
if i.Locale == dg.French {
title = "Tentative par " + cmdAuthor
} else {
title = "Guess by " + cmdAuthor
}
s.InteractionRespond(i.Interaction, &dg.InteractionResponse{
Type: dg.InteractionResponseChannelMessageWithSource,
Data: &dg.InteractionResponseData{
Embeds: []*dg.MessageEmbed{
{
Title: title,
Description: strings.Join(guesses, "\n") + "\n\n**" + strconv.Itoa(green) + " 🟩**\n**" + strconv.Itoa(red) + " 🟥**",
Color: stg.Color,
},
},
},
})
},
}
func main() {
s, _ := dg.New("Bot " + stg.Token)
s.AddHandler(func(s *dg.Session, r *dg.Ready) {
log.Printf("Ready!")
})
err := s.Open()
if err != nil {
log.Panicf("Failed to create session : %s", err)
}
s.AddHandler(func(s *dg.Session, i *dg.InteractionCreate) {
if i.Type != dg.InteractionApplicationCommand {
log.Printf("Warning : Received unknown interaction type (%s)", i.Type.String())
}
if h, ok := handlers[i.ApplicationCommandData().Name]; ok {
h(s, i)
log.Printf("Reveived command \"%s\"", i.ApplicationCommandData().Name)
} else {
log.Printf("Warning : Received unknown command \"%s\" (id:%s)", i.ApplicationCommandData().Name, i.ApplicationCommandData().ID)
}
})
// Create commands
for _, d := range descs {
if _, err := s.ApplicationCommandCreate(s.State.User.ID, stg.Server, d); err != nil {
log.Panicf("Failed to create command : %s", err)
}
}
defer s.Close()
stop := make(chan os.Signal, 1)
signal.Notify(stop, os.Interrupt)
log.Println("Press Ctrl+C to exit")
<-stop
}