mirror of
https://github.com/getcompanion-ai/co-mono.git
synced 2026-04-21 20:04:55 +00:00
mom: Slack bot with abort support, streaming console output, removed sandbox
This commit is contained in:
parent
a7423b954e
commit
aa9e058249
22 changed files with 2741 additions and 58 deletions
259
packages/mom/src/slack.ts
Normal file
259
packages/mom/src/slack.ts
Normal file
|
|
@ -0,0 +1,259 @@
|
|||
import { SocketModeClient } from "@slack/socket-mode";
|
||||
import { WebClient } from "@slack/web-api";
|
||||
import { readFileSync } from "fs";
|
||||
import { basename } from "path";
|
||||
import { type Attachment, ChannelStore } from "./store.js";
|
||||
|
||||
export interface SlackMessage {
|
||||
text: string; // message content (mentions stripped)
|
||||
rawText: string; // original text with mentions
|
||||
user: string; // user ID
|
||||
channel: string; // channel ID
|
||||
ts: string; // timestamp (for threading)
|
||||
attachments: Attachment[]; // file attachments
|
||||
}
|
||||
|
||||
export interface SlackContext {
|
||||
message: SlackMessage;
|
||||
store: ChannelStore;
|
||||
/** Send a new message */
|
||||
respond(text: string): Promise<void>;
|
||||
/** Show/hide typing indicator. If text is provided to respond() after setTyping(true), it updates the typing message instead of posting new. */
|
||||
setTyping(isTyping: boolean): Promise<void>;
|
||||
/** Upload a file to the channel */
|
||||
uploadFile(filePath: string, title?: string): Promise<void>;
|
||||
}
|
||||
|
||||
export interface MomHandler {
|
||||
onChannelMention(ctx: SlackContext): Promise<void>;
|
||||
onDirectMessage(ctx: SlackContext): Promise<void>;
|
||||
}
|
||||
|
||||
export interface MomBotConfig {
|
||||
appToken: string;
|
||||
botToken: string;
|
||||
workingDir: string; // directory for channel data and attachments
|
||||
}
|
||||
|
||||
export class MomBot {
|
||||
private socketClient: SocketModeClient;
|
||||
private webClient: WebClient;
|
||||
private handler: MomHandler;
|
||||
private botUserId: string | null = null;
|
||||
public readonly store: ChannelStore;
|
||||
private userCache: Map<string, { userName: string; displayName: string }> = new Map();
|
||||
|
||||
constructor(handler: MomHandler, config: MomBotConfig) {
|
||||
this.handler = handler;
|
||||
this.socketClient = new SocketModeClient({ appToken: config.appToken });
|
||||
this.webClient = new WebClient(config.botToken);
|
||||
this.store = new ChannelStore({
|
||||
workingDir: config.workingDir,
|
||||
botToken: config.botToken,
|
||||
});
|
||||
|
||||
this.setupEventHandlers();
|
||||
}
|
||||
|
||||
private async getUserInfo(userId: string): Promise<{ userName: string; displayName: string }> {
|
||||
if (this.userCache.has(userId)) {
|
||||
return this.userCache.get(userId)!;
|
||||
}
|
||||
|
||||
try {
|
||||
const result = await this.webClient.users.info({ user: userId });
|
||||
const user = result.user as { name?: string; real_name?: string };
|
||||
const info = {
|
||||
userName: user?.name || userId,
|
||||
displayName: user?.real_name || user?.name || userId,
|
||||
};
|
||||
this.userCache.set(userId, info);
|
||||
return info;
|
||||
} catch {
|
||||
return { userName: userId, displayName: userId };
|
||||
}
|
||||
}
|
||||
|
||||
private setupEventHandlers(): void {
|
||||
// Handle @mentions in channels
|
||||
this.socketClient.on("app_mention", async ({ event, ack }) => {
|
||||
await ack();
|
||||
|
||||
const slackEvent = event as {
|
||||
text: string;
|
||||
channel: string;
|
||||
user: string;
|
||||
ts: string;
|
||||
files?: Array<{ name: string; url_private_download?: string; url_private?: string }>;
|
||||
};
|
||||
|
||||
// Log the mention (message event may not fire for app_mention)
|
||||
await this.logMessage(slackEvent);
|
||||
|
||||
const ctx = this.createContext(slackEvent);
|
||||
await this.handler.onChannelMention(ctx);
|
||||
});
|
||||
|
||||
// Handle all messages (for logging) and DMs (for triggering handler)
|
||||
this.socketClient.on("message", async ({ event, ack }) => {
|
||||
await ack();
|
||||
|
||||
const slackEvent = event as {
|
||||
text?: string;
|
||||
channel: string;
|
||||
user?: string;
|
||||
ts: string;
|
||||
channel_type?: string;
|
||||
subtype?: string;
|
||||
bot_id?: string;
|
||||
files?: Array<{ name: string; url_private_download?: string; url_private?: string }>;
|
||||
};
|
||||
|
||||
// Ignore bot messages
|
||||
if (slackEvent.bot_id) return;
|
||||
// Ignore message edits, etc. (but allow file_share)
|
||||
if (slackEvent.subtype !== undefined && slackEvent.subtype !== "file_share") return;
|
||||
// Ignore if no user
|
||||
if (!slackEvent.user) return;
|
||||
// Ignore messages from the bot itself
|
||||
if (slackEvent.user === this.botUserId) return;
|
||||
// Ignore if no text AND no files
|
||||
if (!slackEvent.text && (!slackEvent.files || slackEvent.files.length === 0)) return;
|
||||
|
||||
// Log ALL messages (channel and DM)
|
||||
await this.logMessage({
|
||||
text: slackEvent.text || "",
|
||||
channel: slackEvent.channel,
|
||||
user: slackEvent.user,
|
||||
ts: slackEvent.ts,
|
||||
files: slackEvent.files,
|
||||
});
|
||||
|
||||
// Only trigger handler for DMs (channel mentions are handled by app_mention event)
|
||||
if (slackEvent.channel_type === "im") {
|
||||
const ctx = this.createContext({
|
||||
text: slackEvent.text || "",
|
||||
channel: slackEvent.channel,
|
||||
user: slackEvent.user,
|
||||
ts: slackEvent.ts,
|
||||
files: slackEvent.files,
|
||||
});
|
||||
await this.handler.onDirectMessage(ctx);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private async logMessage(event: {
|
||||
text: string;
|
||||
channel: string;
|
||||
user: string;
|
||||
ts: string;
|
||||
files?: Array<{ name: string; url_private_download?: string; url_private?: string }>;
|
||||
}): Promise<void> {
|
||||
const attachments = event.files ? this.store.processAttachments(event.channel, event.files, event.ts) : [];
|
||||
const { userName, displayName } = await this.getUserInfo(event.user);
|
||||
|
||||
await this.store.logMessage(event.channel, {
|
||||
ts: event.ts,
|
||||
user: event.user,
|
||||
userName,
|
||||
displayName,
|
||||
text: event.text,
|
||||
attachments,
|
||||
isBot: false,
|
||||
});
|
||||
}
|
||||
|
||||
private createContext(event: {
|
||||
text: string;
|
||||
channel: string;
|
||||
user: string;
|
||||
ts: string;
|
||||
files?: Array<{ name: string; url_private_download?: string; url_private?: string }>;
|
||||
}): SlackContext {
|
||||
const rawText = event.text;
|
||||
const text = rawText.replace(/<@[A-Z0-9]+>/gi, "").trim();
|
||||
|
||||
// Process attachments (for context, already logged by message handler)
|
||||
const attachments = event.files ? this.store.processAttachments(event.channel, event.files, event.ts) : [];
|
||||
|
||||
let typingMessageTs: string | null = null;
|
||||
|
||||
return {
|
||||
message: {
|
||||
text,
|
||||
rawText,
|
||||
user: event.user,
|
||||
channel: event.channel,
|
||||
ts: event.ts,
|
||||
attachments,
|
||||
},
|
||||
store: this.store,
|
||||
respond: async (responseText: string) => {
|
||||
let responseTs: string;
|
||||
|
||||
if (typingMessageTs) {
|
||||
// Update the typing message with the response
|
||||
await this.webClient.chat.update({
|
||||
channel: event.channel,
|
||||
ts: typingMessageTs,
|
||||
text: responseText,
|
||||
});
|
||||
responseTs = typingMessageTs;
|
||||
typingMessageTs = null;
|
||||
} else {
|
||||
// Post a new message
|
||||
const result = await this.webClient.chat.postMessage({
|
||||
channel: event.channel,
|
||||
text: responseText,
|
||||
});
|
||||
responseTs = result.ts as string;
|
||||
}
|
||||
|
||||
// Log the bot response
|
||||
await this.store.logBotResponse(event.channel, responseText, responseTs);
|
||||
},
|
||||
setTyping: async (isTyping: boolean) => {
|
||||
if (isTyping && !typingMessageTs) {
|
||||
// Post a "thinking" message (italic)
|
||||
const result = await this.webClient.chat.postMessage({
|
||||
channel: event.channel,
|
||||
text: "_Thinking..._",
|
||||
});
|
||||
typingMessageTs = result.ts as string;
|
||||
} else if (!isTyping && typingMessageTs) {
|
||||
// Clear typing state (message will be updated by respond())
|
||||
// If respond() wasn't called, delete the typing message
|
||||
await this.webClient.chat.delete({
|
||||
channel: event.channel,
|
||||
ts: typingMessageTs,
|
||||
});
|
||||
typingMessageTs = null;
|
||||
}
|
||||
},
|
||||
uploadFile: async (filePath: string, title?: string) => {
|
||||
const fileName = title || basename(filePath);
|
||||
const fileContent = readFileSync(filePath);
|
||||
|
||||
await this.webClient.files.uploadV2({
|
||||
channel_id: event.channel,
|
||||
file: fileContent,
|
||||
filename: fileName,
|
||||
title: fileName,
|
||||
});
|
||||
},
|
||||
};
|
||||
}
|
||||
|
||||
async start(): Promise<void> {
|
||||
const auth = await this.webClient.auth.test();
|
||||
this.botUserId = auth.user_id as string;
|
||||
await this.socketClient.start();
|
||||
console.log("⚡️ Mom bot connected and listening!");
|
||||
}
|
||||
|
||||
async stop(): Promise<void> {
|
||||
await this.socketClient.disconnect();
|
||||
console.log("Mom bot disconnected.");
|
||||
}
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue