@commandkit/ai 0.1.0-dev.20250609135507
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +5 -0
- package/README.md +19 -0
- package/dist/ai-context-worker.d.ts +7 -0
- package/dist/ai-context-worker.js +17 -0
- package/dist/context.d.ts +15 -0
- package/dist/context.js +16 -0
- package/dist/index.d.ts +6 -0
- package/dist/index.js +25 -0
- package/dist/plugin.d.ts +27 -0
- package/dist/plugin.js +254 -0
- package/dist/types.d.ts +12 -0
- package/dist/types.js +3 -0
- package/package.json +38 -0
package/LICENSE
ADDED
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
Copyright 2025 Avraj Sahota
|
|
2
|
+
|
|
3
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the “Software”), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so.
|
|
4
|
+
|
|
5
|
+
THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
package/README.md
ADDED
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
# `@commandkit/ai`
|
|
2
|
+
|
|
3
|
+
Supercharge your CommandKit project with AI capabilities.
|
|
4
|
+
|
|
5
|
+
## Installation
|
|
6
|
+
|
|
7
|
+
```bash
|
|
8
|
+
npm install @commandkit/ai
|
|
9
|
+
```
|
|
10
|
+
|
|
11
|
+
## Usage
|
|
12
|
+
|
|
13
|
+
```ts
|
|
14
|
+
import { ai } from '@commandkit/ai';
|
|
15
|
+
|
|
16
|
+
export default defineConfig({
|
|
17
|
+
plugins: [ai()],
|
|
18
|
+
})
|
|
19
|
+
```
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import { Message } from 'discord.js';
|
|
2
|
+
import { AiContext } from './context';
|
|
3
|
+
export declare function getAiWorkerContext(): {
|
|
4
|
+
message: Message;
|
|
5
|
+
ctx: AiContext;
|
|
6
|
+
};
|
|
7
|
+
export declare function runInAiWorkerContext<R, F extends (...args: any[]) => R>(ctx: AiContext, message: Message, callback: F): R;
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.getAiWorkerContext = getAiWorkerContext;
|
|
4
|
+
exports.runInAiWorkerContext = runInAiWorkerContext;
|
|
5
|
+
const node_async_hooks_1 = require("node:async_hooks");
|
|
6
|
+
const worker = new node_async_hooks_1.AsyncLocalStorage();
|
|
7
|
+
function getAiWorkerContext() {
|
|
8
|
+
const ctx = worker.getStore();
|
|
9
|
+
if (!ctx) {
|
|
10
|
+
throw new Error('AI context is not available. Ensure you are using AI in a CommandKit environment.');
|
|
11
|
+
}
|
|
12
|
+
return ctx;
|
|
13
|
+
}
|
|
14
|
+
function runInAiWorkerContext(ctx, message, callback) {
|
|
15
|
+
return worker.run({ message, ctx }, callback);
|
|
16
|
+
}
|
|
17
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWktY29udGV4dC13b3JrZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvYWktY29udGV4dC13b3JrZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFNQSxnREFVQztBQUVELG9EQU1DO0FBdkJELHVEQUFxRDtBQUdyRCxNQUFNLE1BQU0sR0FBRyxJQUFJLG9DQUFpQixFQUF3QyxDQUFDO0FBRTdFLFNBQWdCLGtCQUFrQjtJQUNoQyxNQUFNLEdBQUcsR0FBRyxNQUFNLENBQUMsUUFBUSxFQUFFLENBQUM7SUFFOUIsSUFBSSxDQUFDLEdBQUcsRUFBRSxDQUFDO1FBQ1QsTUFBTSxJQUFJLEtBQUssQ0FDYixtRkFBbUYsQ0FDcEYsQ0FBQztJQUNKLENBQUM7SUFFRCxPQUFPLEdBQUcsQ0FBQztBQUNiLENBQUM7QUFFRCxTQUFnQixvQkFBb0IsQ0FDbEMsR0FBYyxFQUNkLE9BQWdCLEVBQ2hCLFFBQVc7SUFFWCxPQUFPLE1BQU0sQ0FBQyxHQUFHLENBQUMsRUFBRSxPQUFPLEVBQUUsR0FBRyxFQUFFLEVBQUUsUUFBUSxDQUFDLENBQUM7QUFDaEQsQ0FBQyJ9
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import type { CommandKit } from 'commandkit';
|
|
2
|
+
import { Client, Message } from 'discord.js';
|
|
3
|
+
export interface AiContextOptions<T extends Record<string, unknown> = Record<string, unknown>> {
|
|
4
|
+
message: Message;
|
|
5
|
+
params: T;
|
|
6
|
+
commandkit: CommandKit;
|
|
7
|
+
}
|
|
8
|
+
export declare class AiContext<T extends Record<string, unknown> = Record<string, unknown>> {
|
|
9
|
+
params: T;
|
|
10
|
+
message: Message;
|
|
11
|
+
client: Client;
|
|
12
|
+
commandkit: CommandKit;
|
|
13
|
+
constructor(options: AiContextOptions<T>);
|
|
14
|
+
setParams(params: T): void;
|
|
15
|
+
}
|
package/dist/context.js
ADDED
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.AiContext = void 0;
|
|
4
|
+
class AiContext {
|
|
5
|
+
constructor(options) {
|
|
6
|
+
this.params = options.params;
|
|
7
|
+
this.message = options.message;
|
|
8
|
+
this.commandkit = options.commandkit;
|
|
9
|
+
this.client = options.commandkit.client;
|
|
10
|
+
}
|
|
11
|
+
setParams(params) {
|
|
12
|
+
this.params = params;
|
|
13
|
+
}
|
|
14
|
+
}
|
|
15
|
+
exports.AiContext = AiContext;
|
|
16
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29udGV4dC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3NyYy9jb250ZXh0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQVdBLE1BQWEsU0FBUztJQVFwQixZQUFtQixPQUE0QjtRQUM3QyxJQUFJLENBQUMsTUFBTSxHQUFHLE9BQU8sQ0FBQyxNQUFNLENBQUM7UUFDN0IsSUFBSSxDQUFDLE9BQU8sR0FBRyxPQUFPLENBQUMsT0FBTyxDQUFDO1FBQy9CLElBQUksQ0FBQyxVQUFVLEdBQUcsT0FBTyxDQUFDLFVBQVUsQ0FBQztRQUNyQyxJQUFJLENBQUMsTUFBTSxHQUFHLE9BQU8sQ0FBQyxVQUFVLENBQUMsTUFBTSxDQUFDO0lBQzFDLENBQUM7SUFFTSxTQUFTLENBQUMsTUFBUztRQUN4QixJQUFJLENBQUMsTUFBTSxHQUFHLE1BQU0sQ0FBQztJQUN2QixDQUFDO0NBQ0Y7QUFsQkQsOEJBa0JDIn0=
|
package/dist/index.d.ts
ADDED
package/dist/index.js
ADDED
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
14
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
|
+
};
|
|
16
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
+
exports.ai = ai;
|
|
18
|
+
const plugin_1 = require("./plugin");
|
|
19
|
+
function ai(options) {
|
|
20
|
+
return new plugin_1.AiPlugin(options ?? {});
|
|
21
|
+
}
|
|
22
|
+
__exportStar(require("./types"), exports);
|
|
23
|
+
__exportStar(require("./plugin"), exports);
|
|
24
|
+
__exportStar(require("./context"), exports);
|
|
25
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUdBLGdCQUVDO0FBTEQscUNBQW9DO0FBR3BDLFNBQWdCLEVBQUUsQ0FBQyxPQUF5QjtJQUMxQyxPQUFPLElBQUksaUJBQVEsQ0FBQyxPQUFPLElBQUksRUFBRSxDQUFDLENBQUM7QUFDckMsQ0FBQztBQUVELDBDQUF3QjtBQUN4QiwyQ0FBeUI7QUFDekIsNENBQTBCIn0=
|
package/dist/plugin.d.ts
ADDED
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import { CommandKitPluginRuntime, RuntimePlugin } from 'commandkit/plugin';
|
|
2
|
+
import { AiPluginOptions, MessageFilter, SelectAiModel } from './types';
|
|
3
|
+
import { Message } from 'discord.js';
|
|
4
|
+
export interface AiConfig {
|
|
5
|
+
description?: string;
|
|
6
|
+
parameters: any;
|
|
7
|
+
}
|
|
8
|
+
export interface ConfigureAI {
|
|
9
|
+
messageFilter?: MessageFilter;
|
|
10
|
+
selectAiModel?: SelectAiModel;
|
|
11
|
+
systemPrompt?: (message: Message) => Promise<string>;
|
|
12
|
+
}
|
|
13
|
+
export declare function configureAI(config: ConfigureAI): void;
|
|
14
|
+
export declare class AiPlugin extends RuntimePlugin<AiPluginOptions> {
|
|
15
|
+
readonly name = "AiPlugin";
|
|
16
|
+
private toolsRecord;
|
|
17
|
+
private defaultTools;
|
|
18
|
+
private onMessageFunc;
|
|
19
|
+
constructor(options: AiPluginOptions);
|
|
20
|
+
activate(ctx: CommandKitPluginRuntime): Promise<void>;
|
|
21
|
+
private createDefaultTools;
|
|
22
|
+
deactivate(ctx: CommandKitPluginRuntime): Promise<void>;
|
|
23
|
+
private handleMessage;
|
|
24
|
+
private startTyping;
|
|
25
|
+
onBeforeCommandsLoad(ctx: CommandKitPluginRuntime): Promise<void>;
|
|
26
|
+
onAfterCommandsLoad(ctx: CommandKitPluginRuntime): Promise<void>;
|
|
27
|
+
}
|
package/dist/plugin.js
ADDED
|
@@ -0,0 +1,254 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.AiPlugin = void 0;
|
|
4
|
+
exports.configureAI = configureAI;
|
|
5
|
+
const plugin_1 = require("commandkit/plugin");
|
|
6
|
+
const commandkit_1 = require("commandkit");
|
|
7
|
+
const context_1 = require("./context");
|
|
8
|
+
const discord_js_1 = require("discord.js");
|
|
9
|
+
const ai_1 = require("ai");
|
|
10
|
+
const zod_1 = require("zod");
|
|
11
|
+
const ai_context_worker_1 = require("./ai-context-worker");
|
|
12
|
+
let messageFilter = null;
|
|
13
|
+
let selectAiModel = null;
|
|
14
|
+
let generateSystemPrompt;
|
|
15
|
+
function configureAI(config) {
|
|
16
|
+
if (config.messageFilter) {
|
|
17
|
+
messageFilter = config.messageFilter;
|
|
18
|
+
}
|
|
19
|
+
if (config.selectAiModel) {
|
|
20
|
+
selectAiModel = config.selectAiModel;
|
|
21
|
+
}
|
|
22
|
+
if (config.systemPrompt) {
|
|
23
|
+
generateSystemPrompt = config.systemPrompt;
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
class AiPlugin extends plugin_1.RuntimePlugin {
|
|
27
|
+
constructor(options) {
|
|
28
|
+
super(options);
|
|
29
|
+
this.name = 'AiPlugin';
|
|
30
|
+
this.toolsRecord = {};
|
|
31
|
+
this.defaultTools = {};
|
|
32
|
+
this.onMessageFunc = null;
|
|
33
|
+
}
|
|
34
|
+
async activate(ctx) {
|
|
35
|
+
this.onMessageFunc = (message) => this.handleMessage(ctx, message);
|
|
36
|
+
ctx.commandkit.client.on(discord_js_1.Events.MessageCreate, this.onMessageFunc);
|
|
37
|
+
this.createDefaultTools(ctx);
|
|
38
|
+
commandkit_1.Logger.info(`Plugin ${this.name} activated`);
|
|
39
|
+
}
|
|
40
|
+
createDefaultTools(ctx) {
|
|
41
|
+
const { commandkit } = ctx;
|
|
42
|
+
const client = commandkit.client;
|
|
43
|
+
this.defaultTools.getUserById = (0, ai_1.tool)({
|
|
44
|
+
description: 'Get user information by ID',
|
|
45
|
+
parameters: zod_1.z.object({
|
|
46
|
+
userId: zod_1.z
|
|
47
|
+
.string()
|
|
48
|
+
.describe('The ID of the user to retrieve. This is a Discord snowflake string.'),
|
|
49
|
+
}),
|
|
50
|
+
execute: async (params) => {
|
|
51
|
+
const user = await client.users.fetch(params.userId, {
|
|
52
|
+
force: false,
|
|
53
|
+
cache: true,
|
|
54
|
+
});
|
|
55
|
+
return user.toJSON();
|
|
56
|
+
},
|
|
57
|
+
});
|
|
58
|
+
this.defaultTools.getChannelById = (0, ai_1.tool)({
|
|
59
|
+
description: 'Get channel information by ID',
|
|
60
|
+
parameters: zod_1.z.object({
|
|
61
|
+
channelId: zod_1.z
|
|
62
|
+
.string()
|
|
63
|
+
.describe('The ID of the channel to retrieve. This is a Discord snowflake string.'),
|
|
64
|
+
}),
|
|
65
|
+
execute: async (params) => {
|
|
66
|
+
const channel = await client.channels.fetch(params.channelId, {
|
|
67
|
+
force: false,
|
|
68
|
+
cache: true,
|
|
69
|
+
});
|
|
70
|
+
if (!channel) {
|
|
71
|
+
throw new Error(`Channel with ID ${params.channelId} not found.`);
|
|
72
|
+
}
|
|
73
|
+
return channel.toJSON();
|
|
74
|
+
},
|
|
75
|
+
});
|
|
76
|
+
this.defaultTools.getGuildById = (0, ai_1.tool)({
|
|
77
|
+
description: 'Get guild information by ID',
|
|
78
|
+
parameters: zod_1.z.object({
|
|
79
|
+
guildId: zod_1.z
|
|
80
|
+
.string()
|
|
81
|
+
.describe('The ID of the guild to retrieve. This is a Discord snowflake string.'),
|
|
82
|
+
}),
|
|
83
|
+
execute: async (params) => {
|
|
84
|
+
const guild = await client.guilds.fetch({
|
|
85
|
+
guild: params.guildId,
|
|
86
|
+
force: false,
|
|
87
|
+
cache: true,
|
|
88
|
+
});
|
|
89
|
+
if (!guild) {
|
|
90
|
+
throw new Error(`Guild with ID ${params.guildId} not found.`);
|
|
91
|
+
}
|
|
92
|
+
return {
|
|
93
|
+
id: guild.id,
|
|
94
|
+
name: guild.name,
|
|
95
|
+
icon: guild.iconURL(),
|
|
96
|
+
memberCount: guild.memberCount,
|
|
97
|
+
};
|
|
98
|
+
},
|
|
99
|
+
});
|
|
100
|
+
this.defaultTools.getCurrentUser = (0, ai_1.tool)({
|
|
101
|
+
description: 'Get information about the current discord bot user',
|
|
102
|
+
parameters: zod_1.z.object({}),
|
|
103
|
+
execute: async () => {
|
|
104
|
+
const user = client.user;
|
|
105
|
+
if (!user) {
|
|
106
|
+
throw new Error('Bot user is not available.');
|
|
107
|
+
}
|
|
108
|
+
return user.toJSON();
|
|
109
|
+
},
|
|
110
|
+
});
|
|
111
|
+
this.defaultTools.getAvailableCommands = (0, ai_1.tool)({
|
|
112
|
+
description: 'Get all available commands',
|
|
113
|
+
parameters: zod_1.z.object({}),
|
|
114
|
+
execute: async () => {
|
|
115
|
+
return ctx.commandkit.commandHandler.getCommandsArray().map((cmd) => ({
|
|
116
|
+
name: cmd.data.command.name,
|
|
117
|
+
description: cmd.data.command.description,
|
|
118
|
+
category: cmd.command.category,
|
|
119
|
+
}));
|
|
120
|
+
},
|
|
121
|
+
});
|
|
122
|
+
}
|
|
123
|
+
async deactivate(ctx) {
|
|
124
|
+
this.toolsRecord = {};
|
|
125
|
+
if (this.onMessageFunc) {
|
|
126
|
+
ctx.commandkit.client.off(discord_js_1.Events.MessageCreate, this.onMessageFunc);
|
|
127
|
+
this.onMessageFunc = null;
|
|
128
|
+
}
|
|
129
|
+
commandkit_1.Logger.info(`Plugin ${this.name} deactivated`);
|
|
130
|
+
}
|
|
131
|
+
async handleMessage(pluginContext, message) {
|
|
132
|
+
if (message.author.bot || !Object.keys(this.toolsRecord).length)
|
|
133
|
+
return;
|
|
134
|
+
const aiModelSelector = selectAiModel;
|
|
135
|
+
if (!message.content?.length || !aiModelSelector)
|
|
136
|
+
return;
|
|
137
|
+
if (!message.channel.isTextBased() || !message.channel.isSendable())
|
|
138
|
+
return;
|
|
139
|
+
const shouldContinue = messageFilter ? await messageFilter(message) : true;
|
|
140
|
+
if (!shouldContinue)
|
|
141
|
+
return;
|
|
142
|
+
const ctx = new context_1.AiContext({
|
|
143
|
+
message,
|
|
144
|
+
params: {},
|
|
145
|
+
commandkit: pluginContext.commandkit,
|
|
146
|
+
});
|
|
147
|
+
const systemPrompt = (await generateSystemPrompt?.(message)) ||
|
|
148
|
+
`You are a helpful AI discord bot. Your name is ${message.client.user.username} and your id is ${message.client.user.id}.
|
|
149
|
+
You are designed to assist users with their questions and tasks. You also have access to various tools that can help you perform tasks.
|
|
150
|
+
Tools are basically like commands that you can execute to perform specific actions based on user input.
|
|
151
|
+
Keep the response short and concise, and only use tools when necessary. Keep the response length under 2000 characters.
|
|
152
|
+
Do not include your own text in the response unless necessary. For text formatting, you can use discord's markdown syntax.
|
|
153
|
+
${message.inGuild() ? `\nYou are currently in a guild named ${message.guild.name} whose id is ${message.guildId}. While in guild, you can fetch member information if needed.` : '\nYou are currently in a direct message with the user.'}`;
|
|
154
|
+
const userInfo = `<user>
|
|
155
|
+
<id>${message.author.id}</id>
|
|
156
|
+
<name>${message.author.username}</name>
|
|
157
|
+
<displayName>${message.author.displayName}</displayName>
|
|
158
|
+
<avatar>${message.author.avatarURL()}</avatar>
|
|
159
|
+
</user>`;
|
|
160
|
+
await (0, ai_context_worker_1.runInAiWorkerContext)(ctx, message, async () => {
|
|
161
|
+
const channel = message.channel;
|
|
162
|
+
const stopTyping = await this.startTyping(channel);
|
|
163
|
+
try {
|
|
164
|
+
const { model, options } = await aiModelSelector(message);
|
|
165
|
+
const result = await (0, ai_1.generateText)({
|
|
166
|
+
abortSignal: AbortSignal.timeout(60000),
|
|
167
|
+
model,
|
|
168
|
+
tools: { ...this.toolsRecord, ...this.defaultTools },
|
|
169
|
+
prompt: `${userInfo}\nUser: ${message.content}\nAI:`,
|
|
170
|
+
system: systemPrompt,
|
|
171
|
+
maxSteps: 5,
|
|
172
|
+
providerOptions: options,
|
|
173
|
+
});
|
|
174
|
+
stopTyping();
|
|
175
|
+
if (!!result.text) {
|
|
176
|
+
await message.reply({
|
|
177
|
+
content: result.text,
|
|
178
|
+
allowedMentions: { parse: [] },
|
|
179
|
+
});
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
catch (e) {
|
|
183
|
+
commandkit_1.Logger.error(`Error processing AI message: ${e}`);
|
|
184
|
+
const channel = message.channel;
|
|
185
|
+
if (channel.isSendable()) {
|
|
186
|
+
await message
|
|
187
|
+
.reply({
|
|
188
|
+
content: 'An error occurred while processing your request.',
|
|
189
|
+
allowedMentions: { parse: [] },
|
|
190
|
+
})
|
|
191
|
+
.catch((e) => commandkit_1.Logger.error(`Failed to send error message: ${e}`));
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
finally {
|
|
195
|
+
stopTyping();
|
|
196
|
+
}
|
|
197
|
+
});
|
|
198
|
+
}
|
|
199
|
+
async startTyping(channel) {
|
|
200
|
+
let stopped = false;
|
|
201
|
+
const runner = async () => {
|
|
202
|
+
if (stopped)
|
|
203
|
+
return clearInterval(typingInterval);
|
|
204
|
+
if (channel.isSendable()) {
|
|
205
|
+
await channel.sendTyping().catch(Object);
|
|
206
|
+
}
|
|
207
|
+
};
|
|
208
|
+
const typingInterval = setInterval(runner, 3000).unref();
|
|
209
|
+
await runner();
|
|
210
|
+
return () => {
|
|
211
|
+
stopped = true;
|
|
212
|
+
clearInterval(typingInterval);
|
|
213
|
+
};
|
|
214
|
+
}
|
|
215
|
+
async onBeforeCommandsLoad(ctx) {
|
|
216
|
+
this.toolsRecord = {};
|
|
217
|
+
}
|
|
218
|
+
async onAfterCommandsLoad(ctx) {
|
|
219
|
+
const commands = ctx.commandkit.commandHandler
|
|
220
|
+
.getCommandsArray()
|
|
221
|
+
.filter((command) => 'ai' in command.data &&
|
|
222
|
+
typeof command.data.ai === 'function' &&
|
|
223
|
+
'aiConfig' in command.data);
|
|
224
|
+
if (!commands.length) {
|
|
225
|
+
commandkit_1.Logger.warn('No commands with AI functionality found. Ensure commands are properly configured.');
|
|
226
|
+
return;
|
|
227
|
+
}
|
|
228
|
+
const tools = new discord_js_1.Collection();
|
|
229
|
+
for (const command of commands) {
|
|
230
|
+
const cmd = command;
|
|
231
|
+
if (!cmd.data.ai || !cmd.data.aiConfig) {
|
|
232
|
+
continue;
|
|
233
|
+
}
|
|
234
|
+
const description = cmd.data.aiConfig.description || cmd.data.command.description;
|
|
235
|
+
const cmdTool = (0, ai_1.tool)({
|
|
236
|
+
description,
|
|
237
|
+
type: 'function',
|
|
238
|
+
parameters: cmd.data.aiConfig.parameters,
|
|
239
|
+
async execute(params) {
|
|
240
|
+
const ctx = (0, ai_context_worker_1.getAiWorkerContext)();
|
|
241
|
+
ctx.ctx.setParams(params);
|
|
242
|
+
return cmd.data.ai(ctx.ctx);
|
|
243
|
+
},
|
|
244
|
+
});
|
|
245
|
+
cmd.tool = cmdTool;
|
|
246
|
+
tools.set(cmd.command.name, cmd);
|
|
247
|
+
}
|
|
248
|
+
this.toolsRecord = Object.fromEntries(tools.map((toolCmd) => {
|
|
249
|
+
return [toolCmd.command.name, toolCmd.tool];
|
|
250
|
+
}));
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
exports.AiPlugin = AiPlugin;
|
|
254
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"plugin.js","sourceRoot":"","sources":["../src/plugin.ts"],"names":[],"mappings":"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"}
|
package/dist/types.d.ts
ADDED
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import { LanguageModelV1, ProviderMetadata } from 'ai';
|
|
2
|
+
import { Message } from 'discord.js';
|
|
3
|
+
import { AiContext } from './context';
|
|
4
|
+
export type CommandFilterFunction = (commandName: string) => boolean;
|
|
5
|
+
export type MessageFilter = (message: Message) => Promise<boolean>;
|
|
6
|
+
export type SelectAiModel = (message: Message) => Promise<{
|
|
7
|
+
model: LanguageModelV1;
|
|
8
|
+
options?: ProviderMetadata;
|
|
9
|
+
}>;
|
|
10
|
+
export interface AiPluginOptions {
|
|
11
|
+
}
|
|
12
|
+
export type AiCommand<T extends Record<string, unknown>> = (ctx: AiContext<T>) => Promise<unknown> | unknown;
|
package/dist/types.js
ADDED
package/package.json
ADDED
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@commandkit/ai",
|
|
3
|
+
"version": "0.1.0-dev.20250609135507",
|
|
4
|
+
"description": "Supercharge your CommandKit bot with AI capabilities",
|
|
5
|
+
"files": [
|
|
6
|
+
"dist"
|
|
7
|
+
],
|
|
8
|
+
"main": "dist/index.js",
|
|
9
|
+
"types": "dist/index.d.ts",
|
|
10
|
+
"repository": {
|
|
11
|
+
"type": "git",
|
|
12
|
+
"url": "git+https://github.com/underctrl-io/commandkit.git"
|
|
13
|
+
},
|
|
14
|
+
"keywords": [
|
|
15
|
+
"commandkit",
|
|
16
|
+
"ai"
|
|
17
|
+
],
|
|
18
|
+
"author": "twilight <hello@twlite.dev>",
|
|
19
|
+
"license": "MIT",
|
|
20
|
+
"bugs": {
|
|
21
|
+
"url": "https://github.com/underctrl-io/commandkit/issues"
|
|
22
|
+
},
|
|
23
|
+
"homepage": "https://github.com/underctrl-io/commandkit#readme",
|
|
24
|
+
"devDependencies": {
|
|
25
|
+
"discord.js": "^14.19.3",
|
|
26
|
+
"typescript": "^5.7.3",
|
|
27
|
+
"commandkit": "1.0.0-dev.20250609135507",
|
|
28
|
+
"tsconfig": "0.0.0-dev.20250609135507"
|
|
29
|
+
},
|
|
30
|
+
"dependencies": {
|
|
31
|
+
"ai": "^4.3.16",
|
|
32
|
+
"zod": "^3.25.48"
|
|
33
|
+
},
|
|
34
|
+
"scripts": {
|
|
35
|
+
"lint": "tsc --noEmit",
|
|
36
|
+
"build": "tsc"
|
|
37
|
+
}
|
|
38
|
+
}
|