Skip to content

Commit 76a6b13

Browse files
committed
feat: new command handler
1 parent 13fde9e commit 76a6b13

File tree

11 files changed

+347
-37
lines changed

11 files changed

+347
-37
lines changed

packages/commandkit/src/CommandKit.ts

+5-2
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ import { AppCommandHandler } from './app/handlers/AppCommandHandler';
1313
import { LocalizationStrategy } from './app/i18n/LocalizationStrategy';
1414
import { CommandsRouter, EventsRouter } from './app/router';
1515
import { AppEventsHandler } from './app/handlers/AppEventsHandler';
16+
import { CommandKitPluginRuntime } from './plugins/runtime/CommandKitPluginRuntime';
1617

1718
export interface CommandKitConfiguration {
1819
defaultLocale: Locale;
@@ -35,8 +36,9 @@ export class CommandKit extends EventEmitter {
3536

3637
public commandsRouter!: CommandsRouter;
3738
public eventsRouter!: EventsRouter;
38-
public commandHandler = new AppCommandHandler(this);
39-
public eventHandler = new AppEventsHandler(this);
39+
public readonly commandHandler = new AppCommandHandler(this);
40+
public readonly eventHandler = new AppEventsHandler(this);
41+
public readonly plugins: CommandKitPluginRuntime;
4042

4143
static instance: CommandKit | undefined = undefined;
4244

@@ -81,6 +83,7 @@ export class CommandKit extends EventEmitter {
8183
}
8284

8385
this.eventInterceptor = new EventInterceptor(options.client);
86+
this.plugins = new CommandKitPluginRuntime(this);
8487

8588
if (!CommandKit.instance) {
8689
CommandKit.instance = this;

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

+1-1
Original file line numberDiff line numberDiff line change
@@ -176,7 +176,7 @@ export class Context<
176176
* @param command The command to forward to.
177177
*/
178178
public async forwardCommand(command: string): Promise<never> {
179-
const target = await this.commandkit.appCommandsHandler.prepareCommandRun(
179+
const target = await this.commandkit.commandHandler.prepareCommandRun(
180180
(this.isInteraction() ? this.interaction : this.message) as Interaction,
181181
);
182182

packages/commandkit/src/app/handlers/AppCommandHandler.ts

+163-1
Original file line numberDiff line numberDiff line change
@@ -1,20 +1,29 @@
11
import type { CommandKit } from '../../CommandKit';
22
import {
33
Awaitable,
4+
ChatInputCommandInteraction,
45
Collection,
56
ContextMenuCommandBuilder,
7+
Events,
68
Interaction,
79
Locale,
810
Message,
11+
PartialMessage,
912
SlashCommandBuilder,
1013
} from 'discord.js';
11-
import { Context } from '../commands/Context';
14+
import {
15+
CommandExecutionMode,
16+
Context,
17+
MiddlewareContext,
18+
} from '../commands/Context';
1219
import { toFileURL } from '../../utils/resolve-file-url';
1320
import { TranslatableCommandOptions } from '../i18n/Translation';
1421
import { MessageCommandParser } from '../commands/MessageCommandParser';
1522
import { CommandKitErrorCodes, isErrorType } from '../../utils/error-codes';
1623
import { ParsedCommand, ParsedMiddleware } from '../router';
1724
import { CommandRegistrar } from '../register/CommandRegistrar';
25+
import { GenericFunction } from '../../context/async-context';
26+
import { Logger } from '../../logger/Logger';
1827

1928
interface AppCommand {
2029
command: SlashCommandBuilder | Record<string, any>;
@@ -73,6 +82,11 @@ export class AppCommandHandler {
7382
private loadedCommands = new Collection<string, LoadedCommand>();
7483
private loadedMiddlewares = new Collection<string, LoadedMiddleware>();
7584
public readonly registrar: CommandRegistrar;
85+
private onInteraction: GenericFunction<[Interaction]> | null = null;
86+
private onMessageCreate: GenericFunction<[Message]> | null = null;
87+
private onMessageUpdate: GenericFunction<
88+
[Message | PartialMessage, Message | PartialMessage]
89+
> | null = null;
7690

7791
public constructor(public readonly commandkit: CommandKit) {
7892
this.registrar = new CommandRegistrar(this.commandkit);
@@ -83,6 +97,154 @@ export class AppCommandHandler {
8397
return loaded;
8498
}
8599

100+
public registerCommandHandler() {
101+
this.onInteraction ??= async (interaction: Interaction) => {
102+
const success = await this.commandkit.plugins.execute(
103+
async (ctx, plugin) => {
104+
return plugin.onBeforeInteraction(ctx, interaction);
105+
},
106+
);
107+
108+
// plugin will handle the interaction
109+
if (success) return;
110+
111+
const isCommandLike =
112+
interaction.isCommand() ||
113+
interaction.isAutocomplete() ||
114+
interaction.isUserContextMenuCommand() ||
115+
interaction.isMessageContextMenuCommand();
116+
117+
if (!isCommandLike) return;
118+
119+
const command = await this.prepareCommandRun(interaction);
120+
121+
if (!command) return;
122+
123+
return this.runCommand(command, interaction);
124+
};
125+
126+
this.onMessageCreate ??= async (message: Message) => {
127+
const success = await this.commandkit.plugins.execute(
128+
async (ctx, plugin) => {
129+
return plugin.onBeforeMessageCommand(ctx, message);
130+
},
131+
);
132+
133+
// plugin will handle the message
134+
if (success) return;
135+
if (message.author.bot) return;
136+
137+
const command = await this.prepareCommandRun(message);
138+
139+
if (!command) return;
140+
141+
return this.runCommand(command, message);
142+
};
143+
144+
this.onMessageUpdate ??= async (
145+
oldMessage: Message | PartialMessage,
146+
newMessage: Message | PartialMessage,
147+
) => {
148+
const success = await this.commandkit.plugins.execute(
149+
async (ctx, plugin) => {
150+
return plugin.onBeforeMessageUpdateCommand(
151+
ctx,
152+
oldMessage,
153+
newMessage,
154+
);
155+
},
156+
);
157+
158+
// plugin will handle the message
159+
if (success) return;
160+
if (oldMessage.partial || newMessage.partial) return;
161+
if (oldMessage.author.bot) return;
162+
163+
const command = await this.prepareCommandRun(newMessage);
164+
165+
if (!command) return;
166+
167+
return this.runCommand(command, newMessage);
168+
};
169+
170+
this.commandkit.client.on(Events.InteractionCreate, this.onInteraction);
171+
this.commandkit.client.on(Events.MessageCreate, this.onMessageCreate);
172+
this.commandkit.client.on(Events.MessageUpdate, this.onMessageUpdate);
173+
}
174+
175+
public getExecutionMode(source: Interaction | Message): CommandExecutionMode {
176+
if (source instanceof Message) return CommandExecutionMode.Message;
177+
if (source.isChatInputCommand()) return CommandExecutionMode.SlashCommand;
178+
if (source.isAutocomplete()) {
179+
return CommandExecutionMode.Autocomplete;
180+
}
181+
if (source.isMessageContextMenuCommand()) {
182+
return CommandExecutionMode.MessageContextMenu;
183+
}
184+
if (source.isUserContextMenuCommand()) {
185+
return CommandExecutionMode.UserContextMenu;
186+
}
187+
188+
return null as never;
189+
}
190+
191+
public async runCommand(
192+
command: PreparedAppCommandExecution,
193+
source: Interaction | Message,
194+
) {
195+
if (
196+
source instanceof Message &&
197+
(source.editedTimestamp || source.partial)
198+
) {
199+
// TODO: handle message edit
200+
return;
201+
}
202+
203+
const executionMode = this.getExecutionMode(source);
204+
205+
const ctx = new MiddlewareContext(this.commandkit, {
206+
executionMode,
207+
interaction: !(source instanceof Message)
208+
? (source as ChatInputCommandInteraction)
209+
: (null as never),
210+
message: source instanceof Message ? source : (null as never),
211+
forwarded: false,
212+
});
213+
214+
for (const middleware of command.middlewares) {
215+
await middleware.data.beforeExecute(ctx);
216+
}
217+
218+
const fn = command.command.data[executionMode];
219+
220+
if (!fn) {
221+
Logger.warn(
222+
`Command ${command.command.command.name} has no handler for ${executionMode}`,
223+
);
224+
}
225+
226+
if (fn) {
227+
try {
228+
const executeCommand = async () => fn(ctx.clone());
229+
const res = await this.commandkit.plugins.execute(
230+
async (ctx, plugin) => {
231+
return plugin.executeCommand(ctx, source, command, executeCommand);
232+
},
233+
);
234+
235+
if (!res) {
236+
await executeCommand();
237+
}
238+
} catch (e) {
239+
Logger.error(e);
240+
}
241+
}
242+
243+
for (const middleware of command.middlewares) {
244+
await middleware.data.afterExecute(ctx);
245+
}
246+
}
247+
86248
public async prepareCommandRun(
87249
source: Interaction | Message,
88250
): Promise<PreparedAppCommandExecution | null> {

packages/commandkit/src/plugins/CompilerPlugin.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ import {
66
} from './runtime/types';
77
import { PluginCommon, PluginOptions } from './PluginCommon';
88
import { MaybeFalsey } from './types';
9+
import { CompilerPluginRuntime } from './runtime/CompilerPluginRuntime';
910

1011
export interface PluginTransformParameters {
1112
args: OnLoadArgs;
@@ -26,7 +27,7 @@ export interface ResolveResult {
2627

2728
export abstract class CompilerPlugin<
2829
T extends PluginOptions = PluginOptions,
29-
> extends PluginCommon<T> {
30+
> extends PluginCommon<T, CompilerPluginRuntime> {
3031
/**
3132
* Called when transformation is requested to this plugin
3233
* @param params The parameters for the transformation

packages/commandkit/src/plugins/PluginCommon.ts

+7-4
Original file line numberDiff line numberDiff line change
@@ -1,20 +1,23 @@
1-
import { ResolvedCommandKitConfig } from '../config';
1+
import { CommonPluginRuntime } from './runtime/runtime';
22

33
export type PluginOptions = Record<string, any>;
44

5-
export abstract class PluginCommon<T extends PluginOptions> {
5+
export abstract class PluginCommon<
6+
T extends PluginOptions,
7+
C extends CommonPluginRuntime = CommonPluginRuntime,
8+
> {
69
public abstract readonly name: string;
710
public constructor(protected readonly options: T) {}
811

912
/**
1013
* Called when this plugin is activated
1114
*/
12-
public async activate(config: ResolvedCommandKitConfig): Promise<void> {}
15+
public async activate(ctx: C): Promise<void> {}
1316

1417
/**
1518
* Called when this plugin is deactivated
1619
*/
17-
public async deactivate(config: ResolvedCommandKitConfig): Promise<void> {}
20+
public async deactivate(ctx: C): Promise<void> {}
1821
}
1922

2023
export function isPlugin(

packages/commandkit/src/plugins/RuntimePlugin.ts

+72-14
Original file line numberDiff line numberDiff line change
@@ -1,44 +1,102 @@
1+
import { Interaction, Message, PartialMessage } from 'discord.js';
12
import { PluginCommon, PluginOptions } from './PluginCommon';
2-
import { CommandKit } from '../CommandKit';
3+
import type { CommandKitPluginRuntime } from './runtime/CommandKitPluginRuntime';
4+
import { PreparedAppCommandExecution } from '../app';
35

46
export abstract class RuntimePlugin<
57
T extends PluginOptions = PluginOptions,
6-
> extends PluginCommon<T> {
8+
> extends PluginCommon<T, CommandKitPluginRuntime> {
79
/**
810
* Called before commands are loaded
9-
* @param commandkit The commandkit instance
1011
*/
11-
public async onBeforeCommandsLoad(commandkit: CommandKit): Promise<void> {}
12+
public async onBeforeCommandsLoad(
13+
ctx: CommandKitPluginRuntime,
14+
): Promise<void> {}
1215

1316
/**
1417
* Called after commands are loaded
15-
* @param commandkit The commandkit instance
1618
*/
17-
public async onAfterCommandsLoad(commandkit: CommandKit): Promise<void> {}
19+
public async onAfterCommandsLoad(
20+
ctx: CommandKitPluginRuntime,
21+
): Promise<void> {}
1822

1923
/**
2024
* Called before events are loaded
21-
* @param commandkit The commandkit instance
2225
*/
23-
public async onBeforeEventsLoad(commandkit: CommandKit): Promise<void> {}
26+
public async onBeforeEventsLoad(
27+
ctx: CommandKitPluginRuntime,
28+
): Promise<void> {}
2429

2530
/**
2631
* Called after events are loaded
27-
* @param commandkit The commandkit instance
2832
*/
29-
public async onAfterEventsLoad(commandkit: CommandKit): Promise<void> {}
33+
public async onAfterEventsLoad(ctx: CommandKitPluginRuntime): Promise<void> {}
3034

3135
/**
3236
* Called before the client logs in
33-
* @param commandkit The commandkit instance
3437
*/
35-
public async onBeforeClientLogin(commandkit: CommandKit): Promise<void> {}
38+
public async onBeforeClientLogin(
39+
ctx: CommandKitPluginRuntime,
40+
): Promise<void> {}
3641

3742
/**
3843
* Called after the client logs in
39-
* @param commandkit The commandkit instance
4044
*/
41-
public async onAfterClientLogin(commandkit: CommandKit): Promise<void> {}
45+
public async onAfterClientLogin(
46+
ctx: CommandKitPluginRuntime,
47+
): Promise<void> {}
48+
49+
/**
50+
* Called before interaction is handled
51+
* @param interaction The interaction
52+
*/
53+
public async onBeforeInteraction(
54+
ctx: CommandKitPluginRuntime,
55+
interaction: Interaction,
56+
): Promise<void> {}
57+
58+
/**
59+
* Called after interaction is handled
60+
* @param interaction The interaction that was handled
61+
*/
62+
public async onAfterInteraction(
63+
ctx: CommandKitPluginRuntime,
64+
interaction: Interaction,
65+
): Promise<void> {}
66+
67+
/**
68+
* Called before message command is processed
69+
* @param message The message
70+
*/
71+
public async onBeforeMessageCommand(
72+
ctx: CommandKitPluginRuntime,
73+
message: Message,
74+
): Promise<void> {}
75+
76+
/**
77+
* Called before message update command is processed
78+
* @param message The message
79+
*/
80+
public async onBeforeMessageUpdateCommand(
81+
ctx: CommandKitPluginRuntime,
82+
oldMessage: Message | PartialMessage,
83+
newMessage: Message | PartialMessage,
84+
): Promise<void> {}
85+
86+
/**
87+
* Called before command is executed. This method can execute the command itself.
88+
* @param source The source that triggered the command
89+
* @param command The command
90+
* @param execute The function that executes the command
91+
*/
92+
public async executeCommand(
93+
ctx: CommandKitPluginRuntime,
94+
source: Interaction | Message,
95+
command: PreparedAppCommandExecution,
96+
execute: () => Promise<any>,
97+
): Promise<boolean> {
98+
return false;
99+
}
42100
}
43101

44102
export function isRuntimePlugin(plugin: unknown): plugin is RuntimePlugin {

0 commit comments

Comments
 (0)