@jubbio/core 1.1.20 → 1.3.2
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/dist/Client.js +13 -14
- package/dist/cjs/Client.d.ts +157 -0
- package/dist/cjs/Client.js +640 -0
- package/dist/cjs/builders/ActionRowBuilder.d.ts +53 -0
- package/dist/cjs/builders/ActionRowBuilder.js +68 -0
- package/dist/cjs/builders/ButtonBuilder.d.ts +77 -0
- package/dist/cjs/builders/ButtonBuilder.js +96 -0
- package/dist/cjs/builders/EmbedBuilder.d.ts +157 -0
- package/dist/cjs/builders/EmbedBuilder.js +208 -0
- package/dist/cjs/builders/ModalBuilder.d.ts +122 -0
- package/dist/cjs/builders/ModalBuilder.js +162 -0
- package/dist/cjs/builders/SelectMenuBuilder.d.ts +123 -0
- package/dist/cjs/builders/SelectMenuBuilder.js +165 -0
- package/dist/cjs/builders/SlashCommandBuilder.d.ts +197 -0
- package/dist/cjs/builders/SlashCommandBuilder.js +324 -0
- package/dist/cjs/builders/index.d.ts +9 -0
- package/dist/cjs/builders/index.js +26 -0
- package/dist/cjs/enums.d.ts +191 -0
- package/dist/cjs/enums.js +211 -0
- package/dist/cjs/index.d.ts +25 -0
- package/dist/cjs/index.js +130 -0
- package/dist/cjs/managers/BaseManager.d.ts +69 -0
- package/dist/cjs/managers/BaseManager.js +106 -0
- package/dist/cjs/managers/ChannelManager.d.ts +95 -0
- package/dist/cjs/managers/ChannelManager.js +205 -0
- package/dist/cjs/managers/GuildMemberManager.d.ts +74 -0
- package/dist/cjs/managers/GuildMemberManager.js +157 -0
- package/dist/cjs/managers/RoleManager.d.ts +84 -0
- package/dist/cjs/managers/RoleManager.js +207 -0
- package/dist/cjs/managers/index.d.ts +7 -0
- package/dist/cjs/managers/index.js +24 -0
- package/dist/cjs/rest/REST.d.ts +527 -0
- package/dist/cjs/rest/REST.js +904 -0
- package/dist/cjs/rest/index.d.ts +1 -0
- package/dist/cjs/rest/index.js +18 -0
- package/dist/cjs/sharding/ShardingManager.d.ts +179 -0
- package/dist/cjs/sharding/ShardingManager.js +375 -0
- package/dist/cjs/sharding/index.d.ts +4 -0
- package/dist/cjs/sharding/index.js +21 -0
- package/dist/cjs/structures/Channel.d.ts +122 -0
- package/dist/cjs/structures/Channel.js +240 -0
- package/dist/cjs/structures/Collection.d.ts +53 -0
- package/dist/cjs/structures/Collection.js +115 -0
- package/dist/cjs/structures/Guild.d.ts +59 -0
- package/dist/cjs/structures/Guild.js +94 -0
- package/dist/cjs/structures/GuildMember.d.ts +174 -0
- package/dist/cjs/structures/GuildMember.js +311 -0
- package/dist/cjs/structures/Interaction.d.ts +245 -0
- package/dist/cjs/structures/Interaction.js +450 -0
- package/dist/cjs/structures/Message.d.ts +98 -0
- package/dist/cjs/structures/Message.js +195 -0
- package/dist/cjs/structures/User.d.ts +37 -0
- package/dist/cjs/structures/User.js +65 -0
- package/dist/cjs/structures/index.d.ts +7 -0
- package/dist/cjs/structures/index.js +25 -0
- package/dist/cjs/structures.d.ts +1 -0
- package/dist/cjs/structures.js +19 -0
- package/dist/cjs/types.d.ts +255 -0
- package/dist/cjs/types.js +3 -0
- package/dist/cjs/utils/BitField.d.ts +66 -0
- package/dist/cjs/utils/BitField.js +138 -0
- package/dist/cjs/utils/Collection.d.ts +116 -0
- package/dist/cjs/utils/Collection.js +265 -0
- package/dist/cjs/utils/Collector.d.ts +152 -0
- package/dist/cjs/utils/Collector.js +314 -0
- package/dist/cjs/utils/DataResolver.d.ts +61 -0
- package/dist/cjs/utils/DataResolver.js +146 -0
- package/dist/cjs/utils/Formatters.d.ts +145 -0
- package/dist/cjs/utils/Formatters.js +213 -0
- package/dist/cjs/utils/IntentsBitField.d.ts +85 -0
- package/dist/cjs/utils/IntentsBitField.js +99 -0
- package/dist/cjs/utils/Partials.d.ts +104 -0
- package/dist/cjs/utils/Partials.js +148 -0
- package/dist/cjs/utils/PermissionsBitField.d.ts +118 -0
- package/dist/cjs/utils/PermissionsBitField.js +145 -0
- package/dist/cjs/utils/SnowflakeUtil.d.ts +63 -0
- package/dist/cjs/utils/SnowflakeUtil.js +93 -0
- package/dist/cjs/utils/Sweepers.d.ts +119 -0
- package/dist/cjs/utils/Sweepers.js +249 -0
- package/dist/cjs/utils/index.d.ts +13 -0
- package/dist/cjs/utils/index.js +30 -0
- package/dist/esm/Client.js +634 -0
- package/dist/esm/builders/ActionRowBuilder.js +64 -0
- package/dist/esm/builders/ButtonBuilder.js +92 -0
- package/dist/esm/builders/EmbedBuilder.js +204 -0
- package/dist/esm/builders/ModalBuilder.js +157 -0
- package/dist/esm/builders/SelectMenuBuilder.js +161 -0
- package/dist/esm/builders/SlashCommandBuilder.js +311 -0
- package/dist/esm/builders/index.js +10 -0
- package/dist/esm/enums.js +208 -0
- package/dist/esm/index.js +34 -0
- package/dist/esm/managers/BaseManager.js +100 -0
- package/dist/esm/managers/ChannelManager.js +200 -0
- package/dist/esm/managers/GuildMemberManager.js +153 -0
- package/dist/esm/managers/RoleManager.js +203 -0
- package/dist/esm/managers/index.js +8 -0
- package/dist/esm/rest/REST.js +900 -0
- package/dist/esm/rest/index.js +2 -0
- package/dist/esm/sharding/ShardingManager.js +366 -0
- package/dist/esm/sharding/index.js +5 -0
- package/dist/esm/structures/Channel.js +232 -0
- package/dist/esm/structures/Collection.js +111 -0
- package/dist/esm/structures/Guild.js +90 -0
- package/dist/esm/structures/GuildMember.js +306 -0
- package/dist/esm/structures/Interaction.js +438 -0
- package/dist/esm/structures/Message.js +191 -0
- package/dist/esm/structures/User.js +61 -0
- package/dist/esm/structures/index.js +9 -0
- package/dist/esm/structures.js +3 -0
- package/dist/esm/types.js +2 -0
- package/dist/esm/utils/BitField.js +134 -0
- package/dist/esm/utils/Collection.js +261 -0
- package/dist/esm/utils/Collector.js +305 -0
- package/dist/esm/utils/DataResolver.js +142 -0
- package/dist/esm/utils/Formatters.js +191 -0
- package/dist/esm/utils/IntentsBitField.js +93 -0
- package/dist/esm/utils/Partials.js +137 -0
- package/dist/esm/utils/PermissionsBitField.js +141 -0
- package/dist/esm/utils/SnowflakeUtil.js +89 -0
- package/dist/esm/utils/Sweepers.js +245 -0
- package/dist/esm/utils/index.js +14 -0
- package/dist/rest/REST.js +8 -1
- package/dist/structures/Interaction.js +3 -5
- package/dist/structures/Message.js +56 -2
- package/package.json +14 -4
|
@@ -0,0 +1,900 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* REST API client for Jubbio
|
|
3
|
+
*/
|
|
4
|
+
export class REST {
|
|
5
|
+
baseUrl;
|
|
6
|
+
token = '';
|
|
7
|
+
// User cache for mention resolution (ID -> username)
|
|
8
|
+
userCache = new Map();
|
|
9
|
+
USER_CACHE_TTL = 5 * 60 * 1000; // 5 dakika
|
|
10
|
+
constructor(baseUrl = 'https://gateway.jubbio.com/api/v1') {
|
|
11
|
+
this.baseUrl = baseUrl;
|
|
12
|
+
}
|
|
13
|
+
// ==================== Mention Helpers ====================
|
|
14
|
+
/**
|
|
15
|
+
* Cache a user for mention resolution
|
|
16
|
+
* Bot'lar interaction'dan gelen user bilgisini cache'leyebilir
|
|
17
|
+
*/
|
|
18
|
+
cacheUser(user) {
|
|
19
|
+
const userId = typeof user.id === 'string' ? parseInt(user.id, 10) : user.id;
|
|
20
|
+
this.userCache.set(userId, {
|
|
21
|
+
id: userId,
|
|
22
|
+
username: user.username,
|
|
23
|
+
displayName: user.displayName || user.display_name,
|
|
24
|
+
cachedAt: Date.now()
|
|
25
|
+
});
|
|
26
|
+
}
|
|
27
|
+
/**
|
|
28
|
+
* Cache multiple users
|
|
29
|
+
*/
|
|
30
|
+
cacheUsers(users) {
|
|
31
|
+
users.forEach(user => this.cacheUser(user));
|
|
32
|
+
}
|
|
33
|
+
/**
|
|
34
|
+
* Get cached user by ID
|
|
35
|
+
*/
|
|
36
|
+
getCachedUser(userId) {
|
|
37
|
+
const cached = this.userCache.get(userId);
|
|
38
|
+
if (cached && Date.now() - cached.cachedAt < this.USER_CACHE_TTL) {
|
|
39
|
+
return cached;
|
|
40
|
+
}
|
|
41
|
+
// Expired, remove from cache
|
|
42
|
+
if (cached) {
|
|
43
|
+
this.userCache.delete(userId);
|
|
44
|
+
}
|
|
45
|
+
return undefined;
|
|
46
|
+
}
|
|
47
|
+
/**
|
|
48
|
+
* Format a user mention
|
|
49
|
+
* Returns both the text format and mentions data
|
|
50
|
+
*
|
|
51
|
+
* @example
|
|
52
|
+
* const mention = rest.formatMention(user);
|
|
53
|
+
* // mention.text = "<@1>"
|
|
54
|
+
* // mention.data = { users: [{ id: 1, username: "ilkay" }] }
|
|
55
|
+
*/
|
|
56
|
+
formatMention(user) {
|
|
57
|
+
const userId = typeof user.id === 'string' ? parseInt(user.id, 10) : user.id;
|
|
58
|
+
return {
|
|
59
|
+
text: `<@${userId}>`,
|
|
60
|
+
data: {
|
|
61
|
+
users: [{ id: userId, username: user.username }]
|
|
62
|
+
}
|
|
63
|
+
};
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* Parse mentions (<@ID>) in content and build mentions data structure
|
|
67
|
+
* Content is kept as-is with <@ID> format (client renders them)
|
|
68
|
+
*
|
|
69
|
+
* @param content - Message content with mentions
|
|
70
|
+
* @param existingMentions - Existing mentions data to merge with
|
|
71
|
+
* @returns Original content and mentions data
|
|
72
|
+
*/
|
|
73
|
+
processMentions(content, existingMentions) {
|
|
74
|
+
const mentions = {
|
|
75
|
+
users: [...(existingMentions?.users || [])],
|
|
76
|
+
roles: [...(existingMentions?.roles || [])],
|
|
77
|
+
everyone: existingMentions?.everyone
|
|
78
|
+
};
|
|
79
|
+
// Track already added user IDs to avoid duplicates
|
|
80
|
+
const addedUserIds = new Set(mentions.users?.map(u => u.id) || []);
|
|
81
|
+
// Parse <@ID> format (user mentions) — keep content as-is, only build mentions data
|
|
82
|
+
const userMentionRegex = /<@!?(\d+)>/g;
|
|
83
|
+
let match;
|
|
84
|
+
while ((match = userMentionRegex.exec(content)) !== null) {
|
|
85
|
+
const userId = parseInt(match[1], 10);
|
|
86
|
+
if (!addedUserIds.has(userId)) {
|
|
87
|
+
// Try to get username from cache for mentions data
|
|
88
|
+
const cachedUser = this.getCachedUser(userId);
|
|
89
|
+
mentions.users.push({ id: userId, username: cachedUser?.username || `User_${userId}` });
|
|
90
|
+
addedUserIds.add(userId);
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
// Parse <@&ID> format (role mentions) — keep content as-is
|
|
94
|
+
const roleMentionRegex = /<@&(\d+)>/g;
|
|
95
|
+
const addedRoleIds = new Set(mentions.roles?.map(r => r.id) || []);
|
|
96
|
+
while ((match = roleMentionRegex.exec(content)) !== null) {
|
|
97
|
+
const roleId = match[1];
|
|
98
|
+
if (!addedRoleIds.has(roleId)) {
|
|
99
|
+
mentions.roles.push({ id: roleId });
|
|
100
|
+
addedRoleIds.add(roleId);
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
// Parse @everyone and @here
|
|
104
|
+
if (content.includes('@everyone')) {
|
|
105
|
+
mentions.everyone = true;
|
|
106
|
+
}
|
|
107
|
+
// Clean up empty arrays
|
|
108
|
+
if (mentions.users?.length === 0)
|
|
109
|
+
delete mentions.users;
|
|
110
|
+
if (mentions.roles?.length === 0)
|
|
111
|
+
delete mentions.roles;
|
|
112
|
+
return { content, mentions };
|
|
113
|
+
}
|
|
114
|
+
/**
|
|
115
|
+
* Prepare message data with processed mentions
|
|
116
|
+
* Automatically converts mentions to our format
|
|
117
|
+
*/
|
|
118
|
+
prepareMessageData(data) {
|
|
119
|
+
const result = { ...data };
|
|
120
|
+
// Resolve components (ActionRowBuilder / ButtonBuilder instances)
|
|
121
|
+
if (data.components && data.components.length > 0) {
|
|
122
|
+
result.components = data.components.map((c) => typeof c.toJSON === 'function' ? c.toJSON() : c);
|
|
123
|
+
}
|
|
124
|
+
let allMentions = { ...data.mentions };
|
|
125
|
+
// Process mentions in content if present
|
|
126
|
+
if (data.content) {
|
|
127
|
+
const { content, mentions } = this.processMentions(data.content, allMentions);
|
|
128
|
+
result.content = content;
|
|
129
|
+
allMentions = mentions;
|
|
130
|
+
}
|
|
131
|
+
// Process mentions in embeds (description, title, footer, fields)
|
|
132
|
+
if (data.embeds && data.embeds.length > 0) {
|
|
133
|
+
result.embeds = data.embeds.map(embed => {
|
|
134
|
+
// Support EmbedBuilder instances - extract raw data via toJSON()
|
|
135
|
+
const rawEmbed = typeof embed.toJSON === 'function' ? embed.toJSON() : embed;
|
|
136
|
+
const processedEmbed = { ...rawEmbed };
|
|
137
|
+
// Normalize thumbnail: string → { url: string }
|
|
138
|
+
if (typeof processedEmbed.thumbnail === 'string') {
|
|
139
|
+
processedEmbed.thumbnail = { url: processedEmbed.thumbnail };
|
|
140
|
+
}
|
|
141
|
+
// Normalize image: string → { url: string }
|
|
142
|
+
if (typeof processedEmbed.image === 'string') {
|
|
143
|
+
processedEmbed.image = { url: processedEmbed.image };
|
|
144
|
+
}
|
|
145
|
+
// Normalize color: hex string → number
|
|
146
|
+
if (typeof processedEmbed.color === 'string') {
|
|
147
|
+
processedEmbed.color = parseInt(processedEmbed.color.replace('#', ''), 16);
|
|
148
|
+
}
|
|
149
|
+
// Normalize timestamp: Date → ISO string
|
|
150
|
+
if (processedEmbed.timestamp instanceof Date) {
|
|
151
|
+
processedEmbed.timestamp = processedEmbed.timestamp.toISOString();
|
|
152
|
+
}
|
|
153
|
+
else if (typeof processedEmbed.timestamp === 'number') {
|
|
154
|
+
processedEmbed.timestamp = new Date(processedEmbed.timestamp).toISOString();
|
|
155
|
+
}
|
|
156
|
+
// Normalize footer.iconURL → footer.icon_url
|
|
157
|
+
if (processedEmbed.footer) {
|
|
158
|
+
if (processedEmbed.footer.iconURL && !processedEmbed.footer.icon_url) {
|
|
159
|
+
processedEmbed.footer = { ...processedEmbed.footer, icon_url: processedEmbed.footer.iconURL };
|
|
160
|
+
delete processedEmbed.footer.iconURL;
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
// Normalize author.iconURL → author.icon_url
|
|
164
|
+
if (processedEmbed.author) {
|
|
165
|
+
if (processedEmbed.author.iconURL && !processedEmbed.author.icon_url) {
|
|
166
|
+
processedEmbed.author = { ...processedEmbed.author, icon_url: processedEmbed.author.iconURL };
|
|
167
|
+
delete processedEmbed.author.iconURL;
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
// Process description
|
|
171
|
+
if (rawEmbed.description) {
|
|
172
|
+
const { content, mentions } = this.processMentions(rawEmbed.description, allMentions);
|
|
173
|
+
processedEmbed.description = content;
|
|
174
|
+
allMentions = mentions;
|
|
175
|
+
}
|
|
176
|
+
// Process title
|
|
177
|
+
if (rawEmbed.title) {
|
|
178
|
+
const { content, mentions } = this.processMentions(rawEmbed.title, allMentions);
|
|
179
|
+
processedEmbed.title = content;
|
|
180
|
+
allMentions = mentions;
|
|
181
|
+
}
|
|
182
|
+
// Process footer text
|
|
183
|
+
if (rawEmbed.footer?.text) {
|
|
184
|
+
const { content, mentions } = this.processMentions(rawEmbed.footer.text, allMentions);
|
|
185
|
+
processedEmbed.footer = { ...rawEmbed.footer, text: content };
|
|
186
|
+
allMentions = mentions;
|
|
187
|
+
}
|
|
188
|
+
// Process fields
|
|
189
|
+
if (rawEmbed.fields && rawEmbed.fields.length > 0) {
|
|
190
|
+
processedEmbed.fields = rawEmbed.fields.map((field) => {
|
|
191
|
+
const processedField = { ...field };
|
|
192
|
+
if (field.value) {
|
|
193
|
+
const { content, mentions } = this.processMentions(field.value, allMentions);
|
|
194
|
+
processedField.value = content;
|
|
195
|
+
allMentions = mentions;
|
|
196
|
+
}
|
|
197
|
+
if (field.name) {
|
|
198
|
+
const { content, mentions } = this.processMentions(field.name, allMentions);
|
|
199
|
+
processedField.name = content;
|
|
200
|
+
allMentions = mentions;
|
|
201
|
+
}
|
|
202
|
+
return processedField;
|
|
203
|
+
});
|
|
204
|
+
}
|
|
205
|
+
return processedEmbed;
|
|
206
|
+
});
|
|
207
|
+
}
|
|
208
|
+
// Add merged mentions to result
|
|
209
|
+
if (allMentions.users?.length || allMentions.roles?.length || allMentions.everyone) {
|
|
210
|
+
result.mentions = allMentions;
|
|
211
|
+
}
|
|
212
|
+
return result;
|
|
213
|
+
}
|
|
214
|
+
/**
|
|
215
|
+
* Set the bot token
|
|
216
|
+
*/
|
|
217
|
+
setToken(token) {
|
|
218
|
+
this.token = token;
|
|
219
|
+
return this;
|
|
220
|
+
}
|
|
221
|
+
/**
|
|
222
|
+
* Make an authenticated request
|
|
223
|
+
*/
|
|
224
|
+
async request(method, path, body) {
|
|
225
|
+
const url = `${this.baseUrl}${path}`;
|
|
226
|
+
// Debug log
|
|
227
|
+
console.log(`[REST] ${method} ${url}`, body ? JSON.stringify(body) : '');
|
|
228
|
+
const response = await fetch(url, {
|
|
229
|
+
method,
|
|
230
|
+
headers: {
|
|
231
|
+
'Authorization': `Bot ${this.token}`,
|
|
232
|
+
'Content-Type': 'application/json'
|
|
233
|
+
},
|
|
234
|
+
body: body ? JSON.stringify(body) : undefined
|
|
235
|
+
});
|
|
236
|
+
if (!response.ok) {
|
|
237
|
+
const error = await response.text();
|
|
238
|
+
throw new Error(`API Error ${response.status}: ${error}`);
|
|
239
|
+
}
|
|
240
|
+
// Handle empty responses
|
|
241
|
+
const text = await response.text();
|
|
242
|
+
if (!text)
|
|
243
|
+
return {};
|
|
244
|
+
return JSON.parse(text);
|
|
245
|
+
}
|
|
246
|
+
// ==================== Messages ====================
|
|
247
|
+
/**
|
|
248
|
+
* Create a message in a channel
|
|
249
|
+
* Mentions use <@ID> format and are kept as-is (client renders them)
|
|
250
|
+
*
|
|
251
|
+
* @example
|
|
252
|
+
* await rest.createMessage(guildId, channelId, {
|
|
253
|
+
* content: 'Hello <@123>!',
|
|
254
|
+
* });
|
|
255
|
+
*/
|
|
256
|
+
async createMessage(guildIdOrChannelId, channelIdOrData, data) {
|
|
257
|
+
// İki kullanım şekli:
|
|
258
|
+
// 1. createMessage(guildId, channelId, data) - guildId ile (tercih edilen)
|
|
259
|
+
// 2. createMessage(channelId, data) - guildId olmadan (eski format, hata verir)
|
|
260
|
+
let guildId;
|
|
261
|
+
let channelId;
|
|
262
|
+
let messageData;
|
|
263
|
+
if (typeof channelIdOrData === 'string' && data) {
|
|
264
|
+
// Yeni format: createMessage(guildId, channelId, data)
|
|
265
|
+
guildId = guildIdOrChannelId;
|
|
266
|
+
channelId = channelIdOrData;
|
|
267
|
+
messageData = this.prepareMessageData(data);
|
|
268
|
+
// Add interaction_id if provided
|
|
269
|
+
if (data.interactionId) {
|
|
270
|
+
messageData.interaction_id = data.interactionId;
|
|
271
|
+
}
|
|
272
|
+
}
|
|
273
|
+
else if (typeof channelIdOrData === 'object') {
|
|
274
|
+
// Eski format: createMessage(channelId, data) - guildId yok
|
|
275
|
+
// Bu format artık desteklenmiyor, hata fırlat
|
|
276
|
+
throw new Error('createMessage requires guildId: createMessage(guildId, channelId, data)');
|
|
277
|
+
}
|
|
278
|
+
else {
|
|
279
|
+
throw new Error('Invalid createMessage arguments');
|
|
280
|
+
}
|
|
281
|
+
return this.request('POST', `/bot/guilds/${guildId}/channels/${channelId}/messages`, messageData);
|
|
282
|
+
}
|
|
283
|
+
/**
|
|
284
|
+
* Create an ephemeral message that is only visible to a specific user
|
|
285
|
+
* Ephemeral messages are NOT saved to database - they are only sent via WebSocket
|
|
286
|
+
*
|
|
287
|
+
* @example
|
|
288
|
+
* // Send a warning only visible to the user
|
|
289
|
+
* await rest.createEphemeralMessage(guildId, channelId, targetUserId, {
|
|
290
|
+
* embeds: [warningEmbed]
|
|
291
|
+
* });
|
|
292
|
+
*/
|
|
293
|
+
async createEphemeralMessage(guildId, channelId, targetUserId, data) {
|
|
294
|
+
const messageData = this.prepareMessageData(data);
|
|
295
|
+
return this.request('POST', `/bot/guilds/${guildId}/channels/${channelId}/messages`, {
|
|
296
|
+
...messageData,
|
|
297
|
+
flags: 64, // EPHEMERAL flag
|
|
298
|
+
target_user_id: typeof targetUserId === 'string' ? parseInt(targetUserId, 10) : targetUserId
|
|
299
|
+
});
|
|
300
|
+
}
|
|
301
|
+
/**
|
|
302
|
+
* Create a DM message
|
|
303
|
+
*/
|
|
304
|
+
async createDMMessage(channelId, data) {
|
|
305
|
+
return this.request('POST', `/bot/dm/${channelId}`, data);
|
|
306
|
+
}
|
|
307
|
+
/**
|
|
308
|
+
* Edit a message
|
|
309
|
+
* Automatically processes mentions
|
|
310
|
+
*/
|
|
311
|
+
async editMessage(guildId, channelId, messageId, data) {
|
|
312
|
+
const path = `/bot/guilds/${guildId}/channels/${channelId}/messages/${messageId}`;
|
|
313
|
+
const processedData = this.prepareMessageData(data);
|
|
314
|
+
return this.request('PATCH', path, processedData);
|
|
315
|
+
}
|
|
316
|
+
/**
|
|
317
|
+
* Delete a message
|
|
318
|
+
*/
|
|
319
|
+
async deleteMessage(guildId, channelId, messageId) {
|
|
320
|
+
const path = `/bot/guilds/${guildId}/channels/${channelId}/messages/${messageId}`;
|
|
321
|
+
await this.request('DELETE', path);
|
|
322
|
+
}
|
|
323
|
+
/**
|
|
324
|
+
* Validate and normalize emoji format for the API.
|
|
325
|
+
* Accepted formats: :name:, <:name:id>, <a:name:id>
|
|
326
|
+
* Unicode emoji characters (👍) are NOT supported.
|
|
327
|
+
*/
|
|
328
|
+
validateEmoji(emoji) {
|
|
329
|
+
const trimmed = emoji.trim();
|
|
330
|
+
// Custom emoji: <:name:id> or <a:name:id>
|
|
331
|
+
if (/^<a?:\w+:\d+>$/.test(trimmed))
|
|
332
|
+
return trimmed;
|
|
333
|
+
// Unicode emoji by name: :name:
|
|
334
|
+
if (/^:\w+:$/.test(trimmed))
|
|
335
|
+
return trimmed;
|
|
336
|
+
throw new Error(`Geçersiz emoji formatı: "${emoji}". Kabul edilen formatlar: :emoji_name:, <:name:id>, <a:name:id>`);
|
|
337
|
+
}
|
|
338
|
+
/**
|
|
339
|
+
* Add a reaction to a message
|
|
340
|
+
* @param emoji - Emoji in :name:, <:name:id>, or <a:name:id> format. Unicode characters (👍) are not supported.
|
|
341
|
+
*/
|
|
342
|
+
async addReaction(guildId, channelId, messageId, emoji) {
|
|
343
|
+
const validated = this.validateEmoji(emoji);
|
|
344
|
+
const path = `/bot/guilds/${guildId}/channels/${channelId}/messages/${messageId}/reactions/${encodeURIComponent(validated)}/@me`;
|
|
345
|
+
await this.request('PUT', path);
|
|
346
|
+
}
|
|
347
|
+
/**
|
|
348
|
+
* Remove a reaction from a message
|
|
349
|
+
* @param emoji - Emoji in :name:, <:name:id>, or <a:name:id> format. Unicode characters (👍) are not supported.
|
|
350
|
+
*/
|
|
351
|
+
async removeReaction(guildId, channelId, messageId, emoji) {
|
|
352
|
+
const validated = this.validateEmoji(emoji);
|
|
353
|
+
const path = `/bot/guilds/${guildId}/channels/${channelId}/messages/${messageId}/reactions/${encodeURIComponent(validated)}/@me`;
|
|
354
|
+
await this.request('DELETE', path);
|
|
355
|
+
}
|
|
356
|
+
/**
|
|
357
|
+
* Upload an attachment to a channel
|
|
358
|
+
*/
|
|
359
|
+
async uploadAttachment(guildId, channelId, file) {
|
|
360
|
+
const FormData = require('form-data');
|
|
361
|
+
const form = new FormData();
|
|
362
|
+
// form-data expects the buffer directly with options
|
|
363
|
+
form.append('file', file.data, {
|
|
364
|
+
filename: file.name,
|
|
365
|
+
contentType: file.contentType || 'text/plain'
|
|
366
|
+
});
|
|
367
|
+
const url = `${this.baseUrl}/bot/guilds/${guildId}/channels/${channelId}/attachments`;
|
|
368
|
+
console.log(`[REST] Uploading attachment: ${file.name} (${file.data.length} bytes)`);
|
|
369
|
+
const response = await fetch(url, {
|
|
370
|
+
method: 'POST',
|
|
371
|
+
headers: {
|
|
372
|
+
'Authorization': `Bot ${this.token}`,
|
|
373
|
+
...form.getHeaders()
|
|
374
|
+
},
|
|
375
|
+
body: form.getBuffer()
|
|
376
|
+
});
|
|
377
|
+
if (!response.ok) {
|
|
378
|
+
const error = await response.text();
|
|
379
|
+
throw new Error(`API Error ${response.status}: ${error}`);
|
|
380
|
+
}
|
|
381
|
+
return response.json();
|
|
382
|
+
}
|
|
383
|
+
/**
|
|
384
|
+
* Create a message with a file attachment
|
|
385
|
+
*/
|
|
386
|
+
async createMessageWithFile(guildId, channelId, data) {
|
|
387
|
+
const FormData = require('form-data');
|
|
388
|
+
const form = new FormData();
|
|
389
|
+
// Add content if provided
|
|
390
|
+
if (data.content) {
|
|
391
|
+
form.append('content', data.content);
|
|
392
|
+
}
|
|
393
|
+
// Add interaction_id if provided (for deferred response matching)
|
|
394
|
+
if (data.interactionId) {
|
|
395
|
+
form.append('interaction_id', data.interactionId);
|
|
396
|
+
}
|
|
397
|
+
// Add file
|
|
398
|
+
form.append('files', data.file.data, {
|
|
399
|
+
filename: data.file.name,
|
|
400
|
+
contentType: data.file.contentType || 'text/plain'
|
|
401
|
+
});
|
|
402
|
+
const url = `${this.baseUrl}/bot/guilds/${guildId}/channels/${channelId}/messages`;
|
|
403
|
+
console.log(`[REST] Creating message with file: ${data.file.name} (${data.file.data.length} bytes)${data.interactionId ? ` [interaction: ${data.interactionId}]` : ''}`);
|
|
404
|
+
const response = await fetch(url, {
|
|
405
|
+
method: 'POST',
|
|
406
|
+
headers: {
|
|
407
|
+
'Authorization': `Bot ${this.token}`,
|
|
408
|
+
...form.getHeaders()
|
|
409
|
+
},
|
|
410
|
+
body: form.getBuffer()
|
|
411
|
+
});
|
|
412
|
+
if (!response.ok) {
|
|
413
|
+
const error = await response.text();
|
|
414
|
+
throw new Error(`API Error ${response.status}: ${error}`);
|
|
415
|
+
}
|
|
416
|
+
return response.json();
|
|
417
|
+
}
|
|
418
|
+
// ==================== Interactions ====================
|
|
419
|
+
/**
|
|
420
|
+
* Create an interaction response
|
|
421
|
+
* Automatically processes mentions in content and embeds
|
|
422
|
+
*/
|
|
423
|
+
async createInteractionResponse(interactionId, token, data) {
|
|
424
|
+
console.log(`📤 Interaction response: ${interactionId} -> type ${data.type}`);
|
|
425
|
+
try {
|
|
426
|
+
// Process mentions in response data if present
|
|
427
|
+
let processedData = data;
|
|
428
|
+
if (data.data && (data.data.content || data.data.embeds)) {
|
|
429
|
+
processedData = {
|
|
430
|
+
...data,
|
|
431
|
+
data: this.prepareMessageData(data.data)
|
|
432
|
+
};
|
|
433
|
+
}
|
|
434
|
+
await this.request('POST', `/interactions/${interactionId}/${token}/callback`, processedData);
|
|
435
|
+
console.log(`✅ Interaction response sent`);
|
|
436
|
+
}
|
|
437
|
+
catch (error) {
|
|
438
|
+
console.error(`❌ Interaction response error:`, error);
|
|
439
|
+
throw error;
|
|
440
|
+
}
|
|
441
|
+
}
|
|
442
|
+
/**
|
|
443
|
+
* Edit the original interaction response
|
|
444
|
+
* If files are provided, creates a new message with files (since webhook edit doesn't support file upload)
|
|
445
|
+
* Automatically processes mentions
|
|
446
|
+
*/
|
|
447
|
+
async editInteractionResponse(token, data, guildId, channelId, interactionId) {
|
|
448
|
+
const appId = this.getApplicationId();
|
|
449
|
+
// Process mentions in content
|
|
450
|
+
const processedData = this.prepareMessageData(data);
|
|
451
|
+
// If files are present and we have guild/channel info, create message with file instead
|
|
452
|
+
if (data.files && data.files.length > 0 && guildId && channelId) {
|
|
453
|
+
console.log(`[REST] editInteractionResponse with ${data.files.length} files - using createMessageWithFile`);
|
|
454
|
+
// Create message with file
|
|
455
|
+
const file = data.files[0]; // For now, support single file
|
|
456
|
+
await this.createMessageWithFile(guildId, channelId, {
|
|
457
|
+
content: processedData.content,
|
|
458
|
+
file: file,
|
|
459
|
+
interactionId: interactionId
|
|
460
|
+
});
|
|
461
|
+
return;
|
|
462
|
+
}
|
|
463
|
+
// If we have guildId, channelId and interactionId, create a new message with interaction_id
|
|
464
|
+
// This is needed because our deferred response doesn't create a message
|
|
465
|
+
if (guildId && channelId && interactionId) {
|
|
466
|
+
console.log(`[REST] editInteractionResponse - creating message with interaction_id: ${interactionId}`);
|
|
467
|
+
const payload = {
|
|
468
|
+
interaction_id: interactionId
|
|
469
|
+
};
|
|
470
|
+
if (processedData.content !== undefined)
|
|
471
|
+
payload.content = processedData.content;
|
|
472
|
+
if (processedData.embeds)
|
|
473
|
+
payload.embeds = processedData.embeds;
|
|
474
|
+
if (processedData.components)
|
|
475
|
+
payload.components = processedData.components;
|
|
476
|
+
if (processedData.mentions)
|
|
477
|
+
payload.mentions = processedData.mentions;
|
|
478
|
+
await this.request('POST', `/bot/guilds/${guildId}/channels/${channelId}/messages`, payload);
|
|
479
|
+
return;
|
|
480
|
+
}
|
|
481
|
+
// Fallback: Regular edit without files (webhook PATCH)
|
|
482
|
+
const payload = {};
|
|
483
|
+
if (processedData.content !== undefined)
|
|
484
|
+
payload.content = processedData.content;
|
|
485
|
+
if (processedData.embeds)
|
|
486
|
+
payload.embeds = processedData.embeds;
|
|
487
|
+
if (processedData.components)
|
|
488
|
+
payload.components = processedData.components;
|
|
489
|
+
if (processedData.mentions)
|
|
490
|
+
payload.mentions = processedData.mentions;
|
|
491
|
+
await this.request('PATCH', `/interactions/webhooks/${appId}/${token}/messages/@original`, payload);
|
|
492
|
+
}
|
|
493
|
+
/**
|
|
494
|
+
* Delete the original interaction response
|
|
495
|
+
*/
|
|
496
|
+
async deleteInteractionResponse(token) {
|
|
497
|
+
const appId = this.getApplicationId();
|
|
498
|
+
await this.request('DELETE', `/interactions/webhooks/${appId}/${token}/messages/@original`);
|
|
499
|
+
}
|
|
500
|
+
/**
|
|
501
|
+
* Create a followup message
|
|
502
|
+
* Automatically processes mentions
|
|
503
|
+
*/
|
|
504
|
+
async createFollowup(token, data) {
|
|
505
|
+
const appId = this.getApplicationId();
|
|
506
|
+
const processedData = this.prepareMessageData(data);
|
|
507
|
+
await this.request('POST', `/interactions/webhooks/${appId}/${token}`, processedData);
|
|
508
|
+
}
|
|
509
|
+
// ==================== Commands ====================
|
|
510
|
+
/**
|
|
511
|
+
* Register global application commands
|
|
512
|
+
*/
|
|
513
|
+
async registerGlobalCommands(commands) {
|
|
514
|
+
const appId = this.getApplicationId();
|
|
515
|
+
for (const command of commands) {
|
|
516
|
+
await this.request('POST', `/applications/${appId}/commands`, command);
|
|
517
|
+
}
|
|
518
|
+
}
|
|
519
|
+
/**
|
|
520
|
+
* Register guild-specific commands
|
|
521
|
+
*/
|
|
522
|
+
async registerGuildCommands(guildId, commands) {
|
|
523
|
+
const appId = this.getApplicationId();
|
|
524
|
+
for (const command of commands) {
|
|
525
|
+
await this.request('POST', `/applications/${appId}/guilds/${guildId}/commands`, command);
|
|
526
|
+
}
|
|
527
|
+
}
|
|
528
|
+
/**
|
|
529
|
+
* Delete a global command
|
|
530
|
+
*/
|
|
531
|
+
async deleteGlobalCommand(commandId) {
|
|
532
|
+
const appId = this.getApplicationId();
|
|
533
|
+
await this.request('DELETE', `/applications/${appId}/commands/${commandId}`);
|
|
534
|
+
}
|
|
535
|
+
/**
|
|
536
|
+
* Delete a guild-specific command
|
|
537
|
+
*/
|
|
538
|
+
async deleteGuildCommand(guildId, commandId) {
|
|
539
|
+
const appId = this.getApplicationId();
|
|
540
|
+
await this.request('DELETE', `/applications/${appId}/guilds/${guildId}/commands/${commandId}`);
|
|
541
|
+
}
|
|
542
|
+
/**
|
|
543
|
+
* List all global commands for this application
|
|
544
|
+
*/
|
|
545
|
+
async listGlobalCommands() {
|
|
546
|
+
const appId = this.getApplicationId();
|
|
547
|
+
return this.request('GET', `/applications/${appId}/commands`);
|
|
548
|
+
}
|
|
549
|
+
/**
|
|
550
|
+
* List all guild-specific commands for this application
|
|
551
|
+
*/
|
|
552
|
+
async listGuildCommands(guildId) {
|
|
553
|
+
const appId = this.getApplicationId();
|
|
554
|
+
return this.request('GET', `/applications/${appId}/guilds/${guildId}/commands`);
|
|
555
|
+
}
|
|
556
|
+
/**
|
|
557
|
+
* Get a specific global command
|
|
558
|
+
*/
|
|
559
|
+
async getGlobalCommand(commandId) {
|
|
560
|
+
const appId = this.getApplicationId();
|
|
561
|
+
return this.request('GET', `/applications/${appId}/commands/${commandId}`);
|
|
562
|
+
}
|
|
563
|
+
/**
|
|
564
|
+
* Get a specific guild command
|
|
565
|
+
*/
|
|
566
|
+
async getGuildCommand(guildId, commandId) {
|
|
567
|
+
const appId = this.getApplicationId();
|
|
568
|
+
return this.request('GET', `/applications/${appId}/guilds/${guildId}/commands/${commandId}`);
|
|
569
|
+
}
|
|
570
|
+
/**
|
|
571
|
+
* Update a global command
|
|
572
|
+
*/
|
|
573
|
+
async updateGlobalCommand(commandId, data) {
|
|
574
|
+
const appId = this.getApplicationId();
|
|
575
|
+
return this.request('PATCH', `/applications/${appId}/commands/${commandId}`, data);
|
|
576
|
+
}
|
|
577
|
+
/**
|
|
578
|
+
* Update a guild-specific command
|
|
579
|
+
*/
|
|
580
|
+
async updateGuildCommand(guildId, commandId, data) {
|
|
581
|
+
const appId = this.getApplicationId();
|
|
582
|
+
return this.request('PATCH', `/applications/${appId}/guilds/${guildId}/commands/${commandId}`, data);
|
|
583
|
+
}
|
|
584
|
+
// ==================== Helpers ====================
|
|
585
|
+
applicationId = '';
|
|
586
|
+
/**
|
|
587
|
+
* Set the application ID
|
|
588
|
+
*/
|
|
589
|
+
setApplicationId(id) {
|
|
590
|
+
this.applicationId = id;
|
|
591
|
+
}
|
|
592
|
+
/**
|
|
593
|
+
* Get the application ID
|
|
594
|
+
*/
|
|
595
|
+
getApplicationId() {
|
|
596
|
+
if (!this.applicationId) {
|
|
597
|
+
throw new Error('Application ID not set. Call setApplicationId() first.');
|
|
598
|
+
}
|
|
599
|
+
return this.applicationId;
|
|
600
|
+
}
|
|
601
|
+
// ==================== Channels ====================
|
|
602
|
+
/**
|
|
603
|
+
* Create a channel in a guild
|
|
604
|
+
*/
|
|
605
|
+
async createChannel(guildId, data) {
|
|
606
|
+
// Map parent_id to category_id for backend compatibility
|
|
607
|
+
const requestData = {
|
|
608
|
+
name: data.name,
|
|
609
|
+
type: data.type ?? 0, // Default to text channel
|
|
610
|
+
};
|
|
611
|
+
// Backend expects category_id, not parent_id
|
|
612
|
+
if (data.category_id) {
|
|
613
|
+
requestData.category_id = data.category_id;
|
|
614
|
+
}
|
|
615
|
+
else if (data.parent_id) {
|
|
616
|
+
requestData.category_id = data.parent_id;
|
|
617
|
+
}
|
|
618
|
+
// Add permission_overwrites if provided
|
|
619
|
+
if (data.permission_overwrites && data.permission_overwrites.length > 0) {
|
|
620
|
+
requestData.permission_overwrites = data.permission_overwrites;
|
|
621
|
+
}
|
|
622
|
+
return this.request('POST', `/bot/guilds/${guildId}/channels`, requestData);
|
|
623
|
+
}
|
|
624
|
+
/**
|
|
625
|
+
* Delete a channel
|
|
626
|
+
*/
|
|
627
|
+
/**
|
|
628
|
+
* Delete a channel
|
|
629
|
+
*/
|
|
630
|
+
async deleteChannel(guildId, channelId) {
|
|
631
|
+
await this.request('DELETE', `/bot/guilds/${guildId}/channels/${channelId}`);
|
|
632
|
+
}
|
|
633
|
+
/**
|
|
634
|
+
* Delete a category
|
|
635
|
+
*/
|
|
636
|
+
async deleteCategory(guildId, categoryId) {
|
|
637
|
+
await this.request('DELETE', `/bot/guilds/${guildId}/categories/${categoryId}`);
|
|
638
|
+
}
|
|
639
|
+
/**
|
|
640
|
+
* Edit channel permission overwrites
|
|
641
|
+
*/
|
|
642
|
+
async editChannelPermissions(channelId, overwriteId, data) {
|
|
643
|
+
await this.request('PUT', `/bot/channels/${channelId}/permissions/${overwriteId}`, data);
|
|
644
|
+
}
|
|
645
|
+
/**
|
|
646
|
+
* Delete channel permission overwrite
|
|
647
|
+
*/
|
|
648
|
+
async deleteChannelPermission(channelId, overwriteId) {
|
|
649
|
+
await this.request('DELETE', `/bot/channels/${channelId}/permissions/${overwriteId}`);
|
|
650
|
+
}
|
|
651
|
+
/**
|
|
652
|
+
* Get messages from a channel
|
|
653
|
+
*/
|
|
654
|
+
async getMessages(guildId, channelId, options) {
|
|
655
|
+
const params = new URLSearchParams();
|
|
656
|
+
if (options?.limit)
|
|
657
|
+
params.append('limit', String(options.limit));
|
|
658
|
+
if (options?.before)
|
|
659
|
+
params.append('before', options.before);
|
|
660
|
+
if (options?.after)
|
|
661
|
+
params.append('after', options.after);
|
|
662
|
+
const query = params.toString() ? `?${params.toString()}` : '';
|
|
663
|
+
const response = await this.request('GET', `/bot/guilds/${guildId}/channels/${channelId}/messages${query}`);
|
|
664
|
+
return response.messages || [];
|
|
665
|
+
}
|
|
666
|
+
// ==================== Members ====================
|
|
667
|
+
/**
|
|
668
|
+
* Get a guild member
|
|
669
|
+
*/
|
|
670
|
+
async getMember(guildId, userId) {
|
|
671
|
+
return this.request('GET', `/bot/guilds/${guildId}/members/${userId}`);
|
|
672
|
+
}
|
|
673
|
+
/**
|
|
674
|
+
* Timeout a guild member
|
|
675
|
+
*/
|
|
676
|
+
async timeoutMember(guildId, userId, duration, reason) {
|
|
677
|
+
if (duration === null) {
|
|
678
|
+
// Clear timeout
|
|
679
|
+
await this.request('POST', `/bot/guilds/${guildId}/members/${userId}/timeout/clear`, {
|
|
680
|
+
reason
|
|
681
|
+
});
|
|
682
|
+
}
|
|
683
|
+
else {
|
|
684
|
+
// Set timeout
|
|
685
|
+
const until = new Date(Date.now() + duration).toISOString();
|
|
686
|
+
await this.request('POST', `/bot/guilds/${guildId}/members/${userId}/timeout`, {
|
|
687
|
+
until,
|
|
688
|
+
reason
|
|
689
|
+
});
|
|
690
|
+
}
|
|
691
|
+
}
|
|
692
|
+
/**
|
|
693
|
+
* Kick a guild member
|
|
694
|
+
*/
|
|
695
|
+
async kickMember(guildId, userId, reason) {
|
|
696
|
+
const query = reason ? `?reason=${encodeURIComponent(reason)}` : '';
|
|
697
|
+
await this.request('DELETE', `/bot/guilds/${guildId}/members/${userId}${query}`);
|
|
698
|
+
}
|
|
699
|
+
/**
|
|
700
|
+
* Ban a guild member
|
|
701
|
+
*/
|
|
702
|
+
async banMember(guildId, userId, options) {
|
|
703
|
+
await this.request('PUT', `/bot/guilds/${guildId}/bans/${userId}`, {
|
|
704
|
+
delete_message_days: options?.deleteMessageDays,
|
|
705
|
+
delete_message_seconds: options?.deleteMessageSeconds,
|
|
706
|
+
reason: options?.reason
|
|
707
|
+
});
|
|
708
|
+
}
|
|
709
|
+
/**
|
|
710
|
+
* Unban a user
|
|
711
|
+
*/
|
|
712
|
+
async unbanMember(guildId, userId, reason) {
|
|
713
|
+
const query = reason ? `?reason=${encodeURIComponent(reason)}` : '';
|
|
714
|
+
await this.request('DELETE', `/bot/guilds/${guildId}/bans/${userId}${query}`);
|
|
715
|
+
}
|
|
716
|
+
/**
|
|
717
|
+
* Edit a guild member
|
|
718
|
+
*/
|
|
719
|
+
async editMember(guildId, userId, data) {
|
|
720
|
+
return this.request('PATCH', `/bot/guilds/${guildId}/members/${userId}`, data);
|
|
721
|
+
}
|
|
722
|
+
/**
|
|
723
|
+
* Add a role to a member
|
|
724
|
+
*/
|
|
725
|
+
async addMemberRole(guildId, userId, roleId, reason) {
|
|
726
|
+
const query = reason ? `?reason=${encodeURIComponent(reason)}` : '';
|
|
727
|
+
await this.request('PUT', `/bot/guilds/${guildId}/members/${userId}/roles/${roleId}${query}`);
|
|
728
|
+
}
|
|
729
|
+
/**
|
|
730
|
+
* Remove a role from a member
|
|
731
|
+
*/
|
|
732
|
+
async removeMemberRole(guildId, userId, roleId, reason) {
|
|
733
|
+
const query = reason ? `?reason=${encodeURIComponent(reason)}` : '';
|
|
734
|
+
await this.request('DELETE', `/bot/guilds/${guildId}/members/${userId}/roles/${roleId}${query}`);
|
|
735
|
+
}
|
|
736
|
+
/**
|
|
737
|
+
* Bulk assign roles to members
|
|
738
|
+
*/
|
|
739
|
+
async bulkAssignRoles(guildId, assignments) {
|
|
740
|
+
return this.request('PUT', `/bot/guilds/${guildId}/members/roles/assign`, { assignments });
|
|
741
|
+
}
|
|
742
|
+
/**
|
|
743
|
+
* Bulk remove roles from members
|
|
744
|
+
*/
|
|
745
|
+
async bulkRemoveRoles(guildId, removals) {
|
|
746
|
+
return this.request('DELETE', `/bot/guilds/${guildId}/members/roles/remove`, { removals });
|
|
747
|
+
}
|
|
748
|
+
/**
|
|
749
|
+
* Bulk delete messages
|
|
750
|
+
*/
|
|
751
|
+
async bulkDeleteMessages(guildId, channelId, messageIds) {
|
|
752
|
+
await this.request('POST', `/bot/guilds/${guildId}/channels/${channelId}/messages/bulk-delete`, {
|
|
753
|
+
messages: messageIds
|
|
754
|
+
});
|
|
755
|
+
}
|
|
756
|
+
// ==================== Guilds ====================
|
|
757
|
+
/**
|
|
758
|
+
* Get a guild
|
|
759
|
+
*/
|
|
760
|
+
async getGuild(guildId) {
|
|
761
|
+
return this.request('GET', `/bot/guilds/${guildId}`);
|
|
762
|
+
}
|
|
763
|
+
/**
|
|
764
|
+
* Get guild channels
|
|
765
|
+
*/
|
|
766
|
+
async getGuildChannels(guildId) {
|
|
767
|
+
return this.request('GET', `/bot/guilds/${guildId}/channels`);
|
|
768
|
+
}
|
|
769
|
+
/**
|
|
770
|
+
* Get guild roles
|
|
771
|
+
*/
|
|
772
|
+
async getRoles(guildId) {
|
|
773
|
+
return this.request('GET', `/bot/guilds/${guildId}/roles`);
|
|
774
|
+
}
|
|
775
|
+
/**
|
|
776
|
+
* Create a role
|
|
777
|
+
*/
|
|
778
|
+
async createRole(guildId, data) {
|
|
779
|
+
return this.request('POST', `/bot/guilds/${guildId}/roles`, data);
|
|
780
|
+
}
|
|
781
|
+
/**
|
|
782
|
+
* Edit a role
|
|
783
|
+
*/
|
|
784
|
+
async editRole(guildId, roleId, data) {
|
|
785
|
+
return this.request('PATCH', `/bot/guilds/${guildId}/roles/${roleId}`, data);
|
|
786
|
+
}
|
|
787
|
+
/**
|
|
788
|
+
* Delete a role
|
|
789
|
+
*/
|
|
790
|
+
async deleteRole(guildId, roleId) {
|
|
791
|
+
await this.request('DELETE', `/bot/guilds/${guildId}/roles/${roleId}`);
|
|
792
|
+
}
|
|
793
|
+
/**
|
|
794
|
+
* Get guild emojis
|
|
795
|
+
*/
|
|
796
|
+
async getEmojis(guildId) {
|
|
797
|
+
return this.request('GET', `/bot/guilds/${guildId}/emojis`);
|
|
798
|
+
}
|
|
799
|
+
/**
|
|
800
|
+
* Get guild bans
|
|
801
|
+
*/
|
|
802
|
+
async getBans(guildId) {
|
|
803
|
+
return this.request('GET', `/bot/guilds/${guildId}/bans`);
|
|
804
|
+
}
|
|
805
|
+
/**
|
|
806
|
+
* Get a specific ban
|
|
807
|
+
*/
|
|
808
|
+
async getBan(guildId, userId) {
|
|
809
|
+
return this.request('GET', `/bot/guilds/${guildId}/bans/${userId}`);
|
|
810
|
+
}
|
|
811
|
+
/**
|
|
812
|
+
* Get guild invites
|
|
813
|
+
*/
|
|
814
|
+
async getGuildInvites(guildId) {
|
|
815
|
+
return this.request('GET', `/bot/guilds/${guildId}/invites`);
|
|
816
|
+
}
|
|
817
|
+
// ==================== Pins ====================
|
|
818
|
+
/**
|
|
819
|
+
* Pin a message
|
|
820
|
+
*/
|
|
821
|
+
async pinMessage(guildId, channelId, messageId) {
|
|
822
|
+
await this.request('PUT', `/bot/guilds/${guildId}/channels/${channelId}/pins/${messageId}`);
|
|
823
|
+
}
|
|
824
|
+
/**
|
|
825
|
+
* Unpin a message
|
|
826
|
+
*/
|
|
827
|
+
async unpinMessage(guildId, channelId, messageId) {
|
|
828
|
+
await this.request('DELETE', `/bot/guilds/${guildId}/channels/${channelId}/pins/${messageId}`);
|
|
829
|
+
}
|
|
830
|
+
/**
|
|
831
|
+
* Get pinned messages
|
|
832
|
+
*/
|
|
833
|
+
async getPinnedMessages(guildId, channelId) {
|
|
834
|
+
return this.request('GET', `/bot/guilds/${guildId}/channels/${channelId}/pins`);
|
|
835
|
+
}
|
|
836
|
+
// ==================== Users ====================
|
|
837
|
+
/**
|
|
838
|
+
* Get a user
|
|
839
|
+
*/
|
|
840
|
+
async getUser(userId) {
|
|
841
|
+
return this.request('GET', `/bot/users/${userId}`);
|
|
842
|
+
}
|
|
843
|
+
/**
|
|
844
|
+
* Get current bot user
|
|
845
|
+
*/
|
|
846
|
+
async getCurrentUser() {
|
|
847
|
+
return this.request('GET', `/bot/users/@me`);
|
|
848
|
+
}
|
|
849
|
+
// ==================== Invites ====================
|
|
850
|
+
/**
|
|
851
|
+
* Create an invite
|
|
852
|
+
*/
|
|
853
|
+
async createInvite(guildId, channelId, data) {
|
|
854
|
+
return this.request('POST', `/bot/guilds/${guildId}/channels/${channelId}/invites`, data || {});
|
|
855
|
+
}
|
|
856
|
+
/**
|
|
857
|
+
* Delete an invite
|
|
858
|
+
*/
|
|
859
|
+
async deleteInvite(inviteCode) {
|
|
860
|
+
await this.request('DELETE', `/bot/invites/${inviteCode}`);
|
|
861
|
+
}
|
|
862
|
+
/**
|
|
863
|
+
* Get an invite
|
|
864
|
+
*/
|
|
865
|
+
async getInvite(inviteCode) {
|
|
866
|
+
return this.request('GET', `/bot/invites/${inviteCode}`);
|
|
867
|
+
}
|
|
868
|
+
// ==================== Webhooks ====================
|
|
869
|
+
/**
|
|
870
|
+
* Get channel webhooks
|
|
871
|
+
*/
|
|
872
|
+
async getChannelWebhooks(guildId, channelId) {
|
|
873
|
+
return this.request('GET', `/bot/guilds/${guildId}/channels/${channelId}/webhooks`);
|
|
874
|
+
}
|
|
875
|
+
/**
|
|
876
|
+
* Get guild webhooks
|
|
877
|
+
*/
|
|
878
|
+
async getGuildWebhooks(guildId) {
|
|
879
|
+
return this.request('GET', `/bot/guilds/${guildId}/webhooks`);
|
|
880
|
+
}
|
|
881
|
+
/**
|
|
882
|
+
* Create a webhook
|
|
883
|
+
*/
|
|
884
|
+
async createWebhook(guildId, channelId, data) {
|
|
885
|
+
return this.request('POST', `/bot/guilds/${guildId}/channels/${channelId}/webhooks`, data);
|
|
886
|
+
}
|
|
887
|
+
/**
|
|
888
|
+
* Update a webhook
|
|
889
|
+
*/
|
|
890
|
+
async updateWebhook(guildId, webhookId, data) {
|
|
891
|
+
return this.request('PATCH', `/bot/guilds/${guildId}/webhooks/${webhookId}`, data);
|
|
892
|
+
}
|
|
893
|
+
/**
|
|
894
|
+
* Delete a webhook
|
|
895
|
+
*/
|
|
896
|
+
async deleteWebhook(guildId, webhookId) {
|
|
897
|
+
return this.request('DELETE', `/bot/guilds/${guildId}/webhooks/${webhookId}`);
|
|
898
|
+
}
|
|
899
|
+
}
|
|
900
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"REST.js","sourceRoot":"","sources":["../../../src/rest/REST.ts"],"names":[],"mappings":"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","sourcesContent":["import { APIMessage, APIApplicationCommand, APIEmbed } from '../types';\r\n\r\n/**\r\n * Mention data structure for our system\r\n */\r\nexport interface MentionUser {\r\n  id: number;\r\n  username: string;\r\n}\r\n\r\nexport interface MentionRole {\r\n  id: string;\r\n  name?: string;\r\n}\r\n\r\nexport interface MentionsData {\r\n  users?: MentionUser[];\r\n  roles?: MentionRole[];\r\n  everyone?: boolean;\r\n}\r\n\r\n/**\r\n * User cache entry\r\n */\r\ninterface CachedUser {\r\n  id: number;\r\n  username: string;\r\n  displayName?: string;\r\n  cachedAt: number;\r\n}\r\n\r\n/**\r\n * REST API client for Jubbio\r\n */\r\nexport class REST {\r\n  private baseUrl: string;\r\n  private token: string = '';\r\n  \r\n  // User cache for mention resolution (ID -> username)\r\n  private userCache: Map<number, CachedUser> = new Map();\r\n  private readonly USER_CACHE_TTL = 5 * 60 * 1000; // 5 dakika\r\n\r\n  constructor(baseUrl: string = 'https://gateway.jubbio.com/api/v1') {\r\n    this.baseUrl = baseUrl;\r\n  }\r\n\r\n  // ==================== Mention Helpers ====================\r\n\r\n  /**\r\n   * Cache a user for mention resolution\r\n   * Bot'lar interaction'dan gelen user bilgisini cache'leyebilir\r\n   */\r\n  cacheUser(user: { id: string | number; username: string; displayName?: string; display_name?: string }): void {\r\n    const userId = typeof user.id === 'string' ? parseInt(user.id, 10) : user.id;\r\n    this.userCache.set(userId, {\r\n      id: userId,\r\n      username: user.username,\r\n      displayName: user.displayName || user.display_name,\r\n      cachedAt: Date.now()\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Cache multiple users\r\n   */\r\n  cacheUsers(users: Array<{ id: string | number; username: string; displayName?: string; display_name?: string }>): void {\r\n    users.forEach(user => this.cacheUser(user));\r\n  }\r\n\r\n  /**\r\n   * Get cached user by ID\r\n   */\r\n  getCachedUser(userId: number): CachedUser | undefined {\r\n    const cached = this.userCache.get(userId);\r\n    if (cached && Date.now() - cached.cachedAt < this.USER_CACHE_TTL) {\r\n      return cached;\r\n    }\r\n    // Expired, remove from cache\r\n    if (cached) {\r\n      this.userCache.delete(userId);\r\n    }\r\n    return undefined;\r\n  }\r\n\r\n  /**\r\n   * Format a user mention\r\n   * Returns both the text format and mentions data\r\n   * \r\n   * @example\r\n   * const mention = rest.formatMention(user);\r\n   * // mention.text = \"<@1>\"\r\n   * // mention.data = { users: [{ id: 1, username: \"ilkay\" }] }\r\n   */\r\n  formatMention(user: { id: string | number; username: string }): { text: string; data: MentionsData } {\r\n    const userId = typeof user.id === 'string' ? parseInt(user.id, 10) : user.id;\r\n    return {\r\n      text: `<@${userId}>`,\r\n      data: {\r\n        users: [{ id: userId, username: user.username }]\r\n      }\r\n    };\r\n  }\r\n\r\n  /**\r\n   * Parse mentions (<@ID>) in content and build mentions data structure\r\n   * Content is kept as-is with <@ID> format (client renders them)\r\n   * \r\n   * @param content - Message content with mentions\r\n   * @param existingMentions - Existing mentions data to merge with\r\n   * @returns Original content and mentions data\r\n   */\r\n  private processMentions(content: string, existingMentions?: MentionsData): { content: string; mentions: MentionsData } {\r\n    const mentions: MentionsData = {\r\n      users: [...(existingMentions?.users || [])],\r\n      roles: [...(existingMentions?.roles || [])],\r\n      everyone: existingMentions?.everyone\r\n    };\r\n\r\n    // Track already added user IDs to avoid duplicates\r\n    const addedUserIds = new Set(mentions.users?.map(u => u.id) || []);\r\n\r\n    // Parse <@ID> format (user mentions) — keep content as-is, only build mentions data\r\n    const userMentionRegex = /<@!?(\\d+)>/g;\r\n    let match;\r\n\r\n    while ((match = userMentionRegex.exec(content)) !== null) {\r\n      const userId = parseInt(match[1], 10);\r\n\r\n      if (!addedUserIds.has(userId)) {\r\n        // Try to get username from cache for mentions data\r\n        const cachedUser = this.getCachedUser(userId);\r\n        mentions.users!.push({ id: userId, username: cachedUser?.username || `User_${userId}` });\r\n        addedUserIds.add(userId);\r\n      }\r\n    }\r\n\r\n    // Parse <@&ID> format (role mentions) — keep content as-is\r\n    const roleMentionRegex = /<@&(\\d+)>/g;\r\n    const addedRoleIds = new Set(mentions.roles?.map(r => r.id) || []);\r\n\r\n    while ((match = roleMentionRegex.exec(content)) !== null) {\r\n      const roleId = match[1];\r\n      if (!addedRoleIds.has(roleId)) {\r\n        mentions.roles!.push({ id: roleId });\r\n        addedRoleIds.add(roleId);\r\n      }\r\n    }\r\n\r\n    // Parse @everyone and @here\r\n    if (content.includes('@everyone')) {\r\n      mentions.everyone = true;\r\n    }\r\n\r\n    // Clean up empty arrays\r\n    if (mentions.users?.length === 0) delete mentions.users;\r\n    if (mentions.roles?.length === 0) delete mentions.roles;\r\n\r\n    return { content, mentions };\r\n  }\r\n\r\n  /**\r\n   * Prepare message data with processed mentions\r\n   * Automatically converts mentions to our format\r\n   */\r\n  private prepareMessageData(data: {\r\n    content?: string;\r\n    embeds?: APIEmbed[];\r\n    components?: any[];\r\n    mentions?: MentionsData;\r\n    message_reference?: { message_id: string };\r\n  }): any {\r\n    const result: any = { ...data };\r\n\r\n    // Resolve components (ActionRowBuilder / ButtonBuilder instances)\r\n    if (data.components && data.components.length > 0) {\r\n      result.components = data.components.map((c: any) =>\r\n        typeof c.toJSON === 'function' ? c.toJSON() : c\r\n      );\r\n    }\r\n\r\n    let allMentions: MentionsData = { ...data.mentions };\r\n\r\n    // Process mentions in content if present\r\n    if (data.content) {\r\n      const { content, mentions } = this.processMentions(data.content, allMentions);\r\n      result.content = content;\r\n      allMentions = mentions;\r\n    }\r\n\r\n    // Process mentions in embeds (description, title, footer, fields)\r\n    if (data.embeds && data.embeds.length > 0) {\r\n      result.embeds = data.embeds.map(embed => {\r\n        // Support EmbedBuilder instances - extract raw data via toJSON()\r\n        const rawEmbed = typeof (embed as any).toJSON === 'function' ? (embed as any).toJSON() : embed;\r\n        const processedEmbed: any = { ...rawEmbed };\r\n\r\n        // Normalize thumbnail: string → { url: string }\r\n        if (typeof processedEmbed.thumbnail === 'string') {\r\n          processedEmbed.thumbnail = { url: processedEmbed.thumbnail };\r\n        }\r\n\r\n        // Normalize image: string → { url: string }\r\n        if (typeof processedEmbed.image === 'string') {\r\n          processedEmbed.image = { url: processedEmbed.image };\r\n        }\r\n\r\n        // Normalize color: hex string → number\r\n        if (typeof processedEmbed.color === 'string') {\r\n          processedEmbed.color = parseInt(processedEmbed.color.replace('#', ''), 16);\r\n        }\r\n\r\n        // Normalize timestamp: Date → ISO string\r\n        if (processedEmbed.timestamp instanceof Date) {\r\n          processedEmbed.timestamp = processedEmbed.timestamp.toISOString();\r\n        } else if (typeof processedEmbed.timestamp === 'number') {\r\n          processedEmbed.timestamp = new Date(processedEmbed.timestamp).toISOString();\r\n        }\r\n\r\n        // Normalize footer.iconURL → footer.icon_url\r\n        if (processedEmbed.footer) {\r\n          if (processedEmbed.footer.iconURL && !processedEmbed.footer.icon_url) {\r\n            processedEmbed.footer = { ...processedEmbed.footer, icon_url: processedEmbed.footer.iconURL };\r\n            delete processedEmbed.footer.iconURL;\r\n          }\r\n        }\r\n\r\n        // Normalize author.iconURL → author.icon_url\r\n        if (processedEmbed.author) {\r\n          if (processedEmbed.author.iconURL && !processedEmbed.author.icon_url) {\r\n            processedEmbed.author = { ...processedEmbed.author, icon_url: processedEmbed.author.iconURL };\r\n            delete processedEmbed.author.iconURL;\r\n          }\r\n        }\r\n        \r\n        // Process description\r\n        if (rawEmbed.description) {\r\n          const { content, mentions } = this.processMentions(rawEmbed.description, allMentions);\r\n          processedEmbed.description = content;\r\n          allMentions = mentions;\r\n        }\r\n        \r\n        // Process title\r\n        if (rawEmbed.title) {\r\n          const { content, mentions } = this.processMentions(rawEmbed.title, allMentions);\r\n          processedEmbed.title = content;\r\n          allMentions = mentions;\r\n        }\r\n        \r\n        // Process footer text\r\n        if (rawEmbed.footer?.text) {\r\n          const { content, mentions } = this.processMentions(rawEmbed.footer.text, allMentions);\r\n          processedEmbed.footer = { ...rawEmbed.footer, text: content };\r\n          allMentions = mentions;\r\n        }\r\n        \r\n        // Process fields\r\n        if (rawEmbed.fields && rawEmbed.fields.length > 0) {\r\n          processedEmbed.fields = rawEmbed.fields.map((field: any) => {\r\n            const processedField: any = { ...field };\r\n            if (field.value) {\r\n              const { content, mentions } = this.processMentions(field.value, allMentions);\r\n              processedField.value = content;\r\n              allMentions = mentions;\r\n            }\r\n            if (field.name) {\r\n              const { content, mentions } = this.processMentions(field.name, allMentions);\r\n              processedField.name = content;\r\n              allMentions = mentions;\r\n            }\r\n            return processedField;\r\n          });\r\n        }\r\n        \r\n        return processedEmbed;\r\n      });\r\n    }\r\n\r\n    // Add merged mentions to result\r\n    if (allMentions.users?.length || allMentions.roles?.length || allMentions.everyone) {\r\n      result.mentions = allMentions;\r\n    }\r\n\r\n    return result;\r\n  }\r\n\r\n  /**\r\n   * Set the bot token\r\n   */\r\n  setToken(token: string): this {\r\n    this.token = token;\r\n    return this;\r\n  }\r\n\r\n  /**\r\n   * Make an authenticated request\r\n   */\r\n  private async request<T>(method: string, path: string, body?: any): Promise<T> {\r\n    const url = `${this.baseUrl}${path}`;\r\n    \r\n    // Debug log\r\n    console.log(`[REST] ${method} ${url}`, body ? JSON.stringify(body) : '');\r\n    \r\n    const response = await fetch(url, {\r\n      method,\r\n      headers: {\r\n        'Authorization': `Bot ${this.token}`,\r\n        'Content-Type': 'application/json'\r\n      },\r\n      body: body ? JSON.stringify(body) : undefined\r\n    });\r\n\r\n    if (!response.ok) {\r\n      const error = await response.text();\r\n      throw new Error(`API Error ${response.status}: ${error}`);\r\n    }\r\n\r\n    // Handle empty responses\r\n    const text = await response.text();\r\n    if (!text) return {} as T;\r\n    \r\n    return JSON.parse(text);\r\n  }\r\n\r\n  // ==================== Messages ====================\r\n\r\n  /**\r\n   * Create a message in a channel\r\n   * Mentions use <@ID> format and are kept as-is (client renders them)\r\n   * \r\n   * @example\r\n   * await rest.createMessage(guildId, channelId, {\r\n   *   content: 'Hello <@123>!',\r\n   * });\r\n   */\r\n  async createMessage(guildIdOrChannelId: string, channelIdOrData: string | {\r\n    content?: string;\r\n    embeds?: APIEmbed[];\r\n    components?: any[];\r\n    mentions?: MentionsData;\r\n    files?: Array<{ name: string; data: Buffer }>;\r\n    message_reference?: { message_id: string };\r\n    interactionId?: string;\r\n  }, data?: {\r\n    content?: string;\r\n    embeds?: APIEmbed[];\r\n    components?: any[];\r\n    mentions?: MentionsData;\r\n    files?: Array<{ name: string; data: Buffer }>;\r\n    message_reference?: { message_id: string };\r\n    interactionId?: string;\r\n  }): Promise<APIMessage> {\r\n    // İki kullanım şekli:\r\n    // 1. createMessage(guildId, channelId, data) - guildId ile (tercih edilen)\r\n    // 2. createMessage(channelId, data) - guildId olmadan (eski format, hata verir)\r\n    \r\n    let guildId: string;\r\n    let channelId: string;\r\n    let messageData: any;\r\n    \r\n    if (typeof channelIdOrData === 'string' && data) {\r\n      // Yeni format: createMessage(guildId, channelId, data)\r\n      guildId = guildIdOrChannelId;\r\n      channelId = channelIdOrData;\r\n      messageData = this.prepareMessageData(data);\r\n      \r\n      // Add interaction_id if provided\r\n      if (data.interactionId) {\r\n        messageData.interaction_id = data.interactionId;\r\n      }\r\n    } else if (typeof channelIdOrData === 'object') {\r\n      // Eski format: createMessage(channelId, data) - guildId yok\r\n      // Bu format artık desteklenmiyor, hata fırlat\r\n      throw new Error('createMessage requires guildId: createMessage(guildId, channelId, data)');\r\n    } else {\r\n      throw new Error('Invalid createMessage arguments');\r\n    }\r\n    \r\n    return this.request<APIMessage>('POST', `/bot/guilds/${guildId}/channels/${channelId}/messages`, messageData);\r\n  }\r\n\r\n  /**\r\n   * Create an ephemeral message that is only visible to a specific user\r\n   * Ephemeral messages are NOT saved to database - they are only sent via WebSocket\r\n   * \r\n   * @example\r\n   * // Send a warning only visible to the user\r\n   * await rest.createEphemeralMessage(guildId, channelId, targetUserId, {\r\n   *   embeds: [warningEmbed]\r\n   * });\r\n   */\r\n  async createEphemeralMessage(guildId: string, channelId: string, targetUserId: string | number, data: {\r\n    content?: string;\r\n    embeds?: APIEmbed[];\r\n    components?: any[];\r\n  }): Promise<{ id: string; ephemeral: boolean; flags: number }> {\r\n    const messageData = this.prepareMessageData(data);\r\n    \r\n    return this.request<{ id: string; ephemeral: boolean; flags: number }>('POST', `/bot/guilds/${guildId}/channels/${channelId}/messages`, {\r\n      ...messageData,\r\n      flags: 64, // EPHEMERAL flag\r\n      target_user_id: typeof targetUserId === 'string' ? parseInt(targetUserId, 10) : targetUserId\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Create a DM message\r\n   */\r\n  async createDMMessage(channelId: string, data: {\r\n    content?: string;\r\n    embeds?: APIEmbed[];\r\n  }): Promise<APIMessage> {\r\n    return this.request<APIMessage>('POST', `/bot/dm/${channelId}`, data);\r\n  }\r\n\r\n  /**\r\n   * Edit a message\r\n   * Automatically processes mentions\r\n   */\r\n  async editMessage(guildId: string, channelId: string, messageId: string, data: {\r\n    content?: string;\r\n    embeds?: APIEmbed[];\r\n    components?: any[];\r\n    mentions?: MentionsData;\r\n  }): Promise<APIMessage> {\r\n    const path = `/bot/guilds/${guildId}/channels/${channelId}/messages/${messageId}`;\r\n    const processedData = this.prepareMessageData(data);\r\n    return this.request<APIMessage>('PATCH', path, processedData);\r\n  }\r\n\r\n  /**\r\n   * Delete a message\r\n   */\r\n  async deleteMessage(guildId: string, channelId: string, messageId: string): Promise<void> {\r\n    const path = `/bot/guilds/${guildId}/channels/${channelId}/messages/${messageId}`;\r\n    await this.request<void>('DELETE', path);\r\n  }\r\n\r\n  /**\r\n   * Validate and normalize emoji format for the API.\r\n   * Accepted formats: :name:, <:name:id>, <a:name:id>\r\n   * Unicode emoji characters (👍) are NOT supported.\r\n   */\r\n  private validateEmoji(emoji: string): string {\r\n    const trimmed = emoji.trim();\r\n    // Custom emoji: <:name:id> or <a:name:id>\r\n    if (/^<a?:\\w+:\\d+>$/.test(trimmed)) return trimmed;\r\n    // Unicode emoji by name: :name:\r\n    if (/^:\\w+:$/.test(trimmed)) return trimmed;\r\n    throw new Error(\r\n      `Geçersiz emoji formatı: \"${emoji}\". Kabul edilen formatlar: :emoji_name:, <:name:id>, <a:name:id>`\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Add a reaction to a message\r\n   * @param emoji - Emoji in :name:, <:name:id>, or <a:name:id> format. Unicode characters (👍) are not supported.\r\n   */\r\n  async addReaction(guildId: string, channelId: string, messageId: string, emoji: string): Promise<void> {\r\n    const validated = this.validateEmoji(emoji);\r\n    const path = `/bot/guilds/${guildId}/channels/${channelId}/messages/${messageId}/reactions/${encodeURIComponent(validated)}/@me`;\r\n    await this.request<void>('PUT', path);\r\n  }\r\n\r\n  /**\r\n   * Remove a reaction from a message\r\n   * @param emoji - Emoji in :name:, <:name:id>, or <a:name:id> format. Unicode characters (👍) are not supported.\r\n   */\r\n  async removeReaction(guildId: string, channelId: string, messageId: string, emoji: string): Promise<void> {\r\n    const validated = this.validateEmoji(emoji);\r\n    const path = `/bot/guilds/${guildId}/channels/${channelId}/messages/${messageId}/reactions/${encodeURIComponent(validated)}/@me`;\r\n    await this.request<void>('DELETE', path);\r\n  }\r\n\r\n\r\n  /**\r\n   * Upload an attachment to a channel\r\n   */\r\n  async uploadAttachment(guildId: string, channelId: string, file: { name: string; data: Buffer; contentType?: string }): Promise<{ id: string; url: string; filename: string }> {\r\n    const FormData = require('form-data');\r\n    const form = new FormData();\r\n    \r\n    // form-data expects the buffer directly with options\r\n    form.append('file', file.data, {\r\n      filename: file.name,\r\n      contentType: file.contentType || 'text/plain'\r\n    });\r\n    \r\n    const url = `${this.baseUrl}/bot/guilds/${guildId}/channels/${channelId}/attachments`;\r\n    \r\n    console.log(`[REST] Uploading attachment: ${file.name} (${file.data.length} bytes)`);\r\n    \r\n    const response = await fetch(url, {\r\n      method: 'POST',\r\n      headers: {\r\n        'Authorization': `Bot ${this.token}`,\r\n        ...form.getHeaders()\r\n      },\r\n      body: form.getBuffer()\r\n    });\r\n    \r\n    if (!response.ok) {\r\n      const error = await response.text();\r\n      throw new Error(`API Error ${response.status}: ${error}`);\r\n    }\r\n    \r\n    return response.json() as Promise<{ id: string; url: string; filename: string }>;\r\n  }\r\n\r\n  /**\r\n   * Create a message with a file attachment\r\n   */\r\n  async createMessageWithFile(guildId: string, channelId: string, data: {\r\n    content?: string;\r\n    file: { name: string; data: Buffer; contentType?: string };\r\n    interactionId?: string;\r\n  }): Promise<APIMessage> {\r\n    const FormData = require('form-data');\r\n    const form = new FormData();\r\n    \r\n    // Add content if provided\r\n    if (data.content) {\r\n      form.append('content', data.content);\r\n    }\r\n    \r\n    // Add interaction_id if provided (for deferred response matching)\r\n    if (data.interactionId) {\r\n      form.append('interaction_id', data.interactionId);\r\n    }\r\n    \r\n    // Add file\r\n    form.append('files', data.file.data, {\r\n      filename: data.file.name,\r\n      contentType: data.file.contentType || 'text/plain'\r\n    });\r\n    \r\n    const url = `${this.baseUrl}/bot/guilds/${guildId}/channels/${channelId}/messages`;\r\n    \r\n    console.log(`[REST] Creating message with file: ${data.file.name} (${data.file.data.length} bytes)${data.interactionId ? ` [interaction: ${data.interactionId}]` : ''}`);\r\n    \r\n    const response = await fetch(url, {\r\n      method: 'POST',\r\n      headers: {\r\n        'Authorization': `Bot ${this.token}`,\r\n        ...form.getHeaders()\r\n      },\r\n      body: form.getBuffer()\r\n    });\r\n    \r\n    if (!response.ok) {\r\n      const error = await response.text();\r\n      throw new Error(`API Error ${response.status}: ${error}`);\r\n    }\r\n    \r\n    return response.json() as Promise<APIMessage>;\r\n  }\r\n\r\n  // ==================== Interactions ====================\r\n\r\n  /**\r\n   * Create an interaction response\r\n   * Automatically processes mentions in content and embeds\r\n   */\r\n  async createInteractionResponse(interactionId: string, token: string, data: {\r\n    type: number;\r\n    data?: any;\r\n  }): Promise<void> {\r\n    console.log(`📤 Interaction response: ${interactionId} -> type ${data.type}`);\r\n    try {\r\n      // Process mentions in response data if present\r\n      let processedData = data;\r\n      if (data.data && (data.data.content || data.data.embeds)) {\r\n        processedData = {\r\n          ...data,\r\n          data: this.prepareMessageData(data.data)\r\n        };\r\n      }\r\n      \r\n      await this.request<void>('POST', `/interactions/${interactionId}/${token}/callback`, processedData);\r\n      console.log(`✅ Interaction response sent`);\r\n    } catch (error) {\r\n      console.error(`❌ Interaction response error:`, error);\r\n      throw error;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Edit the original interaction response\r\n   * If files are provided, creates a new message with files (since webhook edit doesn't support file upload)\r\n   * Automatically processes mentions\r\n   */\r\n  async editInteractionResponse(token: string, data: {\r\n    content?: string;\r\n    embeds?: APIEmbed[];\r\n    components?: any[];\r\n    mentions?: MentionsData;\r\n    files?: Array<{ name: string; data: Buffer; contentType?: string }>;\r\n  }, guildId?: string, channelId?: string, interactionId?: string): Promise<void> {\r\n    const appId = this.getApplicationId();\r\n    \r\n    // Process mentions in content\r\n    const processedData = this.prepareMessageData(data);\r\n    \r\n    // If files are present and we have guild/channel info, create message with file instead\r\n    if (data.files && data.files.length > 0 && guildId && channelId) {\r\n      console.log(`[REST] editInteractionResponse with ${data.files.length} files - using createMessageWithFile`);\r\n      \r\n      // Create message with file\r\n      const file = data.files[0]; // For now, support single file\r\n      await this.createMessageWithFile(guildId, channelId, {\r\n        content: processedData.content,\r\n        file: file,\r\n        interactionId: interactionId\r\n      });\r\n      return;\r\n    }\r\n    \r\n    // If we have guildId, channelId and interactionId, create a new message with interaction_id\r\n    // This is needed because our deferred response doesn't create a message\r\n    if (guildId && channelId && interactionId) {\r\n      console.log(`[REST] editInteractionResponse - creating message with interaction_id: ${interactionId}`);\r\n      \r\n      const payload: any = {\r\n        interaction_id: interactionId\r\n      };\r\n      if (processedData.content !== undefined) payload.content = processedData.content;\r\n      if (processedData.embeds) payload.embeds = processedData.embeds;\r\n      if (processedData.components) payload.components = processedData.components;\r\n      if (processedData.mentions) payload.mentions = processedData.mentions;\r\n      \r\n      await this.request<void>('POST', `/bot/guilds/${guildId}/channels/${channelId}/messages`, payload);\r\n      return;\r\n    }\r\n    \r\n    // Fallback: Regular edit without files (webhook PATCH)\r\n    const payload: any = {};\r\n    if (processedData.content !== undefined) payload.content = processedData.content;\r\n    if (processedData.embeds) payload.embeds = processedData.embeds;\r\n    if (processedData.components) payload.components = processedData.components;\r\n    if (processedData.mentions) payload.mentions = processedData.mentions;\r\n    \r\n    await this.request<void>('PATCH', `/interactions/webhooks/${appId}/${token}/messages/@original`, payload);\r\n  }\r\n\r\n  /**\r\n   * Delete the original interaction response\r\n   */\r\n  async deleteInteractionResponse(token: string): Promise<void> {\r\n    const appId = this.getApplicationId();\r\n    await this.request<void>('DELETE', `/interactions/webhooks/${appId}/${token}/messages/@original`);\r\n  }\r\n\r\n  /**\r\n   * Create a followup message\r\n   * Automatically processes mentions\r\n   */\r\n  async createFollowup(token: string, data: {\r\n    content?: string;\r\n    embeds?: APIEmbed[];\r\n    mentions?: MentionsData;\r\n    flags?: number;\r\n  }): Promise<void> {\r\n    const appId = this.getApplicationId();\r\n    const processedData = this.prepareMessageData(data);\r\n    await this.request<void>('POST', `/interactions/webhooks/${appId}/${token}`, processedData);\r\n  }\r\n\r\n  // ==================== Commands ====================\r\n\r\n  /**\r\n   * Register global application commands\r\n   */\r\n  async registerGlobalCommands(commands: APIApplicationCommand[]): Promise<void> {\r\n    const appId = this.getApplicationId();\r\n    \r\n    for (const command of commands) {\r\n      await this.request<void>('POST', `/applications/${appId}/commands`, command);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Register guild-specific commands\r\n   */\r\n  async registerGuildCommands(guildId: string, commands: APIApplicationCommand[]): Promise<void> {\r\n    const appId = this.getApplicationId();\r\n    \r\n    for (const command of commands) {\r\n      await this.request<void>('POST', `/applications/${appId}/guilds/${guildId}/commands`, command);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Delete a global command\r\n   */\r\n  async deleteGlobalCommand(commandId: string): Promise<void> {\r\n    const appId = this.getApplicationId();\r\n    await this.request<void>('DELETE', `/applications/${appId}/commands/${commandId}`);\r\n  }\r\n\r\n  /**\r\n   * Delete a guild-specific command\r\n   */\r\n  async deleteGuildCommand(guildId: string, commandId: string): Promise<void> {\r\n    const appId = this.getApplicationId();\r\n    await this.request<void>('DELETE', `/applications/${appId}/guilds/${guildId}/commands/${commandId}`);\r\n  }\r\n\r\n  /**\r\n   * List all global commands for this application\r\n   */\r\n  async listGlobalCommands(): Promise<APIApplicationCommand[]> {\r\n    const appId = this.getApplicationId();\r\n    return this.request<APIApplicationCommand[]>('GET', `/applications/${appId}/commands`);\r\n  }\r\n\r\n  /**\r\n   * List all guild-specific commands for this application\r\n   */\r\n  async listGuildCommands(guildId: string): Promise<APIApplicationCommand[]> {\r\n    const appId = this.getApplicationId();\r\n    return this.request<APIApplicationCommand[]>('GET', `/applications/${appId}/guilds/${guildId}/commands`);\r\n  }\r\n\r\n  /**\r\n   * Get a specific global command\r\n   */\r\n  async getGlobalCommand(commandId: string): Promise<APIApplicationCommand> {\r\n    const appId = this.getApplicationId();\r\n    return this.request<APIApplicationCommand>('GET', `/applications/${appId}/commands/${commandId}`);\r\n  }\r\n\r\n  /**\r\n   * Get a specific guild command\r\n   */\r\n  async getGuildCommand(guildId: string, commandId: string): Promise<APIApplicationCommand> {\r\n    const appId = this.getApplicationId();\r\n    return this.request<APIApplicationCommand>('GET', `/applications/${appId}/guilds/${guildId}/commands/${commandId}`);\r\n  }\r\n\r\n  /**\r\n   * Update a global command\r\n   */\r\n  async updateGlobalCommand(commandId: string, data: Partial<APIApplicationCommand>): Promise<APIApplicationCommand> {\r\n    const appId = this.getApplicationId();\r\n    return this.request<APIApplicationCommand>('PATCH', `/applications/${appId}/commands/${commandId}`, data);\r\n  }\r\n\r\n  /**\r\n   * Update a guild-specific command\r\n   */\r\n  async updateGuildCommand(guildId: string, commandId: string, data: Partial<APIApplicationCommand>): Promise<APIApplicationCommand> {\r\n    const appId = this.getApplicationId();\r\n    return this.request<APIApplicationCommand>('PATCH', `/applications/${appId}/guilds/${guildId}/commands/${commandId}`, data);\r\n  }\r\n\r\n  // ==================== Helpers ====================\r\n\r\n  private applicationId: string = '';\r\n\r\n  /**\r\n   * Set the application ID\r\n   */\r\n  setApplicationId(id: string): void {\r\n    this.applicationId = id;\r\n  }\r\n\r\n  /**\r\n   * Get the application ID\r\n   */\r\n  private getApplicationId(): string {\r\n    if (!this.applicationId) {\r\n      throw new Error('Application ID not set. Call setApplicationId() first.');\r\n    }\r\n    return this.applicationId;\r\n  }\r\n\r\n  // ==================== Channels ====================\r\n\r\n  /**\r\n   * Create a channel in a guild\r\n   */\r\n  async createChannel(guildId: string, data: {\r\n    name: string;\r\n    type?: number;\r\n    parent_id?: string | null;\r\n    category_id?: string | null;\r\n    permission_overwrites?: Array<{\r\n      id: string;\r\n      type: number;\r\n      allow?: string;\r\n      deny?: string;\r\n    }>;\r\n  }): Promise<{ id: string; name: string }> {\r\n    // Map parent_id to category_id for backend compatibility\r\n    const requestData: any = {\r\n      name: data.name,\r\n      type: data.type ?? 0, // Default to text channel\r\n    };\r\n    \r\n    // Backend expects category_id, not parent_id\r\n    if (data.category_id) {\r\n      requestData.category_id = data.category_id;\r\n    } else if (data.parent_id) {\r\n      requestData.category_id = data.parent_id;\r\n    }\r\n    \r\n    // Add permission_overwrites if provided\r\n    if (data.permission_overwrites && data.permission_overwrites.length > 0) {\r\n      requestData.permission_overwrites = data.permission_overwrites;\r\n    }\r\n    \r\n    return this.request('POST', `/bot/guilds/${guildId}/channels`, requestData);\r\n  }\r\n\r\n  /**\r\n   * Delete a channel\r\n   */\r\n  /**\r\n   * Delete a channel\r\n   */\r\n  async deleteChannel(guildId: string, channelId: string): Promise<void> {\r\n    await this.request('DELETE', `/bot/guilds/${guildId}/channels/${channelId}`);\r\n  }\r\n\r\n  /**\r\n   * Delete a category\r\n   */\r\n  async deleteCategory(guildId: string, categoryId: string): Promise<void> {\r\n    await this.request('DELETE', `/bot/guilds/${guildId}/categories/${categoryId}`);\r\n  }\r\n\r\n  /**\r\n   * Edit channel permission overwrites\r\n   */\r\n  async editChannelPermissions(channelId: string, overwriteId: string, data: {\r\n    type: number;\r\n    allow?: string;\r\n    deny?: string;\r\n  }): Promise<void> {\r\n    await this.request('PUT', `/bot/channels/${channelId}/permissions/${overwriteId}`, data);\r\n  }\r\n\r\n  /**\r\n   * Delete channel permission overwrite\r\n   */\r\n  async deleteChannelPermission(channelId: string, overwriteId: string): Promise<void> {\r\n    await this.request('DELETE', `/bot/channels/${channelId}/permissions/${overwriteId}`);\r\n  }\r\n\r\n  /**\r\n   * Get messages from a channel\r\n   */\r\n  async getMessages(guildId: string, channelId: string, options?: {\r\n    limit?: number;\r\n    before?: string;\r\n    after?: string;\r\n  }): Promise<APIMessage[]> {\r\n    const params = new URLSearchParams();\r\n    if (options?.limit) params.append('limit', String(options.limit));\r\n    if (options?.before) params.append('before', options.before);\r\n    if (options?.after) params.append('after', options.after);\r\n    \r\n    const query = params.toString() ? `?${params.toString()}` : '';\r\n    const response = await this.request<{ messages: APIMessage[]; page_info: any }>('GET', `/bot/guilds/${guildId}/channels/${channelId}/messages${query}`);\r\n    return response.messages || [];\r\n  }\r\n\r\n  // ==================== Members ====================\r\n\r\n  /**\r\n   * Get a guild member\r\n   */\r\n  async getMember(guildId: string, userId: string): Promise<any> {\r\n    return this.request('GET', `/bot/guilds/${guildId}/members/${userId}`);\r\n  }\r\n\r\n  /**\r\n   * Timeout a guild member\r\n   */\r\n  async timeoutMember(guildId: string, userId: string, duration: number | null, reason?: string): Promise<void> {\r\n    if (duration === null) {\r\n      // Clear timeout\r\n      await this.request('POST', `/bot/guilds/${guildId}/members/${userId}/timeout/clear`, {\r\n        reason\r\n      });\r\n    } else {\r\n      // Set timeout\r\n      const until = new Date(Date.now() + duration).toISOString();\r\n      await this.request('POST', `/bot/guilds/${guildId}/members/${userId}/timeout`, {\r\n        until,\r\n        reason\r\n      });\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Kick a guild member\r\n   */\r\n  async kickMember(guildId: string, userId: string, reason?: string): Promise<void> {\r\n    const query = reason ? `?reason=${encodeURIComponent(reason)}` : '';\r\n    await this.request('DELETE', `/bot/guilds/${guildId}/members/${userId}${query}`);\r\n  }\r\n\r\n  /**\r\n   * Ban a guild member\r\n   */\r\n  async banMember(guildId: string, userId: string, options?: {\r\n    deleteMessageDays?: number;\r\n    deleteMessageSeconds?: number;\r\n    reason?: string;\r\n  }): Promise<void> {\r\n    await this.request('PUT', `/bot/guilds/${guildId}/bans/${userId}`, {\r\n      delete_message_days: options?.deleteMessageDays,\r\n      delete_message_seconds: options?.deleteMessageSeconds,\r\n      reason: options?.reason\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Unban a user\r\n   */\r\n  async unbanMember(guildId: string, userId: string, reason?: string): Promise<void> {\r\n    const query = reason ? `?reason=${encodeURIComponent(reason)}` : '';\r\n    await this.request('DELETE', `/bot/guilds/${guildId}/bans/${userId}${query}`);\r\n  }\r\n\r\n  /**\r\n   * Edit a guild member\r\n   */\r\n  async editMember(guildId: string, userId: string, data: {\r\n    nick?: string | null;\r\n    roles?: string[];\r\n    mute?: boolean;\r\n    deaf?: boolean;\r\n    channel_id?: string | null;\r\n    communication_disabled_until?: string | null;\r\n    reason?: string;\r\n  }): Promise<any> {\r\n    return this.request('PATCH', `/bot/guilds/${guildId}/members/${userId}`, data);\r\n  }\r\n\r\n  /**\r\n   * Add a role to a member\r\n   */\r\n  async addMemberRole(guildId: string, userId: string, roleId: string, reason?: string): Promise<void> {\r\n    const query = reason ? `?reason=${encodeURIComponent(reason)}` : '';\r\n    await this.request('PUT', `/bot/guilds/${guildId}/members/${userId}/roles/${roleId}${query}`);\r\n  }\r\n\r\n  /**\r\n   * Remove a role from a member\r\n   */\r\n  async removeMemberRole(guildId: string, userId: string, roleId: string, reason?: string): Promise<void> {\r\n    const query = reason ? `?reason=${encodeURIComponent(reason)}` : '';\r\n    await this.request('DELETE', `/bot/guilds/${guildId}/members/${userId}/roles/${roleId}${query}`);\r\n  }\r\n\r\n  /**\r\n   * Bulk assign roles to members\r\n   */\r\n  async bulkAssignRoles(guildId: string, assignments: { user_id: string; role_ids: string[] }[]): Promise<any> {\r\n    return this.request('PUT', `/bot/guilds/${guildId}/members/roles/assign`, { assignments });\r\n  }\r\n\r\n  /**\r\n   * Bulk remove roles from members\r\n   */\r\n  async bulkRemoveRoles(guildId: string, removals: { user_id: string; role_ids: string[] }[]): Promise<any> {\r\n    return this.request('DELETE', `/bot/guilds/${guildId}/members/roles/remove`, { removals });\r\n  }\r\n\r\n  /**\r\n   * Bulk delete messages\r\n   */\r\n  async bulkDeleteMessages(guildId: string, channelId: string, messageIds: string[]): Promise<void> {\r\n    await this.request('POST', `/bot/guilds/${guildId}/channels/${channelId}/messages/bulk-delete`, {\r\n      messages: messageIds\r\n    });\r\n  }\r\n\r\n  // ==================== Guilds ====================\r\n\r\n  /**\r\n   * Get a guild\r\n   */\r\n  async getGuild(guildId: string): Promise<any> {\r\n    return this.request('GET', `/bot/guilds/${guildId}`);\r\n  }\r\n\r\n  /**\r\n   * Get guild channels\r\n   */\r\n  async getGuildChannels(guildId: string): Promise<any[]> {\r\n    return this.request('GET', `/bot/guilds/${guildId}/channels`);\r\n  }\r\n\r\n  /**\r\n   * Get guild roles\r\n   */\r\n  async getRoles(guildId: string): Promise<any[]> {\r\n    return this.request('GET', `/bot/guilds/${guildId}/roles`);\r\n  }\r\n\r\n  /**\r\n   * Create a role\r\n   */\r\n  async createRole(guildId: string, data: {\r\n    name?: string;\r\n    color?: number;\r\n    hoist?: boolean;\r\n    mentionable?: boolean;\r\n    permissions?: string;\r\n  }): Promise<any> {\r\n    return this.request('POST', `/bot/guilds/${guildId}/roles`, data);\r\n  }\r\n\r\n  /**\r\n   * Edit a role\r\n   */\r\n  async editRole(guildId: string, roleId: string, data: {\r\n    name?: string;\r\n    color?: number;\r\n    hoist?: boolean;\r\n    mentionable?: boolean;\r\n    permissions?: string;\r\n  }): Promise<any> {\r\n    return this.request('PATCH', `/bot/guilds/${guildId}/roles/${roleId}`, data);\r\n  }\r\n\r\n  /**\r\n   * Delete a role\r\n   */\r\n  async deleteRole(guildId: string, roleId: string): Promise<void> {\r\n    await this.request('DELETE', `/bot/guilds/${guildId}/roles/${roleId}`);\r\n  }\r\n\r\n  /**\r\n   * Get guild emojis\r\n   */\r\n  async getEmojis(guildId: string): Promise<any[]> {\r\n    return this.request('GET', `/bot/guilds/${guildId}/emojis`);\r\n  }\r\n\r\n  /**\r\n   * Get guild bans\r\n   */\r\n  async getBans(guildId: string): Promise<any[]> {\r\n    return this.request('GET', `/bot/guilds/${guildId}/bans`);\r\n  }\r\n\r\n  /**\r\n   * Get a specific ban\r\n   */\r\n  async getBan(guildId: string, userId: string): Promise<any> {\r\n    return this.request('GET', `/bot/guilds/${guildId}/bans/${userId}`);\r\n  }\r\n\r\n  /**\r\n   * Get guild invites\r\n   */\r\n  async getGuildInvites(guildId: string): Promise<any[]> {\r\n    return this.request('GET', `/bot/guilds/${guildId}/invites`);\r\n  }\r\n\r\n  // ==================== Pins ====================\r\n\r\n  /**\r\n   * Pin a message\r\n   */\r\n  async pinMessage(guildId: string, channelId: string, messageId: string): Promise<void> {\r\n    await this.request('PUT', `/bot/guilds/${guildId}/channels/${channelId}/pins/${messageId}`);\r\n  }\r\n\r\n  /**\r\n   * Unpin a message\r\n   */\r\n  async unpinMessage(guildId: string, channelId: string, messageId: string): Promise<void> {\r\n    await this.request('DELETE', `/bot/guilds/${guildId}/channels/${channelId}/pins/${messageId}`);\r\n  }\r\n\r\n  /**\r\n   * Get pinned messages\r\n   */\r\n  async getPinnedMessages(guildId: string, channelId: string): Promise<APIMessage[]> {\r\n    return this.request('GET', `/bot/guilds/${guildId}/channels/${channelId}/pins`);\r\n  }\r\n\r\n  // ==================== Users ====================\r\n\r\n  /**\r\n   * Get a user\r\n   */\r\n  async getUser(userId: string): Promise<any> {\r\n    return this.request('GET', `/bot/users/${userId}`);\r\n  }\r\n\r\n  /**\r\n   * Get current bot user\r\n   */\r\n  async getCurrentUser(): Promise<any> {\r\n    return this.request('GET', `/bot/users/@me`);\r\n  }\r\n\r\n  // ==================== Invites ====================\r\n\r\n  /**\r\n   * Create an invite\r\n   */\r\n  async createInvite(guildId: string, channelId: string, data?: {\r\n    max_age?: number;\r\n    max_uses?: number;\r\n    temporary?: boolean;\r\n    unique?: boolean;\r\n  }): Promise<any> {\r\n    return this.request('POST', `/bot/guilds/${guildId}/channels/${channelId}/invites`, data || {});\r\n  }\r\n\r\n  /**\r\n   * Delete an invite\r\n   */\r\n  async deleteInvite(inviteCode: string): Promise<void> {\r\n    await this.request('DELETE', `/bot/invites/${inviteCode}`);\r\n  }\r\n\r\n  /**\r\n   * Get an invite\r\n   */\r\n  async getInvite(inviteCode: string): Promise<any> {\r\n    return this.request('GET', `/bot/invites/${inviteCode}`);\r\n  }\r\n\r\n  // ==================== Webhooks ====================\r\n\r\n  /**\r\n   * Get channel webhooks\r\n   */\r\n  async getChannelWebhooks(guildId: string, channelId: string): Promise<any[]> {\r\n    return this.request('GET', `/bot/guilds/${guildId}/channels/${channelId}/webhooks`);\r\n  }\r\n\r\n  /**\r\n   * Get guild webhooks\r\n   */\r\n  async getGuildWebhooks(guildId: string): Promise<any[]> {\r\n    return this.request('GET', `/bot/guilds/${guildId}/webhooks`);\r\n  }\r\n\r\n  /**\r\n   * Create a webhook\r\n   */\r\n  async createWebhook(guildId: string, channelId: string, data: {\r\n    name: string;\r\n    avatar?: string;\r\n  }): Promise<any> {\r\n    return this.request('POST', `/bot/guilds/${guildId}/channels/${channelId}/webhooks`, data);\r\n  }\r\n\r\n  /**\r\n   * Update a webhook\r\n   */\r\n  async updateWebhook(guildId: string, webhookId: string, data: {\r\n    name?: string;\r\n    avatar_url?: string;\r\n  }): Promise<any> {\r\n    return this.request('PATCH', `/bot/guilds/${guildId}/webhooks/${webhookId}`, data);\r\n  }\r\n\r\n  /**\r\n   * Delete a webhook\r\n   */\r\n  async deleteWebhook(guildId: string, webhookId: string): Promise<void> {\r\n    return this.request('DELETE', `/bot/guilds/${guildId}/webhooks/${webhookId}`);\r\n  }\r\n}\r\n"]}
|