forked from syntaxbullet/AuroraBot-discord
feat: Add /config admin command to dynamically edit and save bot configuration.
This commit is contained in:
2
.gitignore
vendored
2
.gitignore
vendored
@@ -7,7 +7,7 @@ db-data
|
|||||||
|
|
||||||
node_modules
|
node_modules
|
||||||
|
|
||||||
src/config
|
config/
|
||||||
|
|
||||||
# output
|
# output
|
||||||
out
|
out
|
||||||
|
|||||||
68
src/commands/admin/config.ts
Normal file
68
src/commands/admin/config.ts
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
import { createCommand } from "@lib/utils";
|
||||||
|
import { SlashCommandBuilder, PermissionFlagsBits, ModalBuilder, TextInputBuilder, TextInputStyle, ActionRowBuilder, ModalSubmitInteraction } from "discord.js";
|
||||||
|
import { config, saveConfig } from "@lib/config";
|
||||||
|
import type { GameConfigType } from "@lib/config";
|
||||||
|
import { createSuccessEmbed, createErrorEmbed } from "@lib/embeds";
|
||||||
|
|
||||||
|
export const configCommand = createCommand({
|
||||||
|
data: new SlashCommandBuilder()
|
||||||
|
.setName("config")
|
||||||
|
.setDescription("Edit the bot configuration")
|
||||||
|
.setDefaultMemberPermissions(PermissionFlagsBits.Administrator),
|
||||||
|
execute: async (interaction) => {
|
||||||
|
console.log(`Config command executed by ${interaction.user.tag}`);
|
||||||
|
const replacer = (key: string, value: any) => {
|
||||||
|
if (typeof value === 'bigint') {
|
||||||
|
return value.toString();
|
||||||
|
}
|
||||||
|
return value;
|
||||||
|
};
|
||||||
|
|
||||||
|
const currentConfigJson = JSON.stringify(config, replacer, 4);
|
||||||
|
|
||||||
|
const modal = new ModalBuilder()
|
||||||
|
.setCustomId("config-modal")
|
||||||
|
.setTitle("Edit Configuration");
|
||||||
|
|
||||||
|
const jsonInput = new TextInputBuilder()
|
||||||
|
.setCustomId("json-input")
|
||||||
|
.setLabel("Configuration JSON")
|
||||||
|
.setStyle(TextInputStyle.Paragraph)
|
||||||
|
.setValue(currentConfigJson)
|
||||||
|
.setRequired(true);
|
||||||
|
|
||||||
|
const actionRow = new ActionRowBuilder<TextInputBuilder>().addComponents(jsonInput);
|
||||||
|
modal.addComponents(actionRow);
|
||||||
|
|
||||||
|
await interaction.showModal(modal);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const submitted = await interaction.awaitModalSubmit({
|
||||||
|
time: 300000, // 5 minutes
|
||||||
|
filter: (i) => i.customId === "config-modal" && i.user.id === interaction.user.id
|
||||||
|
});
|
||||||
|
|
||||||
|
const jsonString = submitted.fields.getTextInputValue("json-input");
|
||||||
|
|
||||||
|
try {
|
||||||
|
const newConfig = JSON.parse(jsonString);
|
||||||
|
saveConfig(newConfig as GameConfigType);
|
||||||
|
|
||||||
|
await submitted.reply({
|
||||||
|
embeds: [createSuccessEmbed("Configuration updated successfully.", "Config Saved")]
|
||||||
|
});
|
||||||
|
} catch (parseError) {
|
||||||
|
await submitted.reply({
|
||||||
|
embeds: [createErrorEmbed(`Invalid JSON: ${parseError instanceof Error ? parseError.message : String(parseError)}`, "Config Update Failed")],
|
||||||
|
ephemeral: true
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
// Timeout or other error handling if needed, usually just ignore timeouts for modals
|
||||||
|
if (error instanceof Error && error.message.includes('time')) {
|
||||||
|
// specific timeout handling if desired
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
@@ -11,4 +11,4 @@ await KyokoClient.deployCommands();
|
|||||||
if (!env.DISCORD_BOT_TOKEN) {
|
if (!env.DISCORD_BOT_TOKEN) {
|
||||||
throw new Error("❌ DISCORD_BOT_TOKEN is not set in environment variables.");
|
throw new Error("❌ DISCORD_BOT_TOKEN is not set in environment variables.");
|
||||||
}
|
}
|
||||||
KyokoClient.login(env.DISCORD_BOT_TOKEN);
|
KyokoClient.login(env.DISCORD_BOT_TOKEN);
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
import { readFileSync, existsSync } from 'fs';
|
import { readFileSync, existsSync, writeFileSync } from 'node:fs';
|
||||||
import { join } from 'path';
|
import { join } from 'node:path';
|
||||||
|
|
||||||
const configPath = join(process.cwd(), 'src', 'config', 'config.json');
|
const configPath = join(import.meta.dir, '..', '..', 'config', 'config.json');
|
||||||
|
|
||||||
export interface GameConfigType {
|
export interface GameConfigType {
|
||||||
leveling: {
|
leveling: {
|
||||||
@@ -81,3 +81,16 @@ reloadConfig();
|
|||||||
|
|
||||||
// Backwards compatibility alias
|
// Backwards compatibility alias
|
||||||
export const GameConfig = config;
|
export const GameConfig = config;
|
||||||
|
|
||||||
|
export function saveConfig(newConfig: GameConfigType) {
|
||||||
|
const replacer = (key: string, value: any) => {
|
||||||
|
if (typeof value === 'bigint') {
|
||||||
|
return value.toString();
|
||||||
|
}
|
||||||
|
return value;
|
||||||
|
};
|
||||||
|
|
||||||
|
const jsonString = JSON.stringify(newConfig, replacer, 4);
|
||||||
|
writeFileSync(configPath, jsonString, 'utf-8');
|
||||||
|
reloadConfig();
|
||||||
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user