@cremini/skillpack 1.0.9 → 1.1.1

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.
Files changed (40) hide show
  1. package/README.md +4 -4
  2. package/dist/cli.js +4 -0
  3. package/package.json +4 -3
  4. package/runtime/README.md +11 -1
  5. package/runtime/server/dist/adapters/markdown.js +74 -0
  6. package/runtime/server/dist/adapters/markdown.js.map +1 -0
  7. package/runtime/server/dist/adapters/slack.js +369 -0
  8. package/runtime/server/dist/adapters/slack.js.map +1 -0
  9. package/runtime/server/dist/adapters/telegram.js +199 -0
  10. package/runtime/server/dist/adapters/telegram.js.map +1 -0
  11. package/runtime/server/dist/adapters/types.js +2 -0
  12. package/runtime/server/dist/adapters/types.js.map +1 -0
  13. package/runtime/server/dist/adapters/web.js +201 -0
  14. package/runtime/server/dist/adapters/web.js.map +1 -0
  15. package/runtime/server/dist/agent.js +223 -0
  16. package/runtime/server/dist/agent.js.map +1 -0
  17. package/runtime/server/dist/config.js +79 -0
  18. package/runtime/server/dist/config.js.map +1 -0
  19. package/runtime/server/dist/index.js +146 -0
  20. package/runtime/server/dist/index.js.map +1 -0
  21. package/runtime/server/dist/lifecycle.js +85 -0
  22. package/runtime/server/dist/lifecycle.js.map +1 -0
  23. package/runtime/server/dist/memory.js +195 -0
  24. package/runtime/server/dist/memory.js.map +1 -0
  25. package/runtime/server/package-lock.json +4028 -244
  26. package/runtime/server/package.json +13 -3
  27. package/runtime/start.bat +40 -4
  28. package/runtime/start.sh +30 -2
  29. package/runtime/web/index.html +145 -18
  30. package/runtime/web/js/api-key-dialog.js +153 -0
  31. package/runtime/web/js/api.js +25 -0
  32. package/runtime/web/js/chat-apps-dialog.js +194 -0
  33. package/runtime/web/{app.js → js/chat.js} +112 -193
  34. package/runtime/web/js/config.js +16 -0
  35. package/runtime/web/js/main.js +56 -0
  36. package/runtime/web/js/settings.js +205 -0
  37. package/runtime/web/styles.css +311 -10
  38. package/runtime/server/chat-proxy.js +0 -229
  39. package/runtime/server/index.js +0 -63
  40. package/runtime/server/routes.js +0 -104
package/README.md CHANGED
@@ -1,16 +1,16 @@
1
- # SkillPack.sh - Pack AI Skills into Standalone Apps
1
+ # SkillPack.sh - Pack AI Skills into Local Agents
2
2
 
3
3
  Skillpack by Cremini is built on the idea of distributed intelligence, much like cremini mushrooms that grow from a vast, interconnected mycelial network.
4
4
 
