Skip to content

Commit d8aa864

Browse files
committed
feat: middlewares
1 parent d82ee79 commit d8aa864

File tree

5 files changed

+68
-27
lines changed

5 files changed

+68
-27
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
import { MiddlewareContext } from 'commandkit';
2+
import { MessageFlags } from 'discord.js';
3+
4+
export function beforeExecute(ctx: MiddlewareContext) {
5+
console.log('Pre-command middleware');
6+
7+
const user = ctx.isInteraction() ? ctx.interaction.user : ctx.message.author;
8+
9+
if (ctx.commandName === 'prompt' && user.id === '159985870458322944') {
10+
if (ctx.isSlashCommand()) {
11+
ctx.interaction.reply({
12+
content: 'You are not allowed to use this command.',
13+
flags: MessageFlags.Ephemeral,
14+
});
15+
} else {
16+
ctx.message.reply('You are not allowed to use this command.');
17+
}
18+
19+
ctx.cancel();
20+
}
21+
}
22+
23+
export function afterExecute(ctx: MiddlewareContext) {
24+
console.log('Post-command middleware executed');
25+
}

packages/commandkit/src/app/command-handler/AppCommandHandler.ts

+8-6
Original file line numberDiff line numberDiff line change
@@ -107,12 +107,14 @@ export class AppCommandHandler {
107107
? loadedCommand.data.command.toJSON()
108108
: loadedCommand.data.command;
109109

110-
return json.options.reduce(
111-
(acc: Record<string, unknown>, opt: Record<string, any>) => {
112-
acc[opt.name] = opt.type;
113-
return acc;
114-
},
115-
{} as Record<string, unknown>,
110+
return (
111+
json.options?.reduce(
112+
(acc: Record<string, unknown>, opt: Record<string, any>) => {
113+
acc[opt.name] = opt.type;
114+
return acc;
115+
},
116+
{} as Record<string, unknown>,
117+
) ?? {}
116118
);
117119
},
118120
);

packages/commandkit/src/app/commands/Context.ts

+1-7
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,9 @@
11
import {
22
AutocompleteInteraction,
33
ChatInputCommandInteraction,
4-
ContextMenuCommandInteraction,
54
MessageContextMenuCommandInteraction,
65
Message,
76
Locale,
8-
User,
9-
Channel,
10-
Role,
11-
Attachment,
12-
GuildMember,
137
Interaction,
148
UserContextMenuCommandInteraction,
159
} from 'discord.js';
@@ -346,7 +340,7 @@ export class Context<
346340
}
347341

348342
export class MiddlewareContext<
349-
T extends CommandExecutionMode,
343+
T extends CommandExecutionMode = CommandExecutionMode,
350344
> extends Context<T> {
351345
#cancel = false;
352346

packages/commandkit/src/app/router/CommandsRouter.ts

+9-5
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import { existsSync } from 'node:fs';
22
import { readdir } from 'node:fs/promises';
3-
import { join } from 'node:path';
3+
import path, { join } from 'node:path';
44

55
/**
66
* Matcher type for identifying command and middleware files.
@@ -249,7 +249,7 @@ export class CommandsRouter {
249249
for (const file of files) {
250250
if (this.execMatcher(this.matchers.command, file)) {
251251
const location = this.resolveRelativePath(file);
252-
const parts = location.split('/');
252+
const parts = location.split(path.sep);
253253

254254
const parentSegments: string[] = [];
255255

@@ -283,10 +283,14 @@ export class CommandsRouter {
283283
if (this.execMatcher(this.matchers.middleware, file)) {
284284
const location = this.resolveRelativePath(file);
285285
const name = location.replace(/\.(m|c)?(j|t)sx?$/, '');
286+
const middlewareDir = path.dirname(location);
286287

287-
const command = Array.from(this.commands.values()).filter((command) =>
288-
command.path.startsWith(location),
289-
);
288+
const command = Array.from(this.commands.values()).filter((command) => {
289+
const commandDir = path.dirname(command.path);
290+
return (
291+
commandDir === middlewareDir || commandDir.startsWith(middlewareDir)
292+
);
293+
});
290294

291295
const id = crypto.randomUUID();
292296

packages/commandkit/src/legacy/handlers/command-handler/CommandHandler.ts

+25-9
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,11 @@ export interface hCommandContext {
3636
command: CommandData;
3737
}
3838

39+
enum CommandIndication {
40+
Continue,
41+
Stop,
42+
}
43+
3944
/**
4045
* A handler for client application commands.
4146
*/
@@ -219,20 +224,22 @@ export class CommandHandler {
219224
env.clearAllDeferredFunctions();
220225
}
221226

222-
async #requestExternalHandler(interaction: Interaction<CacheType> | Message) {
227+
async #requestExternalHandler(
228+
interaction: Interaction<CacheType> | Message,
229+
): Promise<CommandIndication> {
223230
const handler = this.#data.commandkitInstance.appCommandsHandler;
224-
if (!handler) return false;
231+
if (!handler) return CommandIndication.Continue;
225232

226233
if (
227234
!(interaction instanceof Message) &&
228235
!(interaction.isCommand() || interaction.isAutocomplete())
229236
) {
230-
return false;
237+
return CommandIndication.Continue;
231238
}
232239

233240
const targetCommand = await handler.prepareCommandRun(interaction);
234241

235-
if (!targetCommand) return false;
242+
if (!targetCommand) return CommandIndication.Continue;
236243

237244
const environment = useEnvironment();
238245

@@ -261,6 +268,9 @@ export class CommandHandler {
261268
const exec = async () => {
262269
if (middlewares.length > 0) {
263270
for (const middleware of middlewares) {
271+
// command was cancelled by middleware
272+
if (context.cancelled) return CommandIndication.Stop;
273+
264274
try {
265275
await middleware.data.beforeExecute(context);
266276
} catch (e) {
@@ -281,6 +291,9 @@ export class CommandHandler {
281291
}
282292
}
283293

294+
// command was cancelled by middleware
295+
if (context.cancelled) return CommandIndication.Stop;
296+
284297
let postStageRunner = true;
285298

286299
try {
@@ -368,7 +381,8 @@ export class CommandHandler {
368381
const shouldDebug = this.#data.commandkitInstance.isDebuggingCommands();
369382

370383
if (!shouldDebug) {
371-
return exec();
384+
await exec();
385+
return CommandIndication.Stop;
372386
}
373387

374388
afterCommand((env) => {
@@ -394,9 +408,8 @@ export class CommandHandler {
394408

395409
try {
396410
environment.markStart(`${command.data.command.name}`);
397-
const res = await exec();
398-
399-
return res;
411+
await exec();
412+
return CommandIndication.Stop;
400413
} finally {
401414
environment.markEnd();
402415
}
@@ -407,9 +420,10 @@ export class CommandHandler {
407420
try {
408421
const result = await this.#requestExternalHandler(interaction);
409422

410-
if (result !== false) return;
423+
if (result === CommandIndication.Stop) return;
411424
} catch (e) {
412425
console.error(e);
426+
return;
413427
}
414428

415429
if (
@@ -487,6 +501,8 @@ export class CommandHandler {
487501

488502
const command = targetCommand[isAutocomplete ? 'autocomplete' : 'run']!;
489503

504+
if (!command) return;
505+
490506
const context = {
491507
interaction,
492508
client: this.#data.client,

0 commit comments

Comments
 (0)