/[sudobot]/branches/8.x/src/commands/settings/RestartCommand.ts
ViewVC logotype

Contents of /branches/8.x/src/commands/settings/RestartCommand.ts

Parent Directory Parent Directory | Revision Log Revision Log


Revision 577 - (show annotations)
Mon Jul 29 18:52:37 2024 UTC (8 months ago) by rakinar2
File MIME type: application/typescript
File size: 7430 byte(s)
chore: add old version archive branches (2.x to 9.x-dev)
1 /**
2 * This file is part of SudoBot.
3 *
4 * Copyright (C) 2021-2023 OSN Developers.
5 *
6 * SudoBot is free software; you can redistribute it and/or modify it
7 * under the terms of the GNU Affero General Public License as published by
8 * the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
10 *
11 * SudoBot is distributed in the hope that it will be useful, but
12 * WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU Affero General Public License for more details.
15 *
16 * You should have received a copy of the GNU Affero General Public License
17 * along with SudoBot. If not, see <https://www.gnu.org/licenses/>.
18 */
19
20 import {
21 ActionRowBuilder,
22 ButtonBuilder,
23 ButtonStyle,
24 CacheType,
25 ChatInputCommandInteraction,
26 Interaction,
27 SlashCommandBuilder,
28 Snowflake
29 } from "discord.js";
30 import { writeFile } from "fs/promises";
31 import path from "path";
32 import Command, { CommandMessage, CommandReturn, ValidationRule } from "../../core/Command";
33 import { GatewayEventListener } from "../../decorators/GatewayEventListener";
34 import { HasEventListeners } from "../../types/HasEventListeners";
35 import { logError } from "../../utils/Logger";
36 import { sudoPrefix } from "../../utils/utils";
37
38 export default class RestartCommand extends Command implements HasEventListeners {
39 public readonly name = "restart";
40 public readonly validationRules: ValidationRule[] = [];
41 public readonly aliases = ["reboot"];
42 public readonly systemAdminOnly = true;
43 public readonly slashCommandBuilder = new SlashCommandBuilder().addStringOption(option =>
44 option.setName("credential_key").setDescription("The key to authenticate with the credentials server, if needed")
45 );
46 public readonly keys: string[] = [];
47
48 public readonly description = "Restarts the bot.";
49
50 @GatewayEventListener("interactionCreate")
51 async onInteractionCreate(interaction: Interaction<CacheType>) {
52 if (!interaction.isButton()) {
53 return;
54 }
55
56 const { customId } = interaction;
57
58 if (!customId.startsWith("restart__yes__") && !customId.startsWith("restart__no__")) {
59 return;
60 }
61
62 const [, , guildId, channelId, userId, keyId] = customId.split("__");
63 const numericKeyId = keyId ? parseInt(keyId) : NaN;
64 const isNaN = Number.isNaN(numericKeyId);
65 const key = !isNaN ? this.keys.at(parseInt(keyId)) : null;
66
67 if (!isNaN && !key) {
68 return;
69 }
70
71 if (!guildId || !channelId || !userId) {
72 return;
73 }
74
75 if (interaction.guildId !== guildId || interaction.channelId !== channelId) {
76 return;
77 }
78
79 if (interaction.user.id !== userId) {
80 await interaction
81 .reply({
82 ephemeral: true,
83 content: "That's not under your control!"
84 })
85 .catch(logError);
86 return;
87 }
88
89 if (customId.startsWith("restart__yes__")) {
90 const buttons = this.buildButtons(guildId, channelId, userId).map(button => button.setDisabled(true));
91
92 await interaction.update({
93 embeds: [
94 {
95 color: 0x007bff,
96 title: "System Restart",
97 description: `${this.emoji("loading")} Restarting${key ? " (with one time 2FA code)" : ""}...`
98 }
99 ],
100 components: [new ActionRowBuilder<ButtonBuilder>().addComponents(...buttons)]
101 });
102
103 const json = JSON.stringify(
104 {
105 guildId,
106 channelId,
107 messageId: interaction.message.id,
108 time: Date.now(),
109 key
110 },
111 null,
112 4
113 );
114
115 await writeFile(path.join(sudoPrefix("tmp", true), "restart.json"), json);
116 process.exit(this.client.configManager.systemConfig.restart_exit_code);
117 }
118
119 if (customId.startsWith("restart__no__")) {
120 const buttons = this.buildButtons(guildId, channelId, userId).map(button => button.setDisabled(true));
121
122 await interaction.update({
123 embeds: [
124 {
125 color: 0xf14a60,
126 title: "System Restart",
127 description: "Operation cancelled."
128 }
129 ],
130 components: [new ActionRowBuilder<ButtonBuilder>().addComponents(...buttons)]
131 });
132 }
133 }
134
135 async execute(message: CommandMessage): Promise<CommandReturn> {
136 if (
137 process.env.CREDENTIAL_SERVER &&
138 (!(message instanceof ChatInputCommandInteraction) || !message.options.getString("credential_key"))
139 ) {
140 await this.error(message, "Please enter the credential server 2FA code to restart the bot!");
141 return;
142 }
143
144 const mfaKey = message instanceof ChatInputCommandInteraction ? message.options.getString("credential_key") : "";
145
146 const reply = await this.deferredReply(message, {
147 embeds: [
148 {
149 color: 0x007bff,
150 title: "System Restart",
151 description: "Are you sure you want to restart the entire system? The bot might go offline for some time."
152 }
153 ],
154 components: [
155 new ActionRowBuilder<ButtonBuilder>().addComponents(
156 ...this.buildButtons(message.guildId!, message.channelId!, message.member!.user.id, mfaKey)
157 )
158 ]
159 });
160
161 if (reply) {
162 setTimeout(() => {
163 reply.edit({
164 embeds: [
165 {
166 color: 0xf14a60,
167 title: "System Restart",
168 description: "Operation cancelled due to inactivity."
169 }
170 ],
171 components: [
172 new ActionRowBuilder<ButtonBuilder>().addComponents(
173 ...this.buildButtons(message.guildId!, message.channelId!, message.member!.user.id, mfaKey, true)
174 )
175 ]
176 });
177 }, 180_000); // 3 minutes
178 }
179 }
180
181 buildButtons(guildId: Snowflake, channelId: Snowflake, userId: Snowflake, key?: string | null, disable?: boolean) {
182 let keyId = key ? this.keys.indexOf(key) : null;
183
184 if (keyId === -1 && key) {
185 this.keys.push(key);
186 keyId = this.keys.length - 1;
187 }
188
189 return [
190 new ButtonBuilder()
191 .setCustomId(`restart__yes__${guildId}__${channelId}__${userId}__${keyId}`)
192 .setLabel("Restart")
193 .setStyle(ButtonStyle.Success)
194 .setDisabled(disable ?? false),
195 new ButtonBuilder()
196 .setCustomId(`restart__no__${guildId}__${channelId}__${userId}__${keyId}`)
197 .setLabel("Cancel")
198 .setStyle(ButtonStyle.Danger)
199 .setDisabled(disable ?? false)
200 ];
201 }
202 }

[email protected]
ViewVC Help
Powered by ViewVC 1.1.26