5
5
  Go to [skillpack.sh](https://skillpack.sh) to pack skills and try existing skill packs.
6
6
 
7
- One command to orchestrate [Skills](https://skills.sh), tools, mcps into a standalone app users can download and use on their own computer to address a well-defined problem or complete specific tasks!
7
+ One command orchestrates [Skills](https://skills.sh) and tools into a Local Agent that users can download and run it on their own computer to get work done. It can also connect to chat platforms like Slack or Telegram, allowing you to easily send instructions to your local agent team anytime.
8
8
 
9
9
  ```bash
10
10
  npx @cremini/skillpack create
11
11
  ```
12
12
 
13
- If skills, tools, and MCPs are like LEGO pieces, a skill pack is the master piece that assembles them into a complete solution.
13
+ If skills and tools are like LEGO pieces, a skill pack is the master piece that assembles them into a complete solution.
14
14
 
15
15
  Each Skill Pack should organize different skills to address a well-defined problem or complete specific tasks. For example, research a company by gathering information from various sources and create a PowerPoint presentation based on the findings.
16
16
 
@@ -92,7 +92,7 @@ skillpack/
92
92
  start.bat
93
93
  ```
94
94
 
95
- Then open [http://127.0.0.1:26313](http://127.0.0.1:26313), enter your API key, and start working and having fun!!!
95
+ Running start.sh will open [http://127.0.0.1:26313](http://127.0.0.1:26313) in your browser. Just enter your API key to get started and enjoy!
96
96
 
97
97
  ## Development
98
98
 
package/dist/cli.js CHANGED
@@ -342,6 +342,10 @@ function collectRuntimeTemplateEntries(runtimeDir) {
342
342
  if (dirEntry.name === "node_modules") {
343
343
  continue;
344
344
  }
345
+ const currentRelative = relativeDir ? path3.posix.join(relativeDir, dirEntry.name) : dirEntry.name;
346
+ if (currentRelative === "server/src" || currentRelative === "server/tsconfig.json") {
347
+ continue;
348
+ }
345
349
  const absolutePath = path3.join(currentDir, dirEntry.name);
346
350
  const relativePath = relativeDir ? path3.posix.join(relativeDir, dirEntry.name) : dirEntry.name;
347
351
  const stats = fs3.statSync(absolutePath);
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@cremini/skillpack",
3
- "version": "1.0.9",
4
- "description": "Turn Skills into a Standalone App with UI",
3
+ "version": "1.1.1",
4
+ "description": "Pack AI Skills into Local Agents",
5
5
  "type": "module",
6
6
  "repository": {
7
7
  "type": "git",
@@ -24,7 +24,8 @@
24
24
  "node": ">=20"
25
25
  },
26
26
  "scripts": {
27
- "build": "tsup",
27
+ "build": "npm run build:runtime && tsup",
28
+ "build:runtime": "cd runtime/server && npx tsc",
28
29
  "dev": "tsup --watch",
29
30
  "check": "tsc --noEmit",
30
31
  "format": "prettier --write .",
package/runtime/README.md CHANGED
@@ -17,10 +17,20 @@ Double-click `start.bat` or run:
17
17
  start.bat
18
18
  ```
19
19
 
20
- After the server starts, your browser opens [http://127.0.0.1:26313](http://127.0.0.1:26313) automatically.
20
+ Both launchers install server dependencies locally if needed, then start the app in the foreground. After the server starts, your browser opens [http://127.0.0.1:26313](http://127.0.0.1:26313) automatically.
21
21
 
22
22
  By default, the server only listens on `127.0.0.1` so the API key you enter stays on the local machine and is not exposed to your LAN.
23
23
 
24
+ ## Process Management
25
+
26
+ `start.sh` runs the server in a wrapper loop that monitors the exit code:
27
+
28
+ - `/restart` or the web restart button exits with code `75`; the wrapper relaunches the process
29
+ - `/shutdown` exits with code `64`; the wrapper treats it as a clean stop and exits
30
+ - Any other exit code (e.g. crash) triggers an automatic restart after 2 seconds
31
+
32
+ All server output (logs and errors) appears directly in the terminal.
33
+
24
34
  ## First Use
25
35
 
26
36
  1. Enter your OpenAI or Anthropic API key in the left sidebar
@@ -0,0 +1,74 @@
1
+ const MARKDOWN_SOURCE_BLOCK_RE = /```(?:md|markdown)\s*\n([\s\S]*?)```/gi;
2
+ const FENCED_CODE_BLOCK_RE = /```[^\n]*\n[\s\S]*?```/g;
3
+ const INLINE_CODE_RE = /`([^`\n]+)`/g;
4
+ const LINK_RE = /\[([^\]]+)\]\((https?:\/\/[^\s)]+)\)/g;
5
+ const PLACEHOLDER_START = "\uE000";
6
+ const PLACEHOLDER_END = "\uE001";
7
+ function unwrapMarkdownSourceBlocks(text) {
8
+ return text.replace(MARKDOWN_SOURCE_BLOCK_RE, (_, content) => content.trim());
9
+ }
10
+ function escapeHtml(text) {
11
+ return text
12
+ .replace(/&/g, "&")
13
+ .replace(/</g, "&lt;")
14
+ .replace(/>/g, "&gt;");
15
+ }
16
+ function escapeHtmlAttribute(text) {
17
+ return escapeHtml(text).replace(/"/g, "&quot;");
18
+ }
19
+ function protect(text, pattern, placeholders, render) {
20
+ const nextText = text.replace(pattern, (...args) => {
21
+ const match = args[0];
22
+ const groups = args.slice(1, -2);
23
+ const token = `${PLACEHOLDER_START}${placeholders.length}${PLACEHOLDER_END}`;
24
+ placeholders.push(render(match, ...groups));
25
+ return token;
26
+ });
27
+ return nextText;
28
+ }
29
+ function restore(text, placeholders) {
30
+ return text.replace(new RegExp(`${PLACEHOLDER_START}(\\d+)${PLACEHOLDER_END}`, "g"), (_, index) => placeholders[Number(index)] ?? "");
31
+ }
32
+ function formatSlackInline(text) {
33
+ let formatted = text;
34
+ formatted = formatted.replace(LINK_RE, (_, label, url) => {
35
+ return `<${url}|${label}>`;
36
+ });
37
+ formatted = formatted.replace(/^(#{1,6})[ \t]+(.+)$/gm, (_, __, content) => `*${content.trim()}*`);
38
+ formatted = formatted.replace(/\*\*([^*\n]+)\*\*/g, "*$1*");
39
+ formatted = formatted.replace(/__([^_\n]+)__/g, "*$1*");
40
+ return formatted;
41
+ }
42
+ function formatTelegramInline(text) {
43
+ let formatted = escapeHtml(text);
44
+ formatted = formatted.replace(LINK_RE, (_, label, url) => {
45
+ return `<a href="${escapeHtmlAttribute(url)}">${escapeHtml(label)}</a>`;
46
+ });
47
+ formatted = formatted.replace(/^(#{1,6})[ \t]+(.+)$/gm, (_, __, content) => `<b>${content.trim()}</b>`);
48
+ formatted = formatted.replace(/\*\*([^*\n]+)\*\*/g, "<b>$1</b>");
49
+ formatted = formatted.replace(/__([^_\n]+)__/g, "<b>$1</b>");
50
+ formatted = formatted.replace(/(^|[^\w<])\*([^*\n]+)\*(?=[^\w>]|$)/g, "$1<i>$2</i>");
51
+ formatted = formatted.replace(/(^|[^\w<])_([^_\n]+)_(?=[^\w>]|$)/g, "$1<i>$2</i>");
52
+ formatted = formatted.replace(/^(?:-|\*) /gm, "• ");
53
+ return formatted;
54
+ }
55
+ export function formatSlackMessage(text) {
56
+ const unwrapped = unwrapMarkdownSourceBlocks(text);
57
+ const placeholders = [];
58
+ const withFenced = protect(unwrapped, FENCED_CODE_BLOCK_RE, placeholders, (block) => block);
59
+ const withInline = protect(withFenced, INLINE_CODE_RE, placeholders, (_match, code) => `\`${code}\``);
60
+ return restore(formatSlackInline(withInline), placeholders);
61
+ }
62
+ export function formatTelegramMessage(text) {
63
+ const unwrapped = unwrapMarkdownSourceBlocks(text);
64
+ const placeholders = [];
65
+ const withFenced = protect(unwrapped, FENCED_CODE_BLOCK_RE, placeholders, (block) => {
66
+ const content = block
67
+ .replace(/^```[^\n]*\n/, "")
68
+ .replace(/\n```$/, "");
69
+ return `<pre><code>${escapeHtml(content)}</code></pre>`;
70
+ });
71
+ const withInline = protect(withFenced, INLINE_CODE_RE, placeholders, (_match, code) => `<code>${escapeHtml(code)}</code>`);
72
+ return restore(formatTelegramInline(withInline), placeholders);
73
+ }
74
+ //# sourceMappingURL=markdown.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"markdown.js","sourceRoot":"","sources":["../../src/adapters/markdown.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,369 @@
1
+ import { App, LogLevel } from "@slack/bolt";
2
+ import { formatSlackMessage } from "./markdown.js";
3
+ const INLINE_COMMANDS = {
4
+ "/clear": "clear",
5
+ "/restart": "restart",
6
+ "/shutdown": "shutdown",
7
+ };
8
+ const SLASH_COMMANDS = {
9
+ "/skillpack-clear": "clear",
10
+ "/skillpack-restart": "restart",
11
+ "/skillpack-shutdown": "shutdown",
12
+ };
13
+ const MAX_MESSAGE_LENGTH = 3500;
14
+ const ACK_REACTION = "eyes";
15
+ // ---------------------------------------------------------------------------
16
+ // SlackAdapter
17
+ // ---------------------------------------------------------------------------
18
+ export class SlackAdapter {
19
+ name = "slack";
20
+ app = null;
21
+ agent = null;
22
+ options;
23
+ botUserId = null;
24
+ lastThreadByChannel = new Map();
25
+ constructor(options) {
26
+ this.options = options;
27
+ }
28
+ async start(ctx) {
29
+ this.agent = ctx.agent;
30
+ this.app = new App({
31
+ token: this.options.botToken,
32
+ appToken: this.options.appToken,
33
+ socketMode: true,
34
+ ignoreSelf: true,
35
+ logLevel: LogLevel.INFO,
36
+ });
37
+ const auth = await this.app.client.auth.test({
38
+ token: this.options.botToken,
39
+ });
40
+ this.botUserId =
41
+ typeof auth.user_id === "string" ? auth.user_id : null;
42
+ this.registerListeners(this.app);
43
+ await this.app.start();
44
+ const identity = this.botUserId ? `<@${this.botUserId}>` : "Slack bot";
45
+ console.log(`[SlackAdapter] Started as ${identity}`);
46
+ }
47
+ async stop() {
48
+ if (this.app) {
49
+ await this.app.stop();
50
+ this.app = null;
51
+ }
52
+ console.log("[SlackAdapter] Stopped");
53
+ }
54
+ // -------------------------------------------------------------------------
55
+ // Listener registration
56
+ // -------------------------------------------------------------------------
57
+ registerListeners(app) {
58
+ app.event("message", async (args) => {
59
+ try {
60
+ await this.handleDirectMessage(args);
61
+ }
62
+ catch (err) {
63
+ console.error("[Slack] Error handling DM:", err);
64
+ }
65
+ });
66
+ app.event("app_mention", async (args) => {
67
+ try {
68
+ await this.handleMention(args);
69
+ }
70
+ catch (err) {
71
+ console.error("[Slack] Error handling mention:", err);
72
+ }
73
+ });
74
+ for (const commandName of Object.keys(SLASH_COMMANDS)) {
75
+ app.command(commandName, async (args) => {
76
+ try {
77
+ await this.handleSlashCommand(args);
78
+ }
79
+ catch (err) {
80
+ console.error(`[Slack] Error handling ${commandName}:`, err);
81
+ await this.safeAck(args.ack, `❌ Error: ${this.getErrorMessage(err)}`);
82
+ }
83
+ });
84
+ }
85
+ }
86
+ // -------------------------------------------------------------------------
87
+ // Event handlers
88
+ // -------------------------------------------------------------------------
89
+ async handleDirectMessage({ event, body, context, client, }) {
90
+ if (!this.agent || !this.isSupportedDmEvent(event)) {
91
+ return;
92
+ }
93
+ const text = (event.text || "").trim();
94
+ if (!text)
95
+ return;
96
+ const teamId = this.getTeamId(body, context);
97
+ const channelId = `slack-dm-${teamId}-${event.channel}`;
98
+ const route = { channel: event.channel };
99
+ await this.tryAckReaction(client, event);
100
+ if (await this.tryHandleInlineCommand(text, channelId, client, route)) {
101
+ return;
102
+ }
103
+ await this.runAgent(channelId, text, client, route);
104
+ }
105
+ async handleMention({ event, body, context, client, }) {
106
+ if (!this.agent || !this.isSupportedMentionEvent(event)) {
107
+ return;
108
+ }
109
+ const teamId = this.getTeamId(body, context);
110
+ const threadTs = event.thread_ts || event.ts;
111
+ const channelId = `slack-thread-${teamId}-${event.channel}-${threadTs}`;
112
+ const route = {
113
+ channel: event.channel,
114
+ threadTs,
115
+ };
116
+ this.lastThreadByChannel.set(this.getChannelKey(teamId, event.channel), threadTs);
117
+ const text = this.stripBotMention(event.text || "").trim();
118
+ if (!text) {
119
+ await this.sendSafe(client, route, "Mention me with a message, or use `/clear` to reset this thread.");
120
+ return;
121
+ }
122
+ await this.tryAckReaction(client, event);
123
+ if (await this.tryHandleInlineCommand(text, channelId, client, route)) {
124
+ return;
125
+ }
126
+ await this.runAgent(channelId, text, client, route);
127
+ }
128
+ async handleSlashCommand({ command, body, context, ack, }) {
129
+ const commandName = command?.command;
130
+ const mapped = commandName ? SLASH_COMMANDS[commandName] : undefined;
131
+ if (!this.agent || !mapped) {
132
+ await this.safeAck(ack, "Unsupported slash command.");
133
+ return;
134
+ }
135
+ const resolved = this.resolveSlashCommandTarget(body || command, context);
136
+ if (!resolved.channelId) {
137
+ await this.safeAck(ack, resolved.message);
138
+ return;
139
+ }
140
+ const result = await this.agent.handleCommand(mapped, resolved.channelId);
141
+ const parts = [result.message || `${commandName} executed.`];
142
+ if (resolved.note) {
143
+ parts.push(resolved.note);
144
+ }
145
+ await this.safeAck(ack, parts.join("\n"));
146
+ }
147
+ // -------------------------------------------------------------------------
148
+ // Agent bridge
149
+ // -------------------------------------------------------------------------
150
+ async runAgent(channelId, text, client, route) {
151
+ if (!this.agent)
152
+ return;
153
+ let finalText = "";
154
+ let hasError = false;
155
+ let errorMessage = "";
156
+ const onEvent = (event) => {
157
+ if (event.type === "text_delta") {
158
+ finalText += event.delta;
159
+ }
160
+ };
161
+ try {
162
+ const result = await this.agent.handleMessage(channelId, text, onEvent);
163
+ if (result.errorMessage) {
164
+ hasError = true;
165
+ errorMessage = result.errorMessage;
166
+ }
167
+ }
168
+ catch (err) {
169
+ hasError = true;
170
+ errorMessage = this.getErrorMessage(err);
171
+ }
172
+ if (hasError) {
173
+ await this.sendSafe(client, route, `❌ Error: ${errorMessage}`);
174
+ return;
175
+ }
176
+ if (!finalText.trim()) {
177
+ await this.sendSafe(client, route, "(No response generated)");
178
+ return;
179
+ }
180
+ await this.sendLongMessage(client, route, finalText);
181
+ }
182
+ // -------------------------------------------------------------------------
183
+ // Helpers
184
+ // -------------------------------------------------------------------------
185
+ async tryHandleInlineCommand(text, channelId, client, route) {
186
+ if (!this.agent)
187
+ return false;
188
+ const commandKey = text.split(/\s/)[0].toLowerCase();
189
+ const command = INLINE_COMMANDS[commandKey];
190
+ if (!command)
191
+ return false;
192
+ const result = await this.agent.handleCommand(command, channelId);
193
+ await this.sendSafe(client, route, result.message || `${commandKey} executed.`);
194
+ return true;
195
+ }
196
+ resolveSlashCommandTarget(payload, context) {
197
+ const teamId = this.getTeamId(payload, context);
198
+ const channel = payload?.channel_id;
199
+ if (!channel) {
200
+ return { message: "Missing Slack channel context." };
201
+ }
202
+ if (this.isDmChannelId(channel)) {
203
+ return {
204
+ channelId: `slack-dm-${teamId}-${channel}`,
205
+ message: "",
206
+ };
207
+ }
208
+ const threadTs = this.lastThreadByChannel.get(this.getChannelKey(teamId, channel));
209
+ if (!threadTs) {
210
+ return {
211
+ message: "No active Skillpack thread found in this channel. Mention the bot first, or run the command inside the thread as `@bot /clear`.",
212
+ };
213
+ }
214
+ return {
215
+ channelId: `slack-thread-${teamId}-${channel}-${threadTs}`,
216
+ message: "",
217
+ note: "Applied to the most recent active Skillpack thread in this channel.",
218
+ };
219
+ }
220
+ isSupportedDmEvent(event) {
221
+ if (!event || event.type !== "message")
222
+ return false;
223
+ if (event.channel_type !== "im")
224
+ return false;
225
+ if (event.subtype)
226
+ return false;
227
+ if (event.bot_id)
228
+ return false;
229
+ if (!event.user || typeof event.text !== "string")
230
+ return false;
231
+ return true;
232
+ }
233
+ isSupportedMentionEvent(event) {
234
+ if (!event || event.type !== "app_mention")
235
+ return false;
236
+ if (event.subtype)
237
+ return false;
238
+ if (event.bot_id)
239
+ return false;
240
+ if (!event.user || typeof event.text !== "string")
241
+ return false;
242
+ return true;
243
+ }
244
+ stripBotMention(text) {
245
+ const mention = this.botUserId
246
+ ? new RegExp(`^\\s*<@${this.escapeRegExp(this.botUserId)}>\\s*`)
247
+ : /^\s*<@[^>]+>\s*/;
248
+ return text.replace(mention, "");
249
+ }
250
+ splitMessage(text) {
251
+ if (text.length <= MAX_MESSAGE_LENGTH) {
252
+ return [text];
253
+ }
254
+ const chunks = [];
255
+ let remaining = text;
256
+ while (remaining.length > 0) {
257
+ if (remaining.length <= MAX_MESSAGE_LENGTH) {
258
+ chunks.push(remaining);
259
+ break;
260
+ }
261
+ let splitAt = remaining.lastIndexOf("\n\n", MAX_MESSAGE_LENGTH);
262
+ if (splitAt < MAX_MESSAGE_LENGTH * 0.5) {
263
+ splitAt = remaining.lastIndexOf("\n", MAX_MESSAGE_LENGTH);
264
+ }
265
+ if (splitAt < MAX_MESSAGE_LENGTH * 0.3) {
266
+ splitAt = remaining.lastIndexOf(" ", MAX_MESSAGE_LENGTH);
267
+ }
268
+ if (splitAt < 1) {
269
+ splitAt = MAX_MESSAGE_LENGTH;
270
+ }
271
+ chunks.push(remaining.slice(0, splitAt));
272
+ remaining = remaining.slice(splitAt).trimStart();
273
+ }
274
+ return chunks;
275
+ }
276
+ async sendLongMessage(client, route, text) {
277
+ for (const chunk of this.splitMessage(text)) {
278
+ await this.sendWithRetry(client, route, chunk);
279
+ }
280
+ }
281
+ async sendSafe(client, route, text) {
282
+ try {
283
+ await this.sendWithRetry(client, route, text);
284
+ }
285
+ catch (err) {
286
+ console.error("[Slack] Failed to send message:", err);
287
+ }
288
+ }
289
+ async sendWithRetry(client, route, text, maxRetries = 3) {
290
+ for (let attempt = 0; attempt <= maxRetries; attempt++) {
291
+ try {
292
+ await client.chat.postMessage({
293
+ channel: route.channel,
294
+ text: formatSlackMessage(text),
295
+ mrkdwn: true,
296
+ thread_ts: route.threadTs,
297
+ reply_broadcast: false,
298
+ });
299
+ return;
300
+ }
301
+ catch (err) {
302
+ const retryAfter = this.getRetryAfterSeconds(err);
303
+ if (retryAfter && attempt < maxRetries) {
304
+ console.log(`[Slack] Rate limited, retrying after ${retryAfter}s...`);
305
+ await new Promise((resolve) => setTimeout(resolve, retryAfter * 1000));
306
+ continue;
307
+ }
308
+ throw err;
309
+ }
310
+ }
311
+ }
312
+ async tryAckReaction(client, event) {
313
+ try {
314
+ await client.reactions.add({
315
+ channel: event.channel,
316
+ timestamp: event.ts,
317
+ name: ACK_REACTION,
318
+ });
319
+ }
320
+ catch (err) {
321
+ console.error("[Slack] Failed to add ack reaction:", err);
322
+ }
323
+ }
324
+ async safeAck(ack, message) {
325
+ if (!ack)
326
+ return;
327
+ try {
328
+ await ack(message);
329
+ }
330
+ catch (err) {
331
+ console.error("[Slack] Failed to ack slash command:", err);
332
+ }
333
+ }
334
+ getRetryAfterSeconds(err) {
335
+ const candidates = [
336
+ err?.data?.retryAfter,
337
+ err?.retryAfter,
338
+ err?.headers?.["retry-after"],
339
+ err?.data?.headers?.["retry-after"],
340
+ ];
341
+ for (const value of candidates) {
342
+ const seconds = Number(value);
343
+ if (Number.isFinite(seconds) && seconds > 0) {
344
+ return seconds;
345
+ }
346
+ }
347
+ return null;
348
+ }
349
+ getTeamId(payload, context) {
350
+ return (context?.teamId ||
351
+ payload?.team_id ||
352
+ payload?.team?.id ||
353
+ payload?.authorizations?.[0]?.team_id ||
354
+ "unknown");
355
+ }
356
+ getChannelKey(teamId, channelId) {
357
+ return `${teamId}:${channelId}`;
358
+ }
359
+ isDmChannelId(channelId) {
360
+ return channelId.startsWith("D");
361
+ }
362
+ getErrorMessage(err) {
363
+ return err instanceof Error ? err.message : String(err);
364
+ }
365
+ escapeRegExp(value) {
366
+ return value.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
367
+ }
368
+ }
369
+ //# sourceMappingURL=slack.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"slack.js","sourceRoot":"","sources":["../../src/adapters/slack.ts"],"names":[],"mappings":"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"}