chore: add option to undeploy commands and not deploy on start

This commit is contained in:
Ahmad 2025-04-16 19:20:17 -04:00
parent 072c34d778
commit 14667ad69f
No known key found for this signature in database
GPG key ID: 8FD8A93530D182BF
5 changed files with 113 additions and 12 deletions

View file

@ -1,7 +1,7 @@
import { Client, ClientOptions, Collection } from 'discord.js';
import { Command } from '@/types/CommandTypes.js';
import { Config } from '@/types/ConfigTypes.js';
import { deployCommands } from '@/util/deployCommand.js';
import { deployCommands, getFilesRecursively } from '@/util/deployCommand.js';
import { registerEvents } from '@/util/eventLoader.js';
/**
@ -29,20 +29,69 @@ export class ExtendedClient extends Client {
private async loadModules() {
try {
const commands = await deployCommands();
if (!commands?.length) {
throw new Error('No commands found');
}
if (process.env.SKIP_COMMAND_DEPLOY === 'true') {
console.log('Skipping command deployment (SKIP_COMMAND_DEPLOY=true)');
const commandFiles = await this.loadCommandsWithoutDeploying();
for (const command of commands) {
this.commands.set(command.data.name, command);
}
await registerEvents(this);
console.log(
`Loaded ${commandFiles.length} commands and registered events (without deployment)`,
);
} else {
const commands = await deployCommands();
if (!commands?.length) {
throw new Error('No commands found');
}
await registerEvents(this);
console.log(`Loaded ${commands.length} commands and registered events`);
for (const command of commands) {
this.commands.set(command.data.name, command);
}
await registerEvents(this);
console.log(`Loaded ${commands.length} commands and registered events`);
}
} catch (error) {
console.error('Error loading modules:', error);
process.exit(1);
}
}
/**
* Loads commands without deploying them to Discord
* @returns Array of command objects
*/
private async loadCommandsWithoutDeploying(): Promise<Command[]> {
try {
const path = await import('path');
const __dirname = path.resolve();
const commandsPath = path.join(__dirname, 'target', 'commands');
const commandFiles = getFilesRecursively(commandsPath);
const commands: Command[] = [];
for (const file of commandFiles) {
const commandModule = await import(`file://${file}`);
const command = commandModule.default;
if (
command instanceof Object &&
'data' in command &&
'execute' in command
) {
commands.push(command);
this.commands.set(command.data.name, command);
} else {
console.warn(
`[WARNING] The command at ${file} is missing a required "data" or "execute" property.`,
);
}
}
return commands;
} catch (error) {
console.error('Error loading commands:', error);
throw error;
}
}
}

View file

@ -17,7 +17,7 @@ const rest = new REST({ version: '10' }).setToken(token);
* @param directory - The directory to get files from
* @returns - An array of file paths
*/
const getFilesRecursively = (directory: string): string[] => {
export const getFilesRecursively = (directory: string): string[] => {
const files: string[] = [];
const filesInDirectory = fs.readdirSync(directory);

View file

@ -0,0 +1,36 @@
import { REST, Routes } from 'discord.js';
import { loadConfig } from './configLoader.js';
const config = loadConfig();
const { token, clientId, guildId } = config;
const rest = new REST({ version: '10' }).setToken(token);
/**
* Undeploys all commands from the Discord API
*/
export const undeployCommands = async () => {
try {
console.log('Undeploying all commands from the Discord API...');
await rest.put(Routes.applicationGuildCommands(clientId, guildId), {
body: [],
});
console.log('Successfully undeployed all commands');
} catch (error) {
console.error('Error undeploying commands:', error);
throw error;
}
};
if (import.meta.url.endsWith(process.argv[1].replace(/\\/g, '/'))) {
undeployCommands()
.then(() => {
console.log('Undeploy process completed successfully');
})
.catch((err) => {
console.error('Undeploy process failed:', err);
process.exitCode = 1;
});
}