This repository has been archived by the owner on Dec 21, 2023. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 95
/
Copy pathcommand_handler.go
159 lines (142 loc) · 5.08 KB
/
command_handler.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
package aternos_discord_bot
import (
"context"
"github.com/bwmarrin/discordgo"
aternos "github.com/sleeyax/aternos-api"
"github.com/sleeyax/aternos-discord-bot/database"
"github.com/sleeyax/aternos-discord-bot/database/models"
"github.com/sleeyax/aternos-discord-bot/message"
"strings"
)
// handleCommands responds to incoming interactive commands on discord.
func (ab *Bot) handleCommands(s *discordgo.Session, i *discordgo.InteractionCreate) {
command := i.ApplicationCommandData()
// wrap functions around our utilities to make life easier
sendText := func(content string) {
respondWithText(s, i, content)
}
sendHiddenText := func(content string) {
respondWithHiddenText(s, i, content)
}
sendErrorText := func(content string, err error) {
respondWithError(s, i, content, err)
}
switch command.Name {
case HelpCommand:
sendHiddenText(message.FormatDefault(faq))
case PingCommand:
sendHiddenText(message.FormatDefault("Pong!"))
case ConfigureCommand:
options := optionsToMap(command.Options)
err := ab.Database.UpdateServerSettings(&models.ServerSettings{
GuildID: i.GuildID,
SessionCookie: options[SessionOption].StringValue(),
ServerCookie: options[ServerOption].StringValue(),
})
if err != nil {
sendErrorText("Failed to save configuration.", err)
break
}
sendText(message.FormatSuccess("Configuration changed successfully."))
case StatusCommand:
fallthrough
case InfoCommand:
fallthrough
case PlayersCommand:
fallthrough
case StartCommand:
fallthrough
case StopCommand:
w, err := ab.getWorker(i.GuildID)
if err != nil {
if err == database.ErrDataNotFound {
sendText(message.FormatWarning("Bot setup incomplete. Use `/configure` to configure the bot."))
} else {
sendErrorText("Failed to get worker", err)
}
break
}
serverInfo, err := w.GetServerInfo()
if err != nil {
if err == aternos.UnauthenticatedError {
sendText(message.FormatError("Invalid credentials. Use `/configure` to reconfigure the bot."))
} else if err == aternos.ForbiddenError {
sendText(message.FormatError("Access forbidden. Please try again later."))
} else {
sendErrorText("Failed to get server info", err)
}
break
}
switch command.Name {
case InfoCommand:
respondWithEmbeds(s, i, []*discordgo.MessageEmbed{
message.CreateServerInfoEmbed(serverInfo),
})
case StatusCommand:
sendText(message.FormatInfo("Server '%s' is currently **%s**.", serverInfo.Name, serverInfo.StatusLabel))
case PlayersCommand:
if len(serverInfo.PlayerList) == 0 {
sendText(message.FormatInfo("No players online right now."))
break
}
sendText(message.FormatInfo("Active players: %s.", strings.Join(serverInfo.PlayerList, ", ")))
case StopCommand:
fallthrough
case StartCommand:
// connect to WSS
if err = w.Init(); err != nil {
sendErrorText("Failed to initialize worker! See `/help` or try again later.", err)
break
}
// stop server
if command.Name == StopCommand {
if serverInfo.Status == aternos.Stopping || serverInfo.Status == aternos.Offline {
sendText(message.FormatInfo("Server already stopped! Type `/status` or `/info` to view the status."))
break
}
sendText(message.FormatInfo("Stopping the server, please wait..."))
// TODO: check if we need to put stop() in a goroutine for better performance
if err = w.Stop(); err != nil {
s.ChannelMessageSend(i.ChannelID, message.FormatError("Failed to stop the server"))
w.Log(err.Error())
}
break
}
// start server
if serverInfo.Status != aternos.Offline && serverInfo.Status != aternos.Stopping {
sendText(message.FormatInfo("Server already started! Type `/status` or `/info` to view the status."))
break
}
sendText(message.FormatInfo("Starting the server, please wait..."))
ctx, cancel := context.WithCancel(context.Background())
go w.On(ctx, func(messageType string, info *aternos.ServerInfo) {
switch messageType {
case "ready":
if command.Name == StartCommand {
if err = w.Start(); err != nil {
s.ChannelMessageSend(i.ChannelID, message.FormatError("Failed to start! Reconfigure the bot with `/configure` and try again. See `/help` if the problem persists."))
w.Log(err.Error())
cancel() // cancel the worker's goroutine
break
}
}
case "status":
switch info.Status {
case aternos.Offline:
fallthrough
case aternos.Online:
notification, _ := message.CreateServerStatusNotificationEmbed(info)
s.ChannelMessageSendEmbed(i.ChannelID, notification)
case aternos.Preparing:
s.ChannelMessageSend(i.ChannelID, message.FormatInfo("Waiting in queue (%d/%d, %s)...", info.Queue.Position, info.Queue.Count, info.Queue.Time))
w.Log("Waiting in queue...")
}
case "connection_error":
s.ChannelMessageSend(i.ChannelID, message.FormatError("Failed to initialize worker (websocket connection timeout)! See `/help` or try again later."))
}
})
}
default:
sendText(message.FormatWarning("Command unavailable. Please try again later or refresh your discord client `CTRL + R`"))
}
}