@delt/claude-alarm 0.6.25 → 0.7.0
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/channel/server.js +10 -0
- package/dist/channel/server.js.map +1 -1
- package/dist/cli.js +56 -2
- package/dist/cli.js.map +1 -1
- package/dist/hub/server.js +46 -2
- package/dist/hub/server.js.map +1 -1
- package/dist/index.d.ts +3 -0
- package/dist/index.js +46 -2
- package/dist/index.js.map +1 -1
- package/package.json +1 -1
package/dist/index.js
CHANGED
|
@@ -684,6 +684,9 @@ var HubServer = class {
|
|
|
684
684
|
// All connected dashboard WebSockets
|
|
685
685
|
dashboardSockets = /* @__PURE__ */ new Set();
|
|
686
686
|
telegramBot;
|
|
687
|
+
heartbeatInterval;
|
|
688
|
+
channelAlive = /* @__PURE__ */ new Map();
|
|
689
|
+
// sessionId -> alive flag
|
|
687
690
|
host;
|
|
688
691
|
port;
|
|
689
692
|
token;
|
|
@@ -735,6 +738,7 @@ var HubServer = class {
|
|
|
735
738
|
}
|
|
736
739
|
async start() {
|
|
737
740
|
this.cleanupUploads();
|
|
741
|
+
this.startHeartbeat();
|
|
738
742
|
return new Promise((resolve, reject) => {
|
|
739
743
|
this.httpServer.on("error", reject);
|
|
740
744
|
this.httpServer.listen(this.port, this.host, () => {
|
|
@@ -746,6 +750,7 @@ var HubServer = class {
|
|
|
746
750
|
}
|
|
747
751
|
stop() {
|
|
748
752
|
return new Promise((resolve) => {
|
|
753
|
+
if (this.heartbeatInterval) clearInterval(this.heartbeatInterval);
|
|
749
754
|
if (this.telegramBot) this.telegramBot.stopPolling();
|
|
750
755
|
for (const ws of this.channelSockets.values()) ws.terminate();
|
|
751
756
|
for (const ws of this.dashboardSockets) ws.terminate();
|
|
@@ -770,7 +775,7 @@ var HubServer = class {
|
|
|
770
775
|
if (origin && (origin.includes("127.0.0.1") || origin.includes("localhost"))) {
|
|
771
776
|
res.setHeader("Access-Control-Allow-Origin", origin);
|
|
772
777
|
}
|
|
773
|
-
res.setHeader("Access-Control-Allow-Methods", "GET, POST, OPTIONS");
|
|
778
|
+
res.setHeader("Access-Control-Allow-Methods", "GET, POST, DELETE, OPTIONS");
|
|
774
779
|
res.setHeader("Access-Control-Allow-Headers", "Content-Type, Authorization");
|
|
775
780
|
if (req.method === "OPTIONS") {
|
|
776
781
|
res.writeHead(204);
|
|
@@ -799,6 +804,22 @@ var HubServer = class {
|
|
|
799
804
|
sessions: this.sessions.count(),
|
|
800
805
|
uptime: Date.now() - this.startTime
|
|
801
806
|
});
|
|
807
|
+
} else if (url.pathname.startsWith("/api/sessions/") && req.method === "DELETE") {
|
|
808
|
+
const sessionId = url.pathname.slice("/api/sessions/".length);
|
|
809
|
+
const ws = this.channelSockets.get(sessionId);
|
|
810
|
+
if (ws) {
|
|
811
|
+
ws.terminate();
|
|
812
|
+
}
|
|
813
|
+
const session = this.sessions.unregister(sessionId);
|
|
814
|
+
this.channelSockets.delete(sessionId);
|
|
815
|
+
this.localChannels.delete(sessionId);
|
|
816
|
+
this.channelAlive.delete(sessionId);
|
|
817
|
+
if (session) {
|
|
818
|
+
this.broadcastToDashboards({ type: "session_disconnected", sessionId });
|
|
819
|
+
this.jsonResponse(res, 200, { ok: true });
|
|
820
|
+
} else {
|
|
821
|
+
this.jsonResponse(res, 404, { error: "Session not found" });
|
|
822
|
+
}
|
|
802
823
|
} else if (url.pathname === "/api/send" && req.method === "POST") {
|
|
803
824
|
this.handleApiSend(req, res);
|
|
804
825
|
} else if (url.pathname === "/api/notify" && req.method === "POST") {
|
|
@@ -889,6 +910,14 @@ var HubServer = class {
|
|
|
889
910
|
handleChannelConnection(ws, req) {
|
|
890
911
|
const isLocal = this.isLocalRequest(req);
|
|
891
912
|
logger.info(`Channel server connected (local: ${isLocal})`);
|
|
913
|
+
ws.on("pong", () => {
|
|
914
|
+
for (const [sessionId, sock] of this.channelSockets) {
|
|
915
|
+
if (sock === ws) {
|
|
916
|
+
this.channelAlive.set(sessionId, true);
|
|
917
|
+
break;
|
|
918
|
+
}
|
|
919
|
+
}
|
|
920
|
+
});
|
|
892
921
|
ws.on("message", (data) => {
|
|
893
922
|
try {
|
|
894
923
|
const msg = JSON.parse(data.toString());
|
|
@@ -903,6 +932,7 @@ var HubServer = class {
|
|
|
903
932
|
const session = this.sessions.unregister(sessionId);
|
|
904
933
|
this.channelSockets.delete(sessionId);
|
|
905
934
|
this.localChannels.delete(sessionId);
|
|
935
|
+
this.channelAlive.delete(sessionId);
|
|
906
936
|
logger.info(`Channel disconnected: ${sessionId}`);
|
|
907
937
|
this.broadcastToDashboards({
|
|
908
938
|
type: "session_disconnected",
|
|
@@ -1217,6 +1247,19 @@ var HubServer = class {
|
|
|
1217
1247
|
this.jsonResponse(res, 500, { error: err.message });
|
|
1218
1248
|
}
|
|
1219
1249
|
}
|
|
1250
|
+
startHeartbeat() {
|
|
1251
|
+
this.heartbeatInterval = setInterval(() => {
|
|
1252
|
+
for (const [sessionId, ws] of this.channelSockets) {
|
|
1253
|
+
if (this.channelAlive.get(sessionId) === false) {
|
|
1254
|
+
logger.info(`Heartbeat timeout, terminating session: ${sessionId}`);
|
|
1255
|
+
ws.terminate();
|
|
1256
|
+
continue;
|
|
1257
|
+
}
|
|
1258
|
+
this.channelAlive.set(sessionId, false);
|
|
1259
|
+
ws.ping();
|
|
1260
|
+
}
|
|
1261
|
+
}, 3e4);
|
|
1262
|
+
}
|
|
1220
1263
|
cleanupUploads() {
|
|
1221
1264
|
try {
|
|
1222
1265
|
if (!fs3.existsSync(UPLOADS_DIR)) return;
|
|
@@ -1267,7 +1310,8 @@ var HubServer = class {
|
|
|
1267
1310
|
}
|
|
1268
1311
|
};
|
|
1269
1312
|
if (process.argv[1] && (process.argv[1].endsWith("hub/server.js") || process.argv[1].endsWith("hub/server.ts"))) {
|
|
1270
|
-
const
|
|
1313
|
+
const config = loadConfig();
|
|
1314
|
+
const hub = new HubServer(config);
|
|
1271
1315
|
hub.start().catch((err) => {
|
|
1272
1316
|
logger.error("Failed to start hub:", err);
|
|
1273
1317
|
process.exit(1);
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/hub/server.ts","../src/shared/logger.ts","../src/shared/constants.ts","../src/hub/session-manager.ts","../src/hub/notifier.ts","../src/hub/telegram.ts","../src/shared/config.ts","../src/channel/hub-client.ts"],"sourcesContent":["import http from 'node:http';\r\nimport fs from 'node:fs';\r\nimport path from 'node:path';\r\nimport { fileURLToPath } from 'node:url';\r\nimport { WebSocketServer, WebSocket } from 'ws';\r\nimport { logger } from '../shared/logger.js';\r\nimport { randomUUID } from 'node:crypto';\r\nimport {\r\n DEFAULT_HUB_HOST,\r\n DEFAULT_HUB_PORT,\r\n WS_PATH_CHANNEL,\r\n WS_PATH_DASHBOARD,\r\n UPLOADS_DIR,\r\n} from '../shared/constants.js';\r\nimport { SessionManager } from './session-manager.js';\r\nimport { Notifier } from './notifier.js';\r\nimport { TelegramBot } from './telegram.js';\r\nimport { loadConfig, saveConfig } from '../shared/config.js';\r\nimport type { ChannelMessage, AppConfig, SessionInfo, WebhookConfig, TelegramConfig } from '../shared/types.js';\r\n\r\nconst __dirname = path.dirname(fileURLToPath(import.meta.url));\r\n\r\nexport class HubServer {\r\n private httpServer: http.Server;\r\n private wssChannel: WebSocketServer;\r\n private wssDashboard: WebSocketServer;\r\n private sessions = new SessionManager();\r\n private notifier = new Notifier();\r\n private startTime = Date.now();\r\n\r\n // Map sessionId -> channel WebSocket\r\n private channelSockets = new Map<string, WebSocket>();\r\n // Track which channel connections are local\r\n private localChannels = new Set<string>();\r\n // All connected dashboard WebSockets\r\n private dashboardSockets = new Set<WebSocket>();\r\n\r\n private telegramBot?: TelegramBot;\r\n\r\n private host: string;\r\n private port: number;\r\n private token?: string;\r\n\r\n constructor(config?: Partial<AppConfig>) {\r\n this.host = config?.hub?.host ?? DEFAULT_HUB_HOST;\r\n this.port = config?.hub?.port ?? DEFAULT_HUB_PORT;\r\n this.token = config?.hub?.token;\r\n\r\n if (config?.notifications) {\r\n this.notifier.configure({\r\n desktop: config.notifications.desktop,\r\n });\r\n }\r\n if (config?.webhooks) {\r\n this.notifier.configure({ webhooks: config.webhooks });\r\n }\r\n const displayHost = this.host === '0.0.0.0' ? '127.0.0.1' : this.host;\r\n this.notifier.configure({ dashboardUrl: `http://${displayHost}:${this.port}` });\r\n\r\n // Initialize Telegram bot if configured\r\n const fullConfig = loadConfig();\r\n if (fullConfig.telegram?.enabled && fullConfig.telegram.botToken && fullConfig.telegram.chatId) {\r\n this.initTelegram(fullConfig.telegram);\r\n }\r\n\r\n // HTTP Server\r\n this.httpServer = http.createServer((req, res) => this.handleHttp(req, res));\r\n\r\n // WebSocket for channel servers\r\n this.wssChannel = new WebSocketServer({ noServer: true });\r\n this.wssChannel.on('connection', (ws: WebSocket, req: http.IncomingMessage) => this.handleChannelConnection(ws, req));\r\n\r\n // WebSocket for dashboard\r\n this.wssDashboard = new WebSocketServer({ noServer: true });\r\n this.wssDashboard.on('connection', (ws) => this.handleDashboardConnection(ws));\r\n\r\n // Route WebSocket upgrade requests\r\n this.httpServer.on('upgrade', (req, socket, head) => {\r\n const url = new URL(req.url!, `http://${req.headers.host}`);\r\n const pathname = url.pathname;\r\n\r\n // Token auth for WebSocket connections (skip for local requests)\r\n if (this.token && !this.isLocalRequest(req)) {\r\n const wsToken = url.searchParams.get('token');\r\n if (wsToken !== this.token) {\r\n socket.destroy();\r\n return;\r\n }\r\n }\r\n\r\n if (pathname === WS_PATH_CHANNEL) {\r\n this.wssChannel.handleUpgrade(req, socket, head, (ws) => {\r\n this.wssChannel.emit('connection', ws, req);\r\n });\r\n } else if (pathname === WS_PATH_DASHBOARD) {\r\n this.wssDashboard.handleUpgrade(req, socket, head, (ws) => {\r\n this.wssDashboard.emit('connection', ws, req);\r\n });\r\n } else {\r\n socket.destroy();\r\n }\r\n });\r\n }\r\n\r\n async start(): Promise<void> {\r\n this.cleanupUploads();\r\n return new Promise((resolve, reject) => {\r\n this.httpServer.on('error', reject);\r\n this.httpServer.listen(this.port, this.host, () => {\r\n const displayHost = this.host === '0.0.0.0' ? '127.0.0.1' : this.host;\r\n logger.info(`Hub server listening on http://${displayHost}:${this.port}`);\r\n resolve();\r\n });\r\n });\r\n }\r\n\r\n stop(): Promise<void> {\r\n return new Promise((resolve) => {\r\n // Stop telegram bot\r\n if (this.telegramBot) this.telegramBot.stopPolling();\r\n\r\n // Force-close all WebSocket connections\r\n for (const ws of this.channelSockets.values()) ws.terminate();\r\n for (const ws of this.dashboardSockets) ws.terminate();\r\n this.channelSockets.clear();\r\n this.dashboardSockets.clear();\r\n\r\n this.wssChannel.close();\r\n this.wssDashboard.close();\r\n this.httpServer.close(() => {\r\n logger.info('Hub server stopped');\r\n resolve();\r\n });\r\n\r\n // Force resolve after 3 seconds if server won't close\r\n setTimeout(() => {\r\n logger.warn('Force shutting down');\r\n resolve();\r\n }, 3000);\r\n });\r\n }\r\n\r\n // --- HTTP Handler ---\r\n\r\n private handleHttp(req: http.IncomingMessage, res: http.ServerResponse): void {\r\n const url = new URL(req.url!, `http://${req.headers.host}`);\r\n\r\n // CORS headers - restrict to same origin\r\n const origin = req.headers.origin;\r\n if (origin && (origin.includes('127.0.0.1') || origin.includes('localhost'))) {\r\n res.setHeader('Access-Control-Allow-Origin', origin);\r\n }\r\n res.setHeader('Access-Control-Allow-Methods', 'GET, POST, OPTIONS');\r\n res.setHeader('Access-Control-Allow-Headers', 'Content-Type, Authorization');\r\n\r\n if (req.method === 'OPTIONS') {\r\n res.writeHead(204);\r\n res.end();\r\n return;\r\n }\r\n\r\n // Token auth for API endpoints (skip dashboard HTML serving)\r\n if (url.pathname !== '/' && this.token) {\r\n if (!this.isLocalRequest(req)) {\r\n const authHeader = req.headers['authorization'];\r\n const bearerToken = authHeader?.startsWith('Bearer ') ? authHeader.slice(7) : null;\r\n if (bearerToken !== this.token) {\r\n this.jsonResponse(res, 401, { error: 'Unauthorized' });\r\n return;\r\n }\r\n }\r\n }\r\n\r\n // Route\r\n if (url.pathname === '/' && req.method === 'GET') {\r\n this.serveDashboard(res);\r\n } else if (url.pathname === '/api/sessions' && req.method === 'GET') {\r\n this.jsonResponse(res, 200, { sessions: this.sessions.getAll() });\r\n } else if (url.pathname === '/api/status' && req.method === 'GET') {\r\n this.jsonResponse(res, 200, {\r\n running: true,\r\n pid: process.pid,\r\n port: this.port,\r\n sessions: this.sessions.count(),\r\n uptime: Date.now() - this.startTime,\r\n });\r\n } else if (url.pathname === '/api/send' && req.method === 'POST') {\r\n this.handleApiSend(req, res);\r\n } else if (url.pathname === '/api/notify' && req.method === 'POST') {\r\n this.handleApiNotify(req, res);\r\n } else if (url.pathname === '/api/webhooks' && req.method === 'GET') {\r\n const config = loadConfig();\r\n this.jsonResponse(res, 200, { webhooks: config.webhooks || [] });\r\n } else if (url.pathname === '/api/webhooks' && req.method === 'POST') {\r\n this.handleWebhookSave(req, res);\r\n } else if (url.pathname === '/api/telegram' && req.method === 'GET') {\r\n const cfg = loadConfig();\r\n const tg = cfg.telegram ?? { botToken: '', chatId: '', enabled: false };\r\n // Mask bot token for security\r\n this.jsonResponse(res, 200, {\r\n telegram: { ...tg, botToken: tg.botToken ? `${tg.botToken.slice(0, 8)}...` : '' },\r\n });\r\n } else if (url.pathname === '/api/telegram' && req.method === 'POST') {\r\n this.handleTelegramSave(req, res);\r\n } else if (url.pathname === '/api/telegram/test' && req.method === 'POST') {\r\n this.handleTelegramTest(req, res);\r\n } else if (url.pathname === '/api/telegram/detect' && req.method === 'POST') {\r\n this.handleTelegramDetect(req, res);\r\n } else {\r\n this.jsonResponse(res, 404, { error: 'Not found' });\r\n }\r\n }\r\n\r\n private serveDashboard(res: http.ServerResponse): void {\r\n // Look for dashboard HTML relative to this file (dist) or source\r\n const candidates = [\r\n path.join(__dirname, '..', 'dashboard', 'index.html'), // from dist/hub/\r\n path.join(__dirname, 'dashboard', 'index.html'), // from dist/ (bundled index.js)\r\n path.join(__dirname, '..', '..', 'src', 'dashboard', 'index.html'), // from dist/hub/ -> src/\r\n path.join(__dirname, '..', 'src', 'dashboard', 'index.html'), // from dist/ -> src/\r\n path.join(process.cwd(), 'dist', 'dashboard', 'index.html'), // from cwd\r\n path.join(process.cwd(), 'src', 'dashboard', 'index.html'), // from cwd/src\r\n ];\r\n logger.debug(`Dashboard candidates: ${JSON.stringify(candidates)}`);\r\n\r\n for (const candidate of candidates) {\r\n if (fs.existsSync(candidate)) {\r\n const html = fs.readFileSync(candidate, 'utf-8');\r\n res.writeHead(200, { 'Content-Type': 'text/html; charset=utf-8' });\r\n res.end(html);\r\n return;\r\n }\r\n }\r\n\r\n res.writeHead(200, { 'Content-Type': 'text/html; charset=utf-8' });\r\n res.end('<html><body><h1>claude-alarm</h1><p>Dashboard HTML not found. Reinstall the package.</p></body></html>');\r\n }\r\n\r\n private async handleApiSend(req: http.IncomingMessage, res: http.ServerResponse): Promise<void> {\r\n const body = await this.readBody(req);\r\n if (!body) { this.jsonResponse(res, 400, { error: 'Invalid JSON' }); return; }\r\n\r\n const { sessionId, content } = body as { sessionId?: string; content?: string };\r\n if (!sessionId || !content) {\r\n this.jsonResponse(res, 400, { error: 'sessionId and content are required' });\r\n return;\r\n }\r\n\r\n const ws = this.channelSockets.get(sessionId);\r\n if (!ws || ws.readyState !== WebSocket.OPEN) {\r\n this.jsonResponse(res, 404, { error: 'Session not connected' });\r\n return;\r\n }\r\n\r\n const msg: ChannelMessage = { type: 'message_to_session', sessionId, content };\r\n ws.send(JSON.stringify(msg));\r\n this.jsonResponse(res, 200, { ok: true });\r\n }\r\n\r\n private async handleApiNotify(req: http.IncomingMessage, res: http.ServerResponse): Promise<void> {\r\n const body = await this.readBody(req);\r\n if (!body) { this.jsonResponse(res, 400, { error: 'Invalid JSON' }); return; }\r\n\r\n const { title, message, level } = body as { title?: string; message?: string; level?: string };\r\n if (!title || !message) {\r\n this.jsonResponse(res, 400, { error: 'title and message are required' });\r\n return;\r\n }\r\n\r\n await this.notifier.notify(title, message, (level as any) ?? 'info');\r\n this.jsonResponse(res, 200, { ok: true });\r\n }\r\n\r\n // --- Channel WebSocket ---\r\n\r\n private handleChannelConnection(ws: WebSocket, req: http.IncomingMessage): void {\r\n const isLocal = this.isLocalRequest(req);\r\n logger.info(`Channel server connected (local: ${isLocal})`);\r\n\r\n ws.on('message', (data) => {\r\n try {\r\n const msg = JSON.parse(data.toString()) as ChannelMessage;\r\n this.handleChannelMessage(ws, isLocal, msg);\r\n } catch {\r\n logger.warn('Invalid message from channel');\r\n }\r\n });\r\n\r\n ws.on('close', () => {\r\n // Find and remove the session for this socket\r\n for (const [sessionId, sock] of this.channelSockets) {\r\n if (sock === ws) {\r\n const session = this.sessions.unregister(sessionId);\r\n this.channelSockets.delete(sessionId);\r\n this.localChannels.delete(sessionId);\r\n logger.info(`Channel disconnected: ${sessionId}`);\r\n this.broadcastToDashboards({\r\n type: 'session_disconnected',\r\n sessionId,\r\n });\r\n break;\r\n }\r\n }\r\n });\r\n }\r\n\r\n private handleChannelMessage(ws: WebSocket, isLocal: boolean, msg: ChannelMessage): void {\r\n switch (msg.type) {\r\n case 'register': {\r\n const session = msg.session;\r\n session.isLocal = isLocal;\r\n const isReregister = !!this.sessions.get(session.id);\r\n this.sessions.register(session);\r\n this.channelSockets.set(session.id, ws);\r\n if (isLocal) this.localChannels.add(session.id);\r\n logger.info(`Session registered: ${session.id} (${session.name}, channel: ${session.channelEnabled ?? false})`);\r\n this.broadcastToDashboards({\r\n type: isReregister ? 'session_updated' : 'session_connected',\r\n session,\r\n });\r\n break;\r\n }\r\n\r\n case 'status': {\r\n const updated = this.sessions.updateStatus(msg.sessionId, msg.status);\r\n if (updated) {\r\n this.broadcastToDashboards({ type: 'session_updated', session: updated });\r\n }\r\n break;\r\n }\r\n\r\n case 'notify': {\r\n this.sessions.updateActivity(msg.sessionId);\r\n const notifySession = this.sessions.get(msg.sessionId);\r\n const notifyLabel = this.getSessionLabel(notifySession);\r\n this.notifier.notifyWithSession(msg.sessionId, notifyLabel, `[${notifyLabel}] ${msg.title}`, msg.message, msg.level ?? 'info');\r\n this.broadcastToDashboards({\r\n type: 'notification',\r\n sessionId: msg.sessionId,\r\n title: msg.title,\r\n message: msg.message,\r\n level: msg.level,\r\n timestamp: Date.now(),\r\n });\r\n break;\r\n }\r\n\r\n case 'reply': {\r\n this.sessions.updateActivity(msg.sessionId);\r\n const replySession = this.sessions.get(msg.sessionId);\r\n const replyLabel = this.getSessionLabel(replySession);\r\n this.notifier.notifyWithSession(msg.sessionId, replyLabel, `[${replyLabel}] Reply`, msg.content.slice(0, 3000), 'info');\r\n this.broadcastToDashboards({\r\n type: 'reply_from_session',\r\n sessionId: msg.sessionId,\r\n content: msg.content,\r\n timestamp: Date.now(),\r\n });\r\n break;\r\n }\r\n\r\n case 'permission_request': {\r\n this.sessions.updateActivity(msg.sessionId);\r\n logger.info(`Permission request [${msg.requestId}] from ${msg.sessionId}: ${msg.toolName}`);\r\n this.broadcastToDashboards({\r\n type: 'permission_request',\r\n sessionId: msg.sessionId,\r\n requestId: msg.requestId,\r\n toolName: msg.toolName,\r\n description: msg.description,\r\n inputPreview: msg.inputPreview,\r\n timestamp: msg.timestamp,\r\n });\r\n // Forward to Telegram\r\n if (this.telegramBot) {\r\n const session = this.sessions.get(msg.sessionId);\r\n const label = this.getSessionLabel(session);\r\n this.telegramBot.sendPermissionRequest(msg.sessionId, label, msg.requestId, msg.toolName, msg.description, msg.inputPreview);\r\n }\r\n break;\r\n }\r\n }\r\n }\r\n\r\n // --- Dashboard WebSocket ---\r\n\r\n private handleDashboardConnection(ws: WebSocket): void {\r\n this.dashboardSockets.add(ws);\r\n logger.info(`Dashboard connected (total: ${this.dashboardSockets.size})`);\r\n\r\n // Send current session list\r\n const sessionsMsg: ChannelMessage = {\r\n type: 'sessions_list',\r\n sessions: this.sessions.getAll(),\r\n };\r\n ws.send(JSON.stringify(sessionsMsg));\r\n\r\n ws.on('message', (data) => {\r\n try {\r\n const msg = JSON.parse(data.toString()) as ChannelMessage;\r\n if (msg.type === 'message_to_session') {\r\n const channelWs = this.channelSockets.get(msg.sessionId);\r\n if (channelWs?.readyState === WebSocket.OPEN) {\r\n channelWs.send(JSON.stringify(msg));\r\n }\r\n } else if (msg.type === 'image_upload') {\r\n this.handleImageUpload(msg);\r\n } else if (msg.type === 'permission_response') {\r\n const channelWs = this.channelSockets.get(msg.sessionId);\r\n if (channelWs?.readyState === WebSocket.OPEN) {\r\n channelWs.send(JSON.stringify(msg));\r\n logger.info(`Permission verdict [${msg.requestId}]: ${msg.behavior} -> session ${msg.sessionId}`);\r\n }\r\n }\r\n } catch {\r\n logger.warn('Invalid message from dashboard');\r\n }\r\n });\r\n\r\n ws.on('close', () => {\r\n this.dashboardSockets.delete(ws);\r\n logger.info(`Dashboard disconnected (total: ${this.dashboardSockets.size})`);\r\n });\r\n }\r\n\r\n // --- Helpers ---\r\n\r\n private broadcastToDashboards(msg: ChannelMessage): void {\r\n const payload = JSON.stringify(msg);\r\n for (const ws of this.dashboardSockets) {\r\n if (ws.readyState === WebSocket.OPEN) {\r\n ws.send(payload);\r\n }\r\n }\r\n }\r\n\r\n private handleImageUpload(msg: ChannelMessage & { type: 'image_upload' }): void {\r\n const { sessionId, imageData, mimeType, originalName, content } = msg;\r\n\r\n // Only allow for local sessions\r\n if (!this.localChannels.has(sessionId)) {\r\n logger.warn(`Image upload rejected: session ${sessionId} is not local`);\r\n return;\r\n }\r\n\r\n const channelWs = this.channelSockets.get(sessionId);\r\n if (!channelWs || channelWs.readyState !== WebSocket.OPEN) return;\r\n\r\n // Validate mime type\r\n const allowedTypes = ['image/png', 'image/jpeg', 'image/gif', 'image/webp'];\r\n if (!allowedTypes.includes(mimeType)) return;\r\n\r\n // Extract base64 data (remove data URL prefix if present)\r\n const base64Data = imageData.replace(/^data:image\\/\\w+;base64,/, '');\r\n const buffer = Buffer.from(base64Data, 'base64');\r\n\r\n // Validate size (10MB max)\r\n if (buffer.length > 10 * 1024 * 1024) {\r\n logger.warn('Image upload rejected: exceeds 10MB');\r\n return;\r\n }\r\n\r\n // Save to uploads dir\r\n fs.mkdirSync(UPLOADS_DIR, { recursive: true });\r\n const ext = mimeType.split('/')[1] === 'jpeg' ? 'jpg' : mimeType.split('/')[1];\r\n const filename = `${randomUUID()}.${ext}`;\r\n const filePath = path.join(UPLOADS_DIR, filename);\r\n fs.writeFileSync(filePath, buffer);\r\n\r\n // Forward file path to channel\r\n const forwardMsg: ChannelMessage = {\r\n type: 'image_to_session',\r\n sessionId,\r\n imagePath: filePath,\r\n mimeType,\r\n originalName,\r\n content,\r\n };\r\n channelWs.send(JSON.stringify(forwardMsg));\r\n logger.info(`Image saved and forwarded: ${filename} (${buffer.length} bytes)`);\r\n\r\n // Cleanup after 5 minutes\r\n setTimeout(() => {\r\n try { fs.unlinkSync(filePath); } catch {}\r\n }, 5 * 60 * 1000);\r\n }\r\n\r\n private async handleWebhookSave(req: http.IncomingMessage, res: http.ServerResponse): Promise<void> {\r\n const body = await this.readBody(req);\r\n if (!body) { this.jsonResponse(res, 400, { error: 'Invalid JSON' }); return; }\r\n const { webhooks } = body as { webhooks?: WebhookConfig[] };\r\n if (!Array.isArray(webhooks)) { this.jsonResponse(res, 400, { error: 'webhooks must be an array' }); return; }\r\n const config = loadConfig();\r\n config.webhooks = webhooks;\r\n saveConfig(config);\r\n this.notifier.configure({ webhooks });\r\n this.jsonResponse(res, 200, { ok: true });\r\n }\r\n\r\n private initTelegram(config: TelegramConfig): void {\r\n this.telegramBot = new TelegramBot(config);\r\n this.telegramBot.getSessions = () => this.sessions.getAll();\r\n this.telegramBot.onMessageToSession = (sessionId, content) => {\r\n const channelWs = this.channelSockets.get(sessionId);\r\n if (channelWs?.readyState === WebSocket.OPEN) {\r\n const msg: ChannelMessage = { type: 'message_to_session', sessionId, content };\r\n channelWs.send(JSON.stringify(msg));\r\n logger.info(`Telegram message forwarded to session: ${sessionId}`);\r\n }\r\n };\r\n this.telegramBot.onImageToSession = (sessionId, imagePath, mimeType, caption) => {\r\n const channelWs = this.channelSockets.get(sessionId);\r\n if (channelWs?.readyState === WebSocket.OPEN) {\r\n const msg: ChannelMessage = { type: 'image_to_session', sessionId, imagePath, mimeType, content: caption };\r\n channelWs.send(JSON.stringify(msg));\r\n logger.info(`Telegram photo forwarded to session: ${sessionId}`);\r\n }\r\n };\r\n this.telegramBot.onPermissionVerdict = (sessionId, requestId, behavior) => {\r\n const channelWs = this.channelSockets.get(sessionId);\r\n if (channelWs?.readyState === WebSocket.OPEN) {\r\n const msg: ChannelMessage = { type: 'permission_response', sessionId, requestId, behavior };\r\n channelWs.send(JSON.stringify(msg));\r\n logger.info(`Telegram permission verdict [${requestId}]: ${behavior} -> session ${sessionId}`);\r\n }\r\n // Also notify dashboards so they can dismiss the permission bar\r\n this.broadcastToDashboards({ type: 'permission_response', sessionId, requestId, behavior });\r\n };\r\n this.notifier.configure({ telegramBot: this.telegramBot });\r\n this.telegramBot.startPolling();\r\n logger.info('Telegram bot initialized');\r\n }\r\n\r\n private async handleTelegramSave(req: http.IncomingMessage, res: http.ServerResponse): Promise<void> {\r\n const body = await this.readBody(req);\r\n if (!body) { this.jsonResponse(res, 400, { error: 'Invalid JSON' }); return; }\r\n const { telegram } = body as { telegram?: TelegramConfig };\r\n if (!telegram) { this.jsonResponse(res, 400, { error: 'telegram config required' }); return; }\r\n\r\n const config = loadConfig();\r\n // If botToken is masked (contains '...'), keep the existing token\r\n if (telegram.botToken.includes('...') && config.telegram?.botToken) {\r\n telegram.botToken = config.telegram.botToken;\r\n }\r\n config.telegram = telegram;\r\n saveConfig(config);\r\n\r\n // Stop existing bot if running\r\n if (this.telegramBot) {\r\n this.telegramBot.stopPolling();\r\n this.telegramBot = undefined;\r\n this.notifier.configure({ telegramBot: undefined as any });\r\n }\r\n\r\n // Start new bot if enabled\r\n if (telegram.enabled && telegram.botToken && telegram.chatId) {\r\n this.initTelegram(telegram);\r\n }\r\n\r\n this.jsonResponse(res, 200, { ok: true });\r\n }\r\n\r\n private async handleTelegramTest(req: http.IncomingMessage, res: http.ServerResponse): Promise<void> {\r\n const body = await this.readBody(req);\r\n if (!body) { this.jsonResponse(res, 400, { error: 'Invalid JSON' }); return; }\r\n const { botToken, chatId } = body as { botToken?: string; chatId?: string };\r\n if (!botToken || !chatId) {\r\n this.jsonResponse(res, 400, { error: 'botToken and chatId required' });\r\n return;\r\n }\r\n\r\n try {\r\n const testRes = await fetch(`https://api.telegram.org/bot${botToken}/sendMessage`, {\r\n method: 'POST',\r\n headers: { 'Content-Type': 'application/json' },\r\n body: JSON.stringify({\r\n chat_id: chatId,\r\n text: 'Claude Alarm test message! Connection successful.',\r\n }),\r\n });\r\n\r\n if (testRes.ok) {\r\n this.jsonResponse(res, 200, { ok: true });\r\n } else {\r\n const err = await testRes.json() as { description?: string };\r\n this.jsonResponse(res, 400, { error: (err as any).description || 'Telegram API error' });\r\n }\r\n } catch (err) {\r\n this.jsonResponse(res, 500, { error: (err as Error).message });\r\n }\r\n }\r\n\r\n private async handleTelegramDetect(req: http.IncomingMessage, res: http.ServerResponse): Promise<void> {\r\n const body = await this.readBody(req);\r\n if (!body) { this.jsonResponse(res, 400, { error: 'Invalid JSON' }); return; }\r\n const { botToken } = body as { botToken?: string };\r\n if (!botToken) {\r\n this.jsonResponse(res, 400, { error: 'botToken required' });\r\n return;\r\n }\r\n\r\n try {\r\n const detectRes = await fetch(`https://api.telegram.org/bot${botToken}/getUpdates?timeout=0&limit=10`, {\r\n signal: AbortSignal.timeout(10000),\r\n });\r\n\r\n if (!detectRes.ok) {\r\n const err = await detectRes.json() as { description?: string };\r\n this.jsonResponse(res, 400, { error: (err as any).description || 'Invalid bot token' });\r\n return;\r\n }\r\n\r\n const data = await detectRes.json() as { ok: boolean; result: Array<{ message?: { chat: { id: number; first_name?: string; title?: string; type: string } } }> };\r\n if (!data.ok || !data.result.length) {\r\n this.jsonResponse(res, 200, { ok: false, chats: [] });\r\n return;\r\n }\r\n\r\n // Extract unique chats\r\n const chatMap = new Map<string, { id: string; name: string; type: string }>();\r\n for (const update of data.result) {\r\n if (update.message?.chat) {\r\n const chat = update.message.chat;\r\n const id = String(chat.id);\r\n if (!chatMap.has(id)) {\r\n chatMap.set(id, {\r\n id,\r\n name: chat.title || chat.first_name || id,\r\n type: chat.type,\r\n });\r\n }\r\n }\r\n }\r\n\r\n this.jsonResponse(res, 200, { ok: true, chats: [...chatMap.values()] });\r\n } catch (err) {\r\n this.jsonResponse(res, 500, { error: (err as Error).message });\r\n }\r\n }\r\n\r\n private cleanupUploads(): void {\r\n try {\r\n if (!fs.existsSync(UPLOADS_DIR)) return;\r\n const files = fs.readdirSync(UPLOADS_DIR);\r\n for (const file of files) {\r\n try { fs.unlinkSync(path.join(UPLOADS_DIR, file)); } catch {}\r\n }\r\n if (files.length > 0) logger.info(`Cleaned up ${files.length} leftover upload(s)`);\r\n } catch {}\r\n }\r\n\r\n private getSessionLabel(session?: SessionInfo): string {\r\n if (!session) return 'unknown';\r\n return session.displayName || session.cwd?.replace(/^.*[/\\\\]/, '') || session.name;\r\n }\r\n\r\n private jsonResponse(res: http.ServerResponse, status: number, body: unknown): void {\r\n res.writeHead(status, { 'Content-Type': 'application/json' });\r\n res.end(JSON.stringify(body));\r\n }\r\n\r\n private isLocalRequest(req: http.IncomingMessage): boolean {\r\n const addr = req.socket.remoteAddress;\r\n return addr === '127.0.0.1' || addr === '::1' || addr === '::ffff:127.0.0.1';\r\n }\r\n\r\n private readBody(req: http.IncomingMessage, maxSize = 1024 * 1024): Promise<unknown | null> {\r\n return new Promise((resolve) => {\r\n let data = '';\r\n let size = 0;\r\n req.on('data', (chunk) => {\r\n size += chunk.length;\r\n if (size > maxSize) {\r\n req.destroy();\r\n resolve(null);\r\n return;\r\n }\r\n data += chunk;\r\n });\r\n req.on('end', () => {\r\n try {\r\n resolve(JSON.parse(data));\r\n } catch {\r\n resolve(null);\r\n }\r\n });\r\n });\r\n }\r\n}\r\n\r\n// Direct execution support\r\nif (process.argv[1] && (\r\n process.argv[1].endsWith('hub/server.js') ||\r\n process.argv[1].endsWith('hub/server.ts')\r\n)) {\r\n const hub = new HubServer();\r\n hub.start().catch((err) => {\r\n logger.error('Failed to start hub:', err);\r\n process.exit(1);\r\n });\r\n\r\n const shutdown = () => {\r\n hub.stop().then(() => process.exit(0));\r\n };\r\n process.on('SIGINT', shutdown);\r\n process.on('SIGTERM', shutdown);\r\n}\r\n","/**\r\n * Logger that writes to stderr only.\r\n * CRITICAL: In MCP channel servers, stdout is used for the stdio protocol.\r\n * Any console.log() would corrupt the protocol. Always use this logger.\r\n */\r\nexport const logger = {\r\n info(msg: string, ...args: unknown[]) {\r\n console.error(`[claude-alarm] ${msg}`, ...args);\r\n },\r\n warn(msg: string, ...args: unknown[]) {\r\n console.error(`[claude-alarm WARN] ${msg}`, ...args);\r\n },\r\n error(msg: string, ...args: unknown[]) {\r\n console.error(`[claude-alarm ERROR] ${msg}`, ...args);\r\n },\r\n debug(msg: string, ...args: unknown[]) {\r\n if (process.env.CLAUDE_ALARM_DEBUG) {\r\n console.error(`[claude-alarm DEBUG] ${msg}`, ...args);\r\n }\r\n },\r\n};\r\n","import path from 'node:path';\r\nimport os from 'node:os';\r\n\r\nexport const DEFAULT_HUB_HOST = '127.0.0.1';\r\nexport const DEFAULT_HUB_PORT = 7900;\r\n\r\nexport const CONFIG_DIR = path.join(os.homedir(), '.claude-alarm');\r\nexport const CONFIG_FILE = path.join(CONFIG_DIR, 'config.json');\r\nexport const PID_FILE = path.join(CONFIG_DIR, 'hub.pid');\r\nexport const LOG_FILE = path.join(CONFIG_DIR, 'hub.log');\r\nexport const UPLOADS_DIR = path.join(CONFIG_DIR, 'uploads');\r\n\r\nexport const WS_PATH_CHANNEL = '/ws/channel';\r\nexport const WS_PATH_DASHBOARD = '/ws/dashboard';\r\n\r\nexport const CHANNEL_SERVER_NAME = 'claude-alarm';\r\nexport const CHANNEL_SERVER_VERSION = '0.1.0';\r\n","import type { SessionInfo, SessionStatus } from '../shared/types.js';\r\n\r\nexport class SessionManager {\r\n private sessions = new Map<string, SessionInfo>();\r\n\r\n register(session: SessionInfo): void {\r\n // Auto-number duplicate names\r\n const baseName = session.cwd?.replace(/^.*[/\\\\]/, '') || session.name;\r\n const existing = Array.from(this.sessions.values()).filter(\r\n s => s.id !== session.id && (s.cwd?.replace(/^.*[/\\\\]/, '') || s.name) === baseName,\r\n );\r\n if (existing.length > 0) {\r\n // Number this one\r\n const usedNums = existing.map(s => {\r\n const m = s.displayName?.match(/\\((\\d+)\\)$/);\r\n return m ? parseInt(m[1], 10) : 1;\r\n });\r\n // Ensure first existing session has (1) if it doesn't yet\r\n for (const s of existing) {\r\n if (!s.displayName?.match(/\\(\\d+\\)$/)) {\r\n s.displayName = `${baseName} (1)`;\r\n }\r\n }\r\n const nextNum = Math.max(...usedNums, 1) + 1;\r\n session.displayName = `${baseName} (${nextNum})`;\r\n } else {\r\n session.displayName = baseName;\r\n }\r\n this.sessions.set(session.id, { ...session });\r\n }\r\n\r\n unregister(sessionId: string): SessionInfo | undefined {\r\n const session = this.sessions.get(sessionId);\r\n this.sessions.delete(sessionId);\r\n return session;\r\n }\r\n\r\n updateStatus(sessionId: string, status: SessionStatus): SessionInfo | undefined {\r\n const session = this.sessions.get(sessionId);\r\n if (session) {\r\n session.status = status;\r\n session.lastActivity = Date.now();\r\n }\r\n return session;\r\n }\r\n\r\n updateActivity(sessionId: string): void {\r\n const session = this.sessions.get(sessionId);\r\n if (session) {\r\n session.lastActivity = Date.now();\r\n }\r\n }\r\n\r\n get(sessionId: string): SessionInfo | undefined {\r\n return this.sessions.get(sessionId);\r\n }\r\n\r\n getAll(): SessionInfo[] {\r\n return Array.from(this.sessions.values());\r\n }\r\n\r\n count(): number {\r\n return this.sessions.size;\r\n }\r\n}\r\n","import notifier from 'node-notifier';\r\nimport { execFile } from 'node:child_process';\r\nimport { logger } from '../shared/logger.js';\r\nimport type { NotifyLevel, WebhookConfig } from '../shared/types.js';\r\nimport type { TelegramBot } from './telegram.js';\r\n\r\nexport class Notifier {\r\n private webhooks: WebhookConfig[] = [];\r\n private desktopEnabled = true;\r\n private notificationSettingsOpened = false;\r\n private dashboardUrl?: string;\r\n private telegramBot?: TelegramBot;\r\n\r\n configure(options: { desktop?: boolean; webhooks?: WebhookConfig[]; dashboardUrl?: string; telegramBot?: TelegramBot }): void {\r\n if (options.dashboardUrl) this.dashboardUrl = options.dashboardUrl;\r\n if (options.desktop !== undefined) this.desktopEnabled = options.desktop;\r\n if (options.webhooks) this.webhooks = options.webhooks;\r\n if (options.telegramBot) this.telegramBot = options.telegramBot;\r\n }\r\n\r\n async notify(title: string, message: string, level: NotifyLevel = 'info'): Promise<void> {\r\n await this.notifyWithSession(undefined, undefined, title, message, level);\r\n }\r\n\r\n async notifyWithSession(sessionId: string | undefined, sessionLabel: string | undefined, title: string, message: string, level: NotifyLevel = 'info'): Promise<void> {\r\n const promises: Promise<void>[] = [];\r\n\r\n if (this.desktopEnabled) {\r\n promises.push(this.sendDesktop(title, message, level));\r\n }\r\n\r\n for (const webhook of this.webhooks) {\r\n promises.push(this.sendWebhook(webhook, title, message, level));\r\n }\r\n\r\n if (this.telegramBot && sessionId && sessionLabel) {\r\n promises.push(this.telegramBot.sendNotification(sessionId, sessionLabel, title, message));\r\n }\r\n\r\n await Promise.allSettled(promises);\r\n }\r\n\r\n private async sendDesktop(title: string, message: string, _level: NotifyLevel): Promise<void> {\r\n if (process.platform === 'win32') {\r\n // Check if notifications are enabled by running snoretoast directly\r\n const enabled = await this.checkWindowsNotifications();\r\n if (!enabled) {\r\n this.openNotificationSettings();\r\n return;\r\n }\r\n }\r\n\r\n return new Promise((resolve) => {\r\n const notification = (notifier as any).notify(\r\n {\r\n title: `Claude Alarm: ${title}`,\r\n message,\r\n sound: true,\r\n wait: true,\r\n },\r\n (err: Error | null) => {\r\n if (err) {\r\n logger.warn(`Desktop notification failed: ${err.message}`);\r\n }\r\n resolve();\r\n },\r\n );\r\n\r\n // No click handler - dashboard is already open and notifications\r\n // can be clicked there to focus the relevant session\r\n });\r\n }\r\n\r\n private checkWindowsNotifications(): Promise<boolean> {\r\n return new Promise((resolve) => {\r\n execFile(\r\n 'powershell',\r\n ['-Command', '(Get-ItemProperty -Path \"HKCU:\\\\Software\\\\Microsoft\\\\Windows\\\\CurrentVersion\\\\PushNotifications\" -Name ToastEnabled -ErrorAction SilentlyContinue).ToastEnabled'],\r\n (err, stdout) => {\r\n if (err) { resolve(true); return; } // assume enabled on error\r\n const value = stdout.trim();\r\n resolve(value !== '0');\r\n },\r\n );\r\n });\r\n }\r\n\r\n private openNotificationSettings(): void {\r\n if (this.notificationSettingsOpened) return;\r\n this.notificationSettingsOpened = true;\r\n\r\n logger.warn('Windows notifications are disabled. Opening notification settings...');\r\n logger.warn('Please enable notifications for this app, then try again.');\r\n\r\n if (process.platform === 'win32') {\r\n execFile('powershell', ['-Command', 'Start-Process ms-settings:notifications']);\r\n }\r\n\r\n // Allow re-opening after 5 minutes\r\n setTimeout(() => { this.notificationSettingsOpened = false; }, 5 * 60 * 1000);\r\n }\r\n\r\n private async sendWebhook(\r\n webhook: WebhookConfig,\r\n title: string,\r\n message: string,\r\n level: NotifyLevel,\r\n ): Promise<void> {\r\n try {\r\n const response = await fetch(webhook.url, {\r\n method: 'POST',\r\n headers: {\r\n 'Content-Type': 'application/json',\r\n ...webhook.headers,\r\n },\r\n body: JSON.stringify({\r\n title,\r\n message,\r\n level,\r\n timestamp: Date.now(),\r\n source: 'claude-alarm',\r\n }),\r\n });\r\n\r\n if (!response.ok) {\r\n logger.warn(`Webhook ${webhook.url} returned ${response.status}`);\r\n }\r\n } catch (err) {\r\n logger.warn(`Webhook ${webhook.url} failed: ${(err as Error).message}`);\r\n }\r\n }\r\n}\r\n","import fs from 'node:fs';\r\nimport path from 'node:path';\r\nimport { randomUUID } from 'node:crypto';\r\nimport { logger } from '../shared/logger.js';\r\nimport { UPLOADS_DIR } from '../shared/constants.js';\r\nimport type { TelegramConfig, SessionInfo } from '../shared/types.js';\r\n\r\nconst TELEGRAM_API = 'https://api.telegram.org/bot';\r\n\r\ninterface TelegramPhotoSize {\r\n file_id: string;\r\n file_unique_id: string;\r\n width: number;\r\n height: number;\r\n file_size?: number;\r\n}\r\n\r\ninterface TelegramMessage {\r\n message_id: number;\r\n chat: { id: number };\r\n text?: string;\r\n caption?: string;\r\n photo?: TelegramPhotoSize[];\r\n reply_to_message?: { message_id: number };\r\n}\r\n\r\ninterface TelegramCallbackQuery {\r\n id: string;\r\n from: { id: number };\r\n message?: TelegramMessage;\r\n data?: string;\r\n}\r\n\r\ninterface TelegramUpdate {\r\n update_id: number;\r\n message?: TelegramMessage;\r\n callback_query?: TelegramCallbackQuery;\r\n}\r\n\r\nexport class TelegramBot {\r\n private config: TelegramConfig;\r\n private offset = 0;\r\n private polling = false;\r\n private pollTimer: ReturnType<typeof setTimeout> | null = null;\r\n\r\n // message_id -> sessionId mapping for reply-based routing\r\n private messageSessionMap = new Map<number, string>();\r\n\r\n // Callback: when a text message arrives from Telegram for a session\r\n public onMessageToSession?: (sessionId: string, content: string) => void;\r\n // Callback: when an image arrives from Telegram for a session\r\n public onImageToSession?: (sessionId: string, imagePath: string, mimeType: string, caption?: string) => void;\r\n // Callback: when a permission verdict arrives from Telegram\r\n public onPermissionVerdict?: (sessionId: string, requestId: string, behavior: 'allow' | 'deny') => void;\r\n // Callback: get current sessions list\r\n public getSessions?: () => SessionInfo[];\r\n // Pending messages for session selection\r\n private pendingMessages = new Map<number, { text?: string; photoFileId?: string; caption?: string }>(); // chatId -> pending\r\n\r\n constructor(config: TelegramConfig) {\r\n this.config = config;\r\n }\r\n\r\n private get apiUrl(): string {\r\n return `${TELEGRAM_API}${this.config.botToken}`;\r\n }\r\n\r\n /** Send a notification message to Telegram */\r\n async sendNotification(sessionId: string, _sessionLabel: string, title: string, message: string): Promise<void> {\r\n const text = `<b>${this.escHtml(title)}</b>\\n${this.mdToHtml(message)}`;\r\n const result = await this.sendMessage(text);\r\n if (result?.message_id) {\r\n this.messageSessionMap.set(result.message_id, sessionId);\r\n // Cleanup old mappings (keep last 200)\r\n if (this.messageSessionMap.size > 200) {\r\n const keys = [...this.messageSessionMap.keys()];\r\n for (let i = 0; i < keys.length - 200; i++) {\r\n this.messageSessionMap.delete(keys[i]);\r\n }\r\n }\r\n }\r\n }\r\n\r\n /** Send a text message to the configured chat */\r\n private async sendMessage(text: string, replyToMessageId?: number, replyMarkup?: unknown): Promise<{ message_id: number } | null> {\r\n try {\r\n const body: Record<string, unknown> = {\r\n chat_id: this.config.chatId,\r\n text,\r\n parse_mode: 'HTML',\r\n };\r\n if (replyToMessageId) body.reply_to_message_id = replyToMessageId;\r\n if (replyMarkup) body.reply_markup = replyMarkup;\r\n\r\n const res = await fetch(`${this.apiUrl}/sendMessage`, {\r\n method: 'POST',\r\n headers: { 'Content-Type': 'application/json' },\r\n body: JSON.stringify(body),\r\n });\r\n\r\n if (!res.ok) {\r\n const err = await res.text();\r\n logger.warn(`Telegram sendMessage failed: ${res.status} ${err}`);\r\n return null;\r\n }\r\n\r\n const data = await res.json() as { ok: boolean; result: { message_id: number } };\r\n return data.ok ? data.result : null;\r\n } catch (err) {\r\n logger.warn(`Telegram sendMessage error: ${(err as Error).message}`);\r\n return null;\r\n }\r\n }\r\n\r\n /** Start long polling for incoming messages */\r\n startPolling(): void {\r\n if (this.polling) return;\r\n this.polling = true;\r\n logger.info('Telegram bot polling started');\r\n this.poll();\r\n }\r\n\r\n /** Stop polling */\r\n stopPolling(): void {\r\n this.polling = false;\r\n if (this.pollTimer) {\r\n clearTimeout(this.pollTimer);\r\n this.pollTimer = null;\r\n }\r\n logger.info('Telegram bot polling stopped');\r\n }\r\n\r\n private async poll(): Promise<void> {\r\n if (!this.polling) return;\r\n\r\n try {\r\n const res = await fetch(`${this.apiUrl}/getUpdates?offset=${this.offset}&timeout=30`, {\r\n signal: AbortSignal.timeout(35000),\r\n });\r\n\r\n if (!res.ok) {\r\n logger.warn(`Telegram getUpdates failed: ${res.status}`);\r\n this.scheduleNextPoll(5000);\r\n return;\r\n }\r\n\r\n const data = await res.json() as { ok: boolean; result: TelegramUpdate[] };\r\n if (data.ok && data.result.length > 0) {\r\n for (const update of data.result) {\r\n this.offset = update.update_id + 1;\r\n if (update.callback_query) {\r\n this.handleCallbackQuery(update.callback_query);\r\n } else if (update.message) {\r\n this.handleIncomingMessage(update.message);\r\n }\r\n }\r\n }\r\n } catch (err) {\r\n if ((err as Error).name !== 'AbortError') {\r\n logger.warn(`Telegram poll error: ${(err as Error).message}`);\r\n }\r\n }\r\n\r\n this.scheduleNextPoll(1000);\r\n }\r\n\r\n private scheduleNextPoll(delay: number): void {\r\n if (!this.polling) return;\r\n this.pollTimer = setTimeout(() => this.poll(), delay);\r\n }\r\n\r\n private async handleIncomingMessage(msg: TelegramMessage): Promise<void> {\r\n // Only process messages from the configured chat\r\n if (String(msg.chat.id) !== String(this.config.chatId)) return;\r\n\r\n const hasPhoto = msg.photo && msg.photo.length > 0;\r\n const text = (msg.text || msg.caption || '').trim();\r\n\r\n if (!text && !hasPhoto) return;\r\n\r\n // Check if it's a reply to a known message\r\n if (msg.reply_to_message) {\r\n const sessionId = this.messageSessionMap.get(msg.reply_to_message.message_id);\r\n if (sessionId) {\r\n if (hasPhoto) {\r\n await this.deliverPhotoToSession(sessionId, msg.photo!, text);\r\n } else {\r\n this.deliverToSession(sessionId, text);\r\n }\r\n return;\r\n }\r\n }\r\n\r\n // Check if it's a session selection command: /s_<index>\r\n if (text) {\r\n const selectMatch = text.match(/^\\/s_(\\d+)$/);\r\n if (selectMatch) {\r\n const pending = this.pendingMessages.get(msg.chat.id);\r\n if (pending) {\r\n this.pendingMessages.delete(msg.chat.id);\r\n const sessions = this.getSessions?.() ?? [];\r\n const idx = parseInt(selectMatch[1], 10) - 1;\r\n if (idx >= 0 && idx < sessions.length) {\r\n if (pending.photoFileId) {\r\n await this.deliverPhotoToSessionByFileId(sessions[idx].id, pending.photoFileId, pending.caption);\r\n } else if (pending.text) {\r\n this.deliverToSession(sessions[idx].id, pending.text);\r\n }\r\n this.sendMessage(`Sent to [${this.getLabel(sessions[idx])}]`);\r\n } else {\r\n this.sendMessage('Invalid session number.');\r\n }\r\n return;\r\n }\r\n }\r\n }\r\n\r\n // No reply context — check sessions count\r\n const sessions = this.getSessions?.() ?? [];\r\n\r\n if (sessions.length === 0) {\r\n this.sendMessage('No active sessions connected.');\r\n return;\r\n }\r\n\r\n if (sessions.length === 1) {\r\n if (hasPhoto) {\r\n await this.deliverPhotoToSession(sessions[0].id, msg.photo!, text);\r\n } else {\r\n this.deliverToSession(sessions[0].id, text);\r\n }\r\n return;\r\n }\r\n\r\n // Multiple sessions — ask user to pick with inline buttons\r\n if (hasPhoto) {\r\n const largest = msg.photo![msg.photo!.length - 1];\r\n this.pendingMessages.set(msg.chat.id, { photoFileId: largest.file_id, caption: text });\r\n } else {\r\n this.pendingMessages.set(msg.chat.id, { text });\r\n }\r\n const buttons = sessions.map((s, i) => ({\r\n text: this.getLabel(s),\r\n callback_data: `sess:${i}:${msg.chat.id}`,\r\n }));\r\n // Arrange buttons in rows of 2\r\n const rows: Array<typeof buttons> = [];\r\n for (let i = 0; i < buttons.length; i += 2) {\r\n rows.push(buttons.slice(i, i + 2));\r\n }\r\n this.sendMessage('Multiple sessions active. Select one:', undefined, { inline_keyboard: rows });\r\n }\r\n\r\n private deliverToSession(sessionId: string, content: string): void {\r\n if (this.onMessageToSession) {\r\n this.onMessageToSession(sessionId, content);\r\n }\r\n }\r\n\r\n private async deliverPhotoToSession(sessionId: string, photos: TelegramPhotoSize[], caption?: string): Promise<void> {\r\n // Get the largest photo (last in array)\r\n const largest = photos[photos.length - 1];\r\n await this.deliverPhotoToSessionByFileId(sessionId, largest.file_id, caption);\r\n }\r\n\r\n private async deliverPhotoToSessionByFileId(sessionId: string, fileId: string, caption?: string): Promise<void> {\r\n try {\r\n // Get file path from Telegram\r\n const fileRes = await fetch(`${this.apiUrl}/getFile?file_id=${fileId}`);\r\n if (!fileRes.ok) { logger.warn('Failed to get Telegram file info'); return; }\r\n const fileData = await fileRes.json() as { ok: boolean; result: { file_path: string } };\r\n if (!fileData.ok) return;\r\n\r\n // Download the file\r\n const downloadUrl = `https://api.telegram.org/file/bot${this.config.botToken}/${fileData.result.file_path}`;\r\n const imgRes = await fetch(downloadUrl);\r\n if (!imgRes.ok) { logger.warn('Failed to download Telegram photo'); return; }\r\n const buffer = Buffer.from(await imgRes.arrayBuffer());\r\n\r\n // Determine extension\r\n const ext = fileData.result.file_path.split('.').pop() || 'jpg';\r\n const mimeType = ext === 'png' ? 'image/png' : ext === 'gif' ? 'image/gif' : ext === 'webp' ? 'image/webp' : 'image/jpeg';\r\n\r\n // Save to uploads dir\r\n fs.mkdirSync(UPLOADS_DIR, { recursive: true });\r\n const filename = `${randomUUID()}.${ext}`;\r\n const filePath = path.join(UPLOADS_DIR, filename);\r\n fs.writeFileSync(filePath, buffer);\r\n logger.info(`Telegram photo saved: ${filename} (${buffer.length} bytes)`);\r\n\r\n // Deliver to session\r\n if (this.onImageToSession) {\r\n this.onImageToSession(sessionId, filePath, mimeType, caption);\r\n }\r\n\r\n // Cleanup after 5 minutes\r\n setTimeout(() => { try { fs.unlinkSync(filePath); } catch {} }, 5 * 60 * 1000);\r\n } catch (err) {\r\n logger.warn(`Telegram photo download failed: ${(err as Error).message}`);\r\n }\r\n }\r\n\r\n private getLabel(session: SessionInfo): string {\r\n return session.displayName || session.cwd?.replace(/^.*[/\\\\]/, '') || session.name;\r\n }\r\n\r\n /** Send a permission request with inline buttons */\r\n async sendPermissionRequest(sessionId: string, sessionLabel: string, requestId: string, toolName: string, description: string, inputPreview: string): Promise<void> {\r\n // Parse inputPreview for readable display\r\n let preview = inputPreview;\r\n let truncated = false;\r\n try {\r\n const p = JSON.parse(inputPreview);\r\n if (p.command) preview = `$ ${p.command}`;\r\n else if (p.title && p.message) preview = p.message;\r\n else if (p.file_path) {\r\n preview = p.file_path;\r\n if (p.content) { preview += '\\n' + p.content.slice(0, 3000); if (p.content.length > 500) truncated = true; }\r\n } else if (p.content && typeof p.content === 'string') {\r\n preview = p.content.slice(0, 3000);\r\n if (p.content.length > 500) truncated = true;\r\n }\r\n } catch {\r\n truncated = true;\r\n const cmdMatch = inputPreview.match(/\"command\"\\s*:\\s*\"((?:[^\"\\\\]|\\\\.)*)\"/);\r\n const contentMatch = inputPreview.match(/\"content\"\\s*:\\s*\"((?:[^\"\\\\]|\\\\.)*)\"/);\r\n if (cmdMatch) preview = `$ ${cmdMatch[1]}`;\r\n else if (contentMatch) preview = contentMatch[1].slice(0, 3000);\r\n }\r\n\r\n const truncNote = truncated ? '\\n\\n<i>...truncated</i>' : '';\r\n const previewSlice = preview.slice(0, 3000);\r\n // Use <code> for short single-line (commands), plain text for longer content\r\n const isShort = !previewSlice.includes('\\n') && previewSlice.length < 100;\r\n const previewHtml = isShort\r\n ? `<code>${this.escHtml(previewSlice)}</code>`\r\n : this.escHtml(previewSlice);\r\n // Friendly tool name for Telegram — use title for notify tools\r\n let displayTool = toolName;\r\n if ((toolName.endsWith('__notify') || toolName === 'notify') && inputPreview) {\r\n try { const pp = JSON.parse(inputPreview); if (pp.title) displayTool = pp.title; } catch {\r\n const tm = inputPreview.match(/\"title\"\\s*:\\s*\"((?:[^\"\\\\]|\\\\.)*)\"/);\r\n if (tm) displayTool = tm[1];\r\n }\r\n } else {\r\n const mcpMatch = toolName.match(/__([^_]+)$/);\r\n if (mcpMatch) displayTool = mcpMatch[1].charAt(0).toUpperCase() + mcpMatch[1].slice(1);\r\n }\r\n\r\n const text = `⚠️ <b>Permission Request</b> — ${this.escHtml(sessionLabel)}\\n\\n` +\r\n `🔧 <b>${this.escHtml(displayTool)}</b>\\n` +\r\n `${previewHtml}${truncNote}`;\r\n\r\n const replyMarkup = {\r\n inline_keyboard: [[\r\n { text: '✅ Allow', callback_data: `perm:allow:${sessionId}:${requestId}` },\r\n { text: '❌ Deny', callback_data: `perm:deny:${sessionId}:${requestId}` },\r\n ]],\r\n };\r\n\r\n await this.sendMessage(text, undefined, replyMarkup);\r\n }\r\n\r\n private async handleCallbackQuery(query: TelegramCallbackQuery): Promise<void> {\r\n if (!query.data) return;\r\n\r\n if (query.data.startsWith('sess:')) {\r\n await this.handleSessionSelectCallback(query);\r\n return;\r\n }\r\n\r\n if (!query.data.startsWith('perm:')) return;\r\n\r\n const parts = query.data.split(':');\r\n if (parts.length < 4) return;\r\n const [, action, sessionId, requestId] = parts;\r\n const behavior = action === 'allow' ? 'allow' : 'deny';\r\n\r\n // Send verdict\r\n if (this.onPermissionVerdict) {\r\n this.onPermissionVerdict(sessionId, requestId, behavior as 'allow' | 'deny');\r\n }\r\n\r\n // Answer callback to remove loading state\r\n await this.answerCallbackQuery(query.id, behavior === 'allow' ? '✅ Allowed' : '❌ Denied');\r\n\r\n // Update message to show result (use escaped original text since it's plain)\r\n if (query.message) {\r\n const label = behavior === 'allow' ? '✅ <b>Allowed</b>' : '❌ <b>Denied</b>';\r\n const original = this.escHtml(query.message.text || '');\r\n await this.editMessageText(query.message.chat.id, query.message.message_id, original + `\\n\\n${label}`);\r\n }\r\n }\r\n\r\n private async handleSessionSelectCallback(query: TelegramCallbackQuery): Promise<void> {\r\n const parts = query.data!.split(':');\r\n if (parts.length < 3) return;\r\n const [, idxStr, chatIdStr] = parts;\r\n const idx = parseInt(idxStr, 10);\r\n const chatId = parseInt(chatIdStr, 10);\r\n\r\n const sessions = this.getSessions?.() ?? [];\r\n if (idx < 0 || idx >= sessions.length) {\r\n await this.answerCallbackQuery(query.id, 'Session not found');\r\n return;\r\n }\r\n\r\n const session = sessions[idx];\r\n const pending = this.pendingMessages.get(chatId);\r\n this.pendingMessages.delete(chatId);\r\n\r\n if (pending) {\r\n if (pending.photoFileId) {\r\n await this.deliverPhotoToSessionByFileId(session.id, pending.photoFileId, pending.caption);\r\n } else if (pending.text) {\r\n this.deliverToSession(session.id, pending.text);\r\n }\r\n }\r\n\r\n await this.answerCallbackQuery(query.id, `Sent to ${this.getLabel(session)}`);\r\n // Update message to show which session was selected\r\n if (query.message) {\r\n await this.editMessageText(query.message.chat.id, query.message.message_id, `✅ Sent to <b>${this.escHtml(this.getLabel(session))}</b>`);\r\n }\r\n }\r\n\r\n private async answerCallbackQuery(callbackQueryId: string, text: string): Promise<void> {\r\n try {\r\n await fetch(`${this.apiUrl}/answerCallbackQuery`, {\r\n method: 'POST',\r\n headers: { 'Content-Type': 'application/json' },\r\n body: JSON.stringify({ callback_query_id: callbackQueryId, text }),\r\n });\r\n } catch (err) {\r\n logger.warn(`Telegram answerCallbackQuery error: ${(err as Error).message}`);\r\n }\r\n }\r\n\r\n private async editMessageText(chatId: number, messageId: number, text: string): Promise<void> {\r\n try {\r\n await fetch(`${this.apiUrl}/editMessageText`, {\r\n method: 'POST',\r\n headers: { 'Content-Type': 'application/json' },\r\n body: JSON.stringify({ chat_id: chatId, message_id: messageId, text, parse_mode: 'HTML' }),\r\n });\r\n } catch (err) {\r\n logger.warn(`Telegram editMessageText error: ${(err as Error).message}`);\r\n }\r\n }\r\n\r\n private escHtml(s: string): string {\r\n return s.replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>');\r\n }\r\n\r\n /** Convert markdown to Telegram HTML (escape first, then apply formatting) */\r\n private mdToHtml(s: string): string {\r\n // Handle tables before escaping (convert to clean text format)\r\n let text = s.replace(\r\n /^(\\|.+\\|)\\n\\|[-| :]+\\|\\n((?:\\|.+\\|\\n?)*)/gm,\r\n (_match, header: string, body: string) => {\r\n const headerCells = header.split('|').filter((c: string) => c.trim()).map((c: string) => c.trim());\r\n const headerLine = headerCells.join(' | ');\r\n const bodyLines = body.trim().split('\\n').map((row: string) => {\r\n return row.split('|').filter((c: string) => c.trim()).map((c: string) => c.trim()).join(' | ');\r\n });\r\n return `**${headerLine}**\\n${bodyLines.join('\\n')}`;\r\n },\r\n );\r\n let html = this.escHtml(text);\r\n // Code blocks: ```...```\r\n html = html.replace(/```(?:\\w*)\\n?([\\s\\S]*?)```/g, '<pre>$1</pre>');\r\n // Inline code: `...`\r\n html = html.replace(/`([^`]+)`/g, '<code>$1</code>');\r\n // Headings: # / ## / ### → bold (Telegram has no heading tags)\r\n html = html.replace(/^#{1,3}\\s+(.+)$/gm, '<b>$1</b>');\r\n // Bold: **...**\r\n html = html.replace(/\\*\\*(.+?)\\*\\*/g, '<b>$1</b>');\r\n // Italic: *...*\r\n html = html.replace(/\\*(.+?)\\*/g, '<i>$1</i>');\r\n return html;\r\n }\r\n\r\n /** Update config (e.g., from dashboard settings) */\r\n updateConfig(config: TelegramConfig): void {\r\n const wasPolling = this.polling;\r\n if (wasPolling) this.stopPolling();\r\n this.config = config;\r\n if (wasPolling && config.enabled) this.startPolling();\r\n }\r\n}\r\n","import fs from 'node:fs';\r\nimport path from 'node:path';\r\nimport { randomUUID } from 'node:crypto';\r\nimport { CONFIG_DIR, CONFIG_FILE, DEFAULT_HUB_HOST, DEFAULT_HUB_PORT } from './constants.js';\r\nimport type { AppConfig } from './types.js';\r\n\r\nconst DEFAULT_CONFIG: AppConfig = {\r\n hub: {\r\n host: DEFAULT_HUB_HOST,\r\n port: DEFAULT_HUB_PORT,\r\n },\r\n notifications: {\r\n desktop: true,\r\n sound: true,\r\n },\r\n webhooks: [],\r\n};\r\n\r\nexport function ensureConfigDir(): void {\r\n if (!fs.existsSync(CONFIG_DIR)) {\r\n fs.mkdirSync(CONFIG_DIR, { recursive: true });\r\n }\r\n}\r\n\r\nexport function loadConfig(): AppConfig {\r\n ensureConfigDir();\r\n let config: AppConfig;\r\n if (!fs.existsSync(CONFIG_FILE)) {\r\n config = { ...DEFAULT_CONFIG, hub: { ...DEFAULT_CONFIG.hub } };\r\n } else {\r\n try {\r\n const raw = fs.readFileSync(CONFIG_FILE, 'utf-8');\r\n const parsed = JSON.parse(raw);\r\n config = { ...DEFAULT_CONFIG, ...parsed, hub: { ...DEFAULT_CONFIG.hub, ...parsed.hub }, ...(parsed.telegram ? { telegram: parsed.telegram } : {}) };\r\n } catch {\r\n config = { ...DEFAULT_CONFIG, hub: { ...DEFAULT_CONFIG.hub } };\r\n }\r\n }\r\n\r\n // Auto-generate token if missing\r\n if (!config.hub.token) {\r\n config.hub.token = randomUUID();\r\n saveConfig(config);\r\n }\r\n\r\n return config;\r\n}\r\n\r\n/** Get the current token, generating one if needed */\r\nexport function getOrCreateToken(): string {\r\n const config = loadConfig();\r\n return config.hub.token!;\r\n}\r\n\r\nexport function saveConfig(config: AppConfig): void {\r\n ensureConfigDir();\r\n fs.writeFileSync(CONFIG_FILE, JSON.stringify(config, null, 2), { encoding: 'utf-8', mode: 0o600 });\r\n}\r\n\r\n/**\r\n * Add claude-alarm as an MCP channel server to .mcp.json\r\n */\r\nexport function setupMcpConfig(targetDir?: string): string {\r\n const dir = targetDir ?? process.cwd();\r\n const mcpPath = path.join(dir, '.mcp.json');\r\n\r\n let mcpConfig: Record<string, any> = {};\r\n if (fs.existsSync(mcpPath)) {\r\n try {\r\n mcpConfig = JSON.parse(fs.readFileSync(mcpPath, 'utf-8'));\r\n } catch {\r\n mcpConfig = {};\r\n }\r\n }\r\n\r\n if (!mcpConfig.mcpServers) {\r\n mcpConfig.mcpServers = {};\r\n }\r\n\r\n mcpConfig.mcpServers['claude-alarm'] = {\r\n command: 'npx',\r\n args: ['-y', '@delt/claude-alarm', 'serve'],\r\n env: {\r\n CLAUDE_ALARM_SESSION_NAME: path.basename(dir),\r\n },\r\n };\r\n\r\n fs.writeFileSync(mcpPath, JSON.stringify(mcpConfig, null, 2), 'utf-8');\r\n return mcpPath;\r\n}\r\n","import WebSocket from 'ws';\r\nimport { logger } from '../shared/logger.js';\r\nimport { DEFAULT_HUB_HOST, DEFAULT_HUB_PORT, WS_PATH_CHANNEL } from '../shared/constants.js';\r\nimport type { ChannelMessage, SessionInfo } from '../shared/types.js';\r\n\r\nexport class HubClient {\r\n private ws: WebSocket | null = null;\r\n private reconnectTimer: ReturnType<typeof setTimeout> | null = null;\r\n private messageHandlers: Array<(msg: ChannelMessage) => void> = [];\r\n private queue: ChannelMessage[] = [];\r\n private connected = false;\r\n\r\n constructor(\r\n private sessionId: string,\r\n private sessionName: string,\r\n private hubHost = DEFAULT_HUB_HOST,\r\n private hubPort = DEFAULT_HUB_PORT,\r\n private token?: string,\r\n ) {}\r\n\r\n connect(): void {\r\n const tokenQuery = this.token ? `?token=${encodeURIComponent(this.token)}` : '';\r\n const url = `ws://${this.hubHost}:${this.hubPort}${WS_PATH_CHANNEL}${tokenQuery}`;\r\n logger.debug(`Connecting to hub at ${url}`);\r\n\r\n try {\r\n this.ws = new WebSocket(url);\r\n\r\n this.ws.on('open', () => {\r\n logger.info('Connected to hub');\r\n this.connected = true;\r\n\r\n // Register this session\r\n const registration: ChannelMessage = {\r\n type: 'register',\r\n session: {\r\n id: this.sessionId,\r\n name: this.sessionName,\r\n status: 'idle',\r\n connectedAt: Date.now(),\r\n lastActivity: Date.now(),\r\n cwd: process.cwd(),\r\n channelEnabled: true,\r\n },\r\n };\r\n this.ws!.send(JSON.stringify(registration));\r\n\r\n // Flush queued messages\r\n for (const msg of this.queue) {\r\n this.ws!.send(JSON.stringify(msg));\r\n }\r\n this.queue = [];\r\n });\r\n\r\n this.ws.on('message', (data) => {\r\n try {\r\n const msg = JSON.parse(data.toString()) as ChannelMessage;\r\n for (const handler of this.messageHandlers) {\r\n handler(msg);\r\n }\r\n } catch (err) {\r\n logger.warn('Failed to parse hub message:', err);\r\n }\r\n });\r\n\r\n this.ws.on('close', () => {\r\n logger.info('Disconnected from hub');\r\n this.connected = false;\r\n this.scheduleReconnect();\r\n });\r\n\r\n this.ws.on('error', (err) => {\r\n logger.debug(`Hub connection error: ${err.message}`);\r\n this.connected = false;\r\n });\r\n } catch {\r\n logger.debug('Failed to connect to hub, will retry');\r\n this.scheduleReconnect();\r\n }\r\n }\r\n\r\n send(msg: ChannelMessage): void {\r\n if (this.connected && this.ws?.readyState === WebSocket.OPEN) {\r\n this.ws.send(JSON.stringify(msg));\r\n } else {\r\n if (this.queue.length < 100) {\r\n this.queue.push(msg);\r\n }\r\n logger.debug('Hub not connected, message queued');\r\n }\r\n }\r\n\r\n onMessage(handler: (msg: ChannelMessage) => void): void {\r\n this.messageHandlers.push(handler);\r\n }\r\n\r\n disconnect(): void {\r\n if (this.reconnectTimer) {\r\n clearTimeout(this.reconnectTimer);\r\n this.reconnectTimer = null;\r\n }\r\n if (this.ws) {\r\n this.ws.close();\r\n this.ws = null;\r\n }\r\n this.connected = false;\r\n }\r\n\r\n private scheduleReconnect(): void {\r\n if (this.reconnectTimer) return;\r\n this.reconnectTimer = setTimeout(() => {\r\n this.reconnectTimer = null;\r\n this.connect();\r\n }, 5000);\r\n }\r\n}\r\n"],"mappings":"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","names":["fs","path","randomUUID","path","sessions","path","fs","path","randomUUID","fs","randomUUID","fs","path","path","fs","randomUUID","WebSocket","WebSocket"]}
|
|
1
|
+
{"version":3,"sources":["../src/hub/server.ts","../src/shared/logger.ts","../src/shared/constants.ts","../src/hub/session-manager.ts","../src/hub/notifier.ts","../src/hub/telegram.ts","../src/shared/config.ts","../src/channel/hub-client.ts"],"sourcesContent":["import http from 'node:http';\r\nimport fs from 'node:fs';\r\nimport path from 'node:path';\r\nimport { fileURLToPath } from 'node:url';\r\nimport { WebSocketServer, WebSocket } from 'ws';\r\nimport { logger } from '../shared/logger.js';\r\nimport { randomUUID } from 'node:crypto';\r\nimport {\r\n DEFAULT_HUB_HOST,\r\n DEFAULT_HUB_PORT,\r\n WS_PATH_CHANNEL,\r\n WS_PATH_DASHBOARD,\r\n UPLOADS_DIR,\r\n} from '../shared/constants.js';\r\nimport { SessionManager } from './session-manager.js';\r\nimport { Notifier } from './notifier.js';\r\nimport { TelegramBot } from './telegram.js';\r\nimport { loadConfig, saveConfig } from '../shared/config.js';\r\nimport type { ChannelMessage, AppConfig, SessionInfo, WebhookConfig, TelegramConfig } from '../shared/types.js';\r\n\r\nconst __dirname = path.dirname(fileURLToPath(import.meta.url));\r\n\r\nexport class HubServer {\r\n private httpServer: http.Server;\r\n private wssChannel: WebSocketServer;\r\n private wssDashboard: WebSocketServer;\r\n private sessions = new SessionManager();\r\n private notifier = new Notifier();\r\n private startTime = Date.now();\r\n\r\n // Map sessionId -> channel WebSocket\r\n private channelSockets = new Map<string, WebSocket>();\r\n // Track which channel connections are local\r\n private localChannels = new Set<string>();\r\n // All connected dashboard WebSockets\r\n private dashboardSockets = new Set<WebSocket>();\r\n\r\n private telegramBot?: TelegramBot;\r\n private heartbeatInterval?: ReturnType<typeof setInterval>;\r\n private channelAlive = new Map<string, boolean>(); // sessionId -> alive flag\r\n\r\n private host: string;\r\n private port: number;\r\n private token?: string;\r\n\r\n constructor(config?: Partial<AppConfig>) {\r\n this.host = config?.hub?.host ?? DEFAULT_HUB_HOST;\r\n this.port = config?.hub?.port ?? DEFAULT_HUB_PORT;\r\n this.token = config?.hub?.token;\r\n\r\n if (config?.notifications) {\r\n this.notifier.configure({\r\n desktop: config.notifications.desktop,\r\n });\r\n }\r\n if (config?.webhooks) {\r\n this.notifier.configure({ webhooks: config.webhooks });\r\n }\r\n const displayHost = this.host === '0.0.0.0' ? '127.0.0.1' : this.host;\r\n this.notifier.configure({ dashboardUrl: `http://${displayHost}:${this.port}` });\r\n\r\n // Initialize Telegram bot if configured\r\n const fullConfig = loadConfig();\r\n if (fullConfig.telegram?.enabled && fullConfig.telegram.botToken && fullConfig.telegram.chatId) {\r\n this.initTelegram(fullConfig.telegram);\r\n }\r\n\r\n // HTTP Server\r\n this.httpServer = http.createServer((req, res) => this.handleHttp(req, res));\r\n\r\n // WebSocket for channel servers\r\n this.wssChannel = new WebSocketServer({ noServer: true });\r\n this.wssChannel.on('connection', (ws: WebSocket, req: http.IncomingMessage) => this.handleChannelConnection(ws, req));\r\n\r\n // WebSocket for dashboard\r\n this.wssDashboard = new WebSocketServer({ noServer: true });\r\n this.wssDashboard.on('connection', (ws) => this.handleDashboardConnection(ws));\r\n\r\n // Route WebSocket upgrade requests\r\n this.httpServer.on('upgrade', (req, socket, head) => {\r\n const url = new URL(req.url!, `http://${req.headers.host}`);\r\n const pathname = url.pathname;\r\n\r\n // Token auth for WebSocket connections (skip for local requests)\r\n if (this.token && !this.isLocalRequest(req)) {\r\n const wsToken = url.searchParams.get('token');\r\n if (wsToken !== this.token) {\r\n socket.destroy();\r\n return;\r\n }\r\n }\r\n\r\n if (pathname === WS_PATH_CHANNEL) {\r\n this.wssChannel.handleUpgrade(req, socket, head, (ws) => {\r\n this.wssChannel.emit('connection', ws, req);\r\n });\r\n } else if (pathname === WS_PATH_DASHBOARD) {\r\n this.wssDashboard.handleUpgrade(req, socket, head, (ws) => {\r\n this.wssDashboard.emit('connection', ws, req);\r\n });\r\n } else {\r\n socket.destroy();\r\n }\r\n });\r\n }\r\n\r\n async start(): Promise<void> {\r\n this.cleanupUploads();\r\n this.startHeartbeat();\r\n return new Promise((resolve, reject) => {\r\n this.httpServer.on('error', reject);\r\n this.httpServer.listen(this.port, this.host, () => {\r\n const displayHost = this.host === '0.0.0.0' ? '127.0.0.1' : this.host;\r\n logger.info(`Hub server listening on http://${displayHost}:${this.port}`);\r\n resolve();\r\n });\r\n });\r\n }\r\n\r\n stop(): Promise<void> {\r\n return new Promise((resolve) => {\r\n // Stop heartbeat\r\n if (this.heartbeatInterval) clearInterval(this.heartbeatInterval);\r\n\r\n // Stop telegram bot\r\n if (this.telegramBot) this.telegramBot.stopPolling();\r\n\r\n // Force-close all WebSocket connections\r\n for (const ws of this.channelSockets.values()) ws.terminate();\r\n for (const ws of this.dashboardSockets) ws.terminate();\r\n this.channelSockets.clear();\r\n this.dashboardSockets.clear();\r\n\r\n this.wssChannel.close();\r\n this.wssDashboard.close();\r\n this.httpServer.close(() => {\r\n logger.info('Hub server stopped');\r\n resolve();\r\n });\r\n\r\n // Force resolve after 3 seconds if server won't close\r\n setTimeout(() => {\r\n logger.warn('Force shutting down');\r\n resolve();\r\n }, 3000);\r\n });\r\n }\r\n\r\n // --- HTTP Handler ---\r\n\r\n private handleHttp(req: http.IncomingMessage, res: http.ServerResponse): void {\r\n const url = new URL(req.url!, `http://${req.headers.host}`);\r\n\r\n // CORS headers - restrict to same origin\r\n const origin = req.headers.origin;\r\n if (origin && (origin.includes('127.0.0.1') || origin.includes('localhost'))) {\r\n res.setHeader('Access-Control-Allow-Origin', origin);\r\n }\r\n res.setHeader('Access-Control-Allow-Methods', 'GET, POST, DELETE, OPTIONS');\r\n res.setHeader('Access-Control-Allow-Headers', 'Content-Type, Authorization');\r\n\r\n if (req.method === 'OPTIONS') {\r\n res.writeHead(204);\r\n res.end();\r\n return;\r\n }\r\n\r\n // Token auth for API endpoints (skip dashboard HTML serving)\r\n if (url.pathname !== '/' && this.token) {\r\n if (!this.isLocalRequest(req)) {\r\n const authHeader = req.headers['authorization'];\r\n const bearerToken = authHeader?.startsWith('Bearer ') ? authHeader.slice(7) : null;\r\n if (bearerToken !== this.token) {\r\n this.jsonResponse(res, 401, { error: 'Unauthorized' });\r\n return;\r\n }\r\n }\r\n }\r\n\r\n // Route\r\n if (url.pathname === '/' && req.method === 'GET') {\r\n this.serveDashboard(res);\r\n } else if (url.pathname === '/api/sessions' && req.method === 'GET') {\r\n this.jsonResponse(res, 200, { sessions: this.sessions.getAll() });\r\n } else if (url.pathname === '/api/status' && req.method === 'GET') {\r\n this.jsonResponse(res, 200, {\r\n running: true,\r\n pid: process.pid,\r\n port: this.port,\r\n sessions: this.sessions.count(),\r\n uptime: Date.now() - this.startTime,\r\n });\r\n } else if (url.pathname.startsWith('/api/sessions/') && req.method === 'DELETE') {\r\n const sessionId = url.pathname.slice('/api/sessions/'.length);\r\n const ws = this.channelSockets.get(sessionId);\r\n if (ws) { ws.terminate(); }\r\n const session = this.sessions.unregister(sessionId);\r\n this.channelSockets.delete(sessionId);\r\n this.localChannels.delete(sessionId);\r\n this.channelAlive.delete(sessionId);\r\n if (session) {\r\n this.broadcastToDashboards({ type: 'session_disconnected', sessionId });\r\n this.jsonResponse(res, 200, { ok: true });\r\n } else {\r\n this.jsonResponse(res, 404, { error: 'Session not found' });\r\n }\r\n } else if (url.pathname === '/api/send' && req.method === 'POST') {\r\n this.handleApiSend(req, res);\r\n } else if (url.pathname === '/api/notify' && req.method === 'POST') {\r\n this.handleApiNotify(req, res);\r\n } else if (url.pathname === '/api/webhooks' && req.method === 'GET') {\r\n const config = loadConfig();\r\n this.jsonResponse(res, 200, { webhooks: config.webhooks || [] });\r\n } else if (url.pathname === '/api/webhooks' && req.method === 'POST') {\r\n this.handleWebhookSave(req, res);\r\n } else if (url.pathname === '/api/telegram' && req.method === 'GET') {\r\n const cfg = loadConfig();\r\n const tg = cfg.telegram ?? { botToken: '', chatId: '', enabled: false };\r\n // Mask bot token for security\r\n this.jsonResponse(res, 200, {\r\n telegram: { ...tg, botToken: tg.botToken ? `${tg.botToken.slice(0, 8)}...` : '' },\r\n });\r\n } else if (url.pathname === '/api/telegram' && req.method === 'POST') {\r\n this.handleTelegramSave(req, res);\r\n } else if (url.pathname === '/api/telegram/test' && req.method === 'POST') {\r\n this.handleTelegramTest(req, res);\r\n } else if (url.pathname === '/api/telegram/detect' && req.method === 'POST') {\r\n this.handleTelegramDetect(req, res);\r\n } else {\r\n this.jsonResponse(res, 404, { error: 'Not found' });\r\n }\r\n }\r\n\r\n private serveDashboard(res: http.ServerResponse): void {\r\n // Look for dashboard HTML relative to this file (dist) or source\r\n const candidates = [\r\n path.join(__dirname, '..', 'dashboard', 'index.html'), // from dist/hub/\r\n path.join(__dirname, 'dashboard', 'index.html'), // from dist/ (bundled index.js)\r\n path.join(__dirname, '..', '..', 'src', 'dashboard', 'index.html'), // from dist/hub/ -> src/\r\n path.join(__dirname, '..', 'src', 'dashboard', 'index.html'), // from dist/ -> src/\r\n path.join(process.cwd(), 'dist', 'dashboard', 'index.html'), // from cwd\r\n path.join(process.cwd(), 'src', 'dashboard', 'index.html'), // from cwd/src\r\n ];\r\n logger.debug(`Dashboard candidates: ${JSON.stringify(candidates)}`);\r\n\r\n for (const candidate of candidates) {\r\n if (fs.existsSync(candidate)) {\r\n const html = fs.readFileSync(candidate, 'utf-8');\r\n res.writeHead(200, { 'Content-Type': 'text/html; charset=utf-8' });\r\n res.end(html);\r\n return;\r\n }\r\n }\r\n\r\n res.writeHead(200, { 'Content-Type': 'text/html; charset=utf-8' });\r\n res.end('<html><body><h1>claude-alarm</h1><p>Dashboard HTML not found. Reinstall the package.</p></body></html>');\r\n }\r\n\r\n private async handleApiSend(req: http.IncomingMessage, res: http.ServerResponse): Promise<void> {\r\n const body = await this.readBody(req);\r\n if (!body) { this.jsonResponse(res, 400, { error: 'Invalid JSON' }); return; }\r\n\r\n const { sessionId, content } = body as { sessionId?: string; content?: string };\r\n if (!sessionId || !content) {\r\n this.jsonResponse(res, 400, { error: 'sessionId and content are required' });\r\n return;\r\n }\r\n\r\n const ws = this.channelSockets.get(sessionId);\r\n if (!ws || ws.readyState !== WebSocket.OPEN) {\r\n this.jsonResponse(res, 404, { error: 'Session not connected' });\r\n return;\r\n }\r\n\r\n const msg: ChannelMessage = { type: 'message_to_session', sessionId, content };\r\n ws.send(JSON.stringify(msg));\r\n this.jsonResponse(res, 200, { ok: true });\r\n }\r\n\r\n private async handleApiNotify(req: http.IncomingMessage, res: http.ServerResponse): Promise<void> {\r\n const body = await this.readBody(req);\r\n if (!body) { this.jsonResponse(res, 400, { error: 'Invalid JSON' }); return; }\r\n\r\n const { title, message, level } = body as { title?: string; message?: string; level?: string };\r\n if (!title || !message) {\r\n this.jsonResponse(res, 400, { error: 'title and message are required' });\r\n return;\r\n }\r\n\r\n await this.notifier.notify(title, message, (level as any) ?? 'info');\r\n this.jsonResponse(res, 200, { ok: true });\r\n }\r\n\r\n // --- Channel WebSocket ---\r\n\r\n private handleChannelConnection(ws: WebSocket, req: http.IncomingMessage): void {\r\n const isLocal = this.isLocalRequest(req);\r\n logger.info(`Channel server connected (local: ${isLocal})`);\r\n\r\n // Track pong responses for heartbeat\r\n ws.on('pong', () => {\r\n for (const [sessionId, sock] of this.channelSockets) {\r\n if (sock === ws) { this.channelAlive.set(sessionId, true); break; }\r\n }\r\n });\r\n\r\n ws.on('message', (data) => {\r\n try {\r\n const msg = JSON.parse(data.toString()) as ChannelMessage;\r\n this.handleChannelMessage(ws, isLocal, msg);\r\n } catch {\r\n logger.warn('Invalid message from channel');\r\n }\r\n });\r\n\r\n ws.on('close', () => {\r\n // Find and remove the session for this socket\r\n for (const [sessionId, sock] of this.channelSockets) {\r\n if (sock === ws) {\r\n const session = this.sessions.unregister(sessionId);\r\n this.channelSockets.delete(sessionId);\r\n this.localChannels.delete(sessionId);\r\n this.channelAlive.delete(sessionId);\r\n logger.info(`Channel disconnected: ${sessionId}`);\r\n this.broadcastToDashboards({\r\n type: 'session_disconnected',\r\n sessionId,\r\n });\r\n break;\r\n }\r\n }\r\n });\r\n }\r\n\r\n private handleChannelMessage(ws: WebSocket, isLocal: boolean, msg: ChannelMessage): void {\r\n switch (msg.type) {\r\n case 'register': {\r\n const session = msg.session;\r\n session.isLocal = isLocal;\r\n const isReregister = !!this.sessions.get(session.id);\r\n this.sessions.register(session);\r\n this.channelSockets.set(session.id, ws);\r\n if (isLocal) this.localChannels.add(session.id);\r\n logger.info(`Session registered: ${session.id} (${session.name}, channel: ${session.channelEnabled ?? false})`);\r\n this.broadcastToDashboards({\r\n type: isReregister ? 'session_updated' : 'session_connected',\r\n session,\r\n });\r\n break;\r\n }\r\n\r\n case 'status': {\r\n const updated = this.sessions.updateStatus(msg.sessionId, msg.status);\r\n if (updated) {\r\n this.broadcastToDashboards({ type: 'session_updated', session: updated });\r\n }\r\n break;\r\n }\r\n\r\n case 'notify': {\r\n this.sessions.updateActivity(msg.sessionId);\r\n const notifySession = this.sessions.get(msg.sessionId);\r\n const notifyLabel = this.getSessionLabel(notifySession);\r\n this.notifier.notifyWithSession(msg.sessionId, notifyLabel, `[${notifyLabel}] ${msg.title}`, msg.message, msg.level ?? 'info');\r\n this.broadcastToDashboards({\r\n type: 'notification',\r\n sessionId: msg.sessionId,\r\n title: msg.title,\r\n message: msg.message,\r\n level: msg.level,\r\n timestamp: Date.now(),\r\n });\r\n break;\r\n }\r\n\r\n case 'reply': {\r\n this.sessions.updateActivity(msg.sessionId);\r\n const replySession = this.sessions.get(msg.sessionId);\r\n const replyLabel = this.getSessionLabel(replySession);\r\n this.notifier.notifyWithSession(msg.sessionId, replyLabel, `[${replyLabel}] Reply`, msg.content.slice(0, 3000), 'info');\r\n this.broadcastToDashboards({\r\n type: 'reply_from_session',\r\n sessionId: msg.sessionId,\r\n content: msg.content,\r\n timestamp: Date.now(),\r\n });\r\n break;\r\n }\r\n\r\n case 'permission_request': {\r\n this.sessions.updateActivity(msg.sessionId);\r\n logger.info(`Permission request [${msg.requestId}] from ${msg.sessionId}: ${msg.toolName}`);\r\n this.broadcastToDashboards({\r\n type: 'permission_request',\r\n sessionId: msg.sessionId,\r\n requestId: msg.requestId,\r\n toolName: msg.toolName,\r\n description: msg.description,\r\n inputPreview: msg.inputPreview,\r\n timestamp: msg.timestamp,\r\n });\r\n // Forward to Telegram\r\n if (this.telegramBot) {\r\n const session = this.sessions.get(msg.sessionId);\r\n const label = this.getSessionLabel(session);\r\n this.telegramBot.sendPermissionRequest(msg.sessionId, label, msg.requestId, msg.toolName, msg.description, msg.inputPreview);\r\n }\r\n break;\r\n }\r\n }\r\n }\r\n\r\n // --- Dashboard WebSocket ---\r\n\r\n private handleDashboardConnection(ws: WebSocket): void {\r\n this.dashboardSockets.add(ws);\r\n logger.info(`Dashboard connected (total: ${this.dashboardSockets.size})`);\r\n\r\n // Send current session list\r\n const sessionsMsg: ChannelMessage = {\r\n type: 'sessions_list',\r\n sessions: this.sessions.getAll(),\r\n };\r\n ws.send(JSON.stringify(sessionsMsg));\r\n\r\n ws.on('message', (data) => {\r\n try {\r\n const msg = JSON.parse(data.toString()) as ChannelMessage;\r\n if (msg.type === 'message_to_session') {\r\n const channelWs = this.channelSockets.get(msg.sessionId);\r\n if (channelWs?.readyState === WebSocket.OPEN) {\r\n channelWs.send(JSON.stringify(msg));\r\n }\r\n } else if (msg.type === 'image_upload') {\r\n this.handleImageUpload(msg);\r\n } else if (msg.type === 'permission_response') {\r\n const channelWs = this.channelSockets.get(msg.sessionId);\r\n if (channelWs?.readyState === WebSocket.OPEN) {\r\n channelWs.send(JSON.stringify(msg));\r\n logger.info(`Permission verdict [${msg.requestId}]: ${msg.behavior} -> session ${msg.sessionId}`);\r\n }\r\n }\r\n } catch {\r\n logger.warn('Invalid message from dashboard');\r\n }\r\n });\r\n\r\n ws.on('close', () => {\r\n this.dashboardSockets.delete(ws);\r\n logger.info(`Dashboard disconnected (total: ${this.dashboardSockets.size})`);\r\n });\r\n }\r\n\r\n // --- Helpers ---\r\n\r\n private broadcastToDashboards(msg: ChannelMessage): void {\r\n const payload = JSON.stringify(msg);\r\n for (const ws of this.dashboardSockets) {\r\n if (ws.readyState === WebSocket.OPEN) {\r\n ws.send(payload);\r\n }\r\n }\r\n }\r\n\r\n private handleImageUpload(msg: ChannelMessage & { type: 'image_upload' }): void {\r\n const { sessionId, imageData, mimeType, originalName, content } = msg;\r\n\r\n // Only allow for local sessions\r\n if (!this.localChannels.has(sessionId)) {\r\n logger.warn(`Image upload rejected: session ${sessionId} is not local`);\r\n return;\r\n }\r\n\r\n const channelWs = this.channelSockets.get(sessionId);\r\n if (!channelWs || channelWs.readyState !== WebSocket.OPEN) return;\r\n\r\n // Validate mime type\r\n const allowedTypes = ['image/png', 'image/jpeg', 'image/gif', 'image/webp'];\r\n if (!allowedTypes.includes(mimeType)) return;\r\n\r\n // Extract base64 data (remove data URL prefix if present)\r\n const base64Data = imageData.replace(/^data:image\\/\\w+;base64,/, '');\r\n const buffer = Buffer.from(base64Data, 'base64');\r\n\r\n // Validate size (10MB max)\r\n if (buffer.length > 10 * 1024 * 1024) {\r\n logger.warn('Image upload rejected: exceeds 10MB');\r\n return;\r\n }\r\n\r\n // Save to uploads dir\r\n fs.mkdirSync(UPLOADS_DIR, { recursive: true });\r\n const ext = mimeType.split('/')[1] === 'jpeg' ? 'jpg' : mimeType.split('/')[1];\r\n const filename = `${randomUUID()}.${ext}`;\r\n const filePath = path.join(UPLOADS_DIR, filename);\r\n fs.writeFileSync(filePath, buffer);\r\n\r\n // Forward file path to channel\r\n const forwardMsg: ChannelMessage = {\r\n type: 'image_to_session',\r\n sessionId,\r\n imagePath: filePath,\r\n mimeType,\r\n originalName,\r\n content,\r\n };\r\n channelWs.send(JSON.stringify(forwardMsg));\r\n logger.info(`Image saved and forwarded: ${filename} (${buffer.length} bytes)`);\r\n\r\n // Cleanup after 5 minutes\r\n setTimeout(() => {\r\n try { fs.unlinkSync(filePath); } catch {}\r\n }, 5 * 60 * 1000);\r\n }\r\n\r\n private async handleWebhookSave(req: http.IncomingMessage, res: http.ServerResponse): Promise<void> {\r\n const body = await this.readBody(req);\r\n if (!body) { this.jsonResponse(res, 400, { error: 'Invalid JSON' }); return; }\r\n const { webhooks } = body as { webhooks?: WebhookConfig[] };\r\n if (!Array.isArray(webhooks)) { this.jsonResponse(res, 400, { error: 'webhooks must be an array' }); return; }\r\n const config = loadConfig();\r\n config.webhooks = webhooks;\r\n saveConfig(config);\r\n this.notifier.configure({ webhooks });\r\n this.jsonResponse(res, 200, { ok: true });\r\n }\r\n\r\n private initTelegram(config: TelegramConfig): void {\r\n this.telegramBot = new TelegramBot(config);\r\n this.telegramBot.getSessions = () => this.sessions.getAll();\r\n this.telegramBot.onMessageToSession = (sessionId, content) => {\r\n const channelWs = this.channelSockets.get(sessionId);\r\n if (channelWs?.readyState === WebSocket.OPEN) {\r\n const msg: ChannelMessage = { type: 'message_to_session', sessionId, content };\r\n channelWs.send(JSON.stringify(msg));\r\n logger.info(`Telegram message forwarded to session: ${sessionId}`);\r\n }\r\n };\r\n this.telegramBot.onImageToSession = (sessionId, imagePath, mimeType, caption) => {\r\n const channelWs = this.channelSockets.get(sessionId);\r\n if (channelWs?.readyState === WebSocket.OPEN) {\r\n const msg: ChannelMessage = { type: 'image_to_session', sessionId, imagePath, mimeType, content: caption };\r\n channelWs.send(JSON.stringify(msg));\r\n logger.info(`Telegram photo forwarded to session: ${sessionId}`);\r\n }\r\n };\r\n this.telegramBot.onPermissionVerdict = (sessionId, requestId, behavior) => {\r\n const channelWs = this.channelSockets.get(sessionId);\r\n if (channelWs?.readyState === WebSocket.OPEN) {\r\n const msg: ChannelMessage = { type: 'permission_response', sessionId, requestId, behavior };\r\n channelWs.send(JSON.stringify(msg));\r\n logger.info(`Telegram permission verdict [${requestId}]: ${behavior} -> session ${sessionId}`);\r\n }\r\n // Also notify dashboards so they can dismiss the permission bar\r\n this.broadcastToDashboards({ type: 'permission_response', sessionId, requestId, behavior });\r\n };\r\n this.notifier.configure({ telegramBot: this.telegramBot });\r\n this.telegramBot.startPolling();\r\n logger.info('Telegram bot initialized');\r\n }\r\n\r\n private async handleTelegramSave(req: http.IncomingMessage, res: http.ServerResponse): Promise<void> {\r\n const body = await this.readBody(req);\r\n if (!body) { this.jsonResponse(res, 400, { error: 'Invalid JSON' }); return; }\r\n const { telegram } = body as { telegram?: TelegramConfig };\r\n if (!telegram) { this.jsonResponse(res, 400, { error: 'telegram config required' }); return; }\r\n\r\n const config = loadConfig();\r\n // If botToken is masked (contains '...'), keep the existing token\r\n if (telegram.botToken.includes('...') && config.telegram?.botToken) {\r\n telegram.botToken = config.telegram.botToken;\r\n }\r\n config.telegram = telegram;\r\n saveConfig(config);\r\n\r\n // Stop existing bot if running\r\n if (this.telegramBot) {\r\n this.telegramBot.stopPolling();\r\n this.telegramBot = undefined;\r\n this.notifier.configure({ telegramBot: undefined as any });\r\n }\r\n\r\n // Start new bot if enabled\r\n if (telegram.enabled && telegram.botToken && telegram.chatId) {\r\n this.initTelegram(telegram);\r\n }\r\n\r\n this.jsonResponse(res, 200, { ok: true });\r\n }\r\n\r\n private async handleTelegramTest(req: http.IncomingMessage, res: http.ServerResponse): Promise<void> {\r\n const body = await this.readBody(req);\r\n if (!body) { this.jsonResponse(res, 400, { error: 'Invalid JSON' }); return; }\r\n const { botToken, chatId } = body as { botToken?: string; chatId?: string };\r\n if (!botToken || !chatId) {\r\n this.jsonResponse(res, 400, { error: 'botToken and chatId required' });\r\n return;\r\n }\r\n\r\n try {\r\n const testRes = await fetch(`https://api.telegram.org/bot${botToken}/sendMessage`, {\r\n method: 'POST',\r\n headers: { 'Content-Type': 'application/json' },\r\n body: JSON.stringify({\r\n chat_id: chatId,\r\n text: 'Claude Alarm test message! Connection successful.',\r\n }),\r\n });\r\n\r\n if (testRes.ok) {\r\n this.jsonResponse(res, 200, { ok: true });\r\n } else {\r\n const err = await testRes.json() as { description?: string };\r\n this.jsonResponse(res, 400, { error: (err as any).description || 'Telegram API error' });\r\n }\r\n } catch (err) {\r\n this.jsonResponse(res, 500, { error: (err as Error).message });\r\n }\r\n }\r\n\r\n private async handleTelegramDetect(req: http.IncomingMessage, res: http.ServerResponse): Promise<void> {\r\n const body = await this.readBody(req);\r\n if (!body) { this.jsonResponse(res, 400, { error: 'Invalid JSON' }); return; }\r\n const { botToken } = body as { botToken?: string };\r\n if (!botToken) {\r\n this.jsonResponse(res, 400, { error: 'botToken required' });\r\n return;\r\n }\r\n\r\n try {\r\n const detectRes = await fetch(`https://api.telegram.org/bot${botToken}/getUpdates?timeout=0&limit=10`, {\r\n signal: AbortSignal.timeout(10000),\r\n });\r\n\r\n if (!detectRes.ok) {\r\n const err = await detectRes.json() as { description?: string };\r\n this.jsonResponse(res, 400, { error: (err as any).description || 'Invalid bot token' });\r\n return;\r\n }\r\n\r\n const data = await detectRes.json() as { ok: boolean; result: Array<{ message?: { chat: { id: number; first_name?: string; title?: string; type: string } } }> };\r\n if (!data.ok || !data.result.length) {\r\n this.jsonResponse(res, 200, { ok: false, chats: [] });\r\n return;\r\n }\r\n\r\n // Extract unique chats\r\n const chatMap = new Map<string, { id: string; name: string; type: string }>();\r\n for (const update of data.result) {\r\n if (update.message?.chat) {\r\n const chat = update.message.chat;\r\n const id = String(chat.id);\r\n if (!chatMap.has(id)) {\r\n chatMap.set(id, {\r\n id,\r\n name: chat.title || chat.first_name || id,\r\n type: chat.type,\r\n });\r\n }\r\n }\r\n }\r\n\r\n this.jsonResponse(res, 200, { ok: true, chats: [...chatMap.values()] });\r\n } catch (err) {\r\n this.jsonResponse(res, 500, { error: (err as Error).message });\r\n }\r\n }\r\n\r\n private startHeartbeat(): void {\r\n // Ping channel WebSockets every 30s, terminate unresponsive ones\r\n this.heartbeatInterval = setInterval(() => {\r\n for (const [sessionId, ws] of this.channelSockets) {\r\n if (this.channelAlive.get(sessionId) === false) {\r\n // No pong received since last ping — terminate\r\n logger.info(`Heartbeat timeout, terminating session: ${sessionId}`);\r\n ws.terminate();\r\n continue;\r\n }\r\n this.channelAlive.set(sessionId, false);\r\n ws.ping();\r\n }\r\n }, 30000);\r\n }\r\n\r\n private cleanupUploads(): void {\r\n try {\r\n if (!fs.existsSync(UPLOADS_DIR)) return;\r\n const files = fs.readdirSync(UPLOADS_DIR);\r\n for (const file of files) {\r\n try { fs.unlinkSync(path.join(UPLOADS_DIR, file)); } catch {}\r\n }\r\n if (files.length > 0) logger.info(`Cleaned up ${files.length} leftover upload(s)`);\r\n } catch {}\r\n }\r\n\r\n private getSessionLabel(session?: SessionInfo): string {\r\n if (!session) return 'unknown';\r\n return session.displayName || session.cwd?.replace(/^.*[/\\\\]/, '') || session.name;\r\n }\r\n\r\n private jsonResponse(res: http.ServerResponse, status: number, body: unknown): void {\r\n res.writeHead(status, { 'Content-Type': 'application/json' });\r\n res.end(JSON.stringify(body));\r\n }\r\n\r\n private isLocalRequest(req: http.IncomingMessage): boolean {\r\n const addr = req.socket.remoteAddress;\r\n return addr === '127.0.0.1' || addr === '::1' || addr === '::ffff:127.0.0.1';\r\n }\r\n\r\n private readBody(req: http.IncomingMessage, maxSize = 1024 * 1024): Promise<unknown | null> {\r\n return new Promise((resolve) => {\r\n let data = '';\r\n let size = 0;\r\n req.on('data', (chunk) => {\r\n size += chunk.length;\r\n if (size > maxSize) {\r\n req.destroy();\r\n resolve(null);\r\n return;\r\n }\r\n data += chunk;\r\n });\r\n req.on('end', () => {\r\n try {\r\n resolve(JSON.parse(data));\r\n } catch {\r\n resolve(null);\r\n }\r\n });\r\n });\r\n }\r\n}\r\n\r\n// Direct execution support\r\nif (process.argv[1] && (\r\n process.argv[1].endsWith('hub/server.js') ||\r\n process.argv[1].endsWith('hub/server.ts')\r\n)) {\r\n const config = loadConfig();\r\n const hub = new HubServer(config);\r\n hub.start().catch((err) => {\r\n logger.error('Failed to start hub:', err);\r\n process.exit(1);\r\n });\r\n\r\n const shutdown = () => {\r\n hub.stop().then(() => process.exit(0));\r\n };\r\n process.on('SIGINT', shutdown);\r\n process.on('SIGTERM', shutdown);\r\n}\r\n","/**\r\n * Logger that writes to stderr only.\r\n * CRITICAL: In MCP channel servers, stdout is used for the stdio protocol.\r\n * Any console.log() would corrupt the protocol. Always use this logger.\r\n */\r\nexport const logger = {\r\n info(msg: string, ...args: unknown[]) {\r\n console.error(`[claude-alarm] ${msg}`, ...args);\r\n },\r\n warn(msg: string, ...args: unknown[]) {\r\n console.error(`[claude-alarm WARN] ${msg}`, ...args);\r\n },\r\n error(msg: string, ...args: unknown[]) {\r\n console.error(`[claude-alarm ERROR] ${msg}`, ...args);\r\n },\r\n debug(msg: string, ...args: unknown[]) {\r\n if (process.env.CLAUDE_ALARM_DEBUG) {\r\n console.error(`[claude-alarm DEBUG] ${msg}`, ...args);\r\n }\r\n },\r\n};\r\n","import path from 'node:path';\r\nimport os from 'node:os';\r\n\r\nexport const DEFAULT_HUB_HOST = '127.0.0.1';\r\nexport const DEFAULT_HUB_PORT = 7900;\r\n\r\nexport const CONFIG_DIR = path.join(os.homedir(), '.claude-alarm');\r\nexport const CONFIG_FILE = path.join(CONFIG_DIR, 'config.json');\r\nexport const PID_FILE = path.join(CONFIG_DIR, 'hub.pid');\r\nexport const LOG_FILE = path.join(CONFIG_DIR, 'hub.log');\r\nexport const UPLOADS_DIR = path.join(CONFIG_DIR, 'uploads');\r\n\r\nexport const WS_PATH_CHANNEL = '/ws/channel';\r\nexport const WS_PATH_DASHBOARD = '/ws/dashboard';\r\n\r\nexport const CHANNEL_SERVER_NAME = 'claude-alarm';\r\nexport const CHANNEL_SERVER_VERSION = '0.1.0';\r\n","import type { SessionInfo, SessionStatus } from '../shared/types.js';\r\n\r\nexport class SessionManager {\r\n private sessions = new Map<string, SessionInfo>();\r\n\r\n register(session: SessionInfo): void {\r\n // Auto-number duplicate names\r\n const baseName = session.cwd?.replace(/^.*[/\\\\]/, '') || session.name;\r\n const existing = Array.from(this.sessions.values()).filter(\r\n s => s.id !== session.id && (s.cwd?.replace(/^.*[/\\\\]/, '') || s.name) === baseName,\r\n );\r\n if (existing.length > 0) {\r\n // Number this one\r\n const usedNums = existing.map(s => {\r\n const m = s.displayName?.match(/\\((\\d+)\\)$/);\r\n return m ? parseInt(m[1], 10) : 1;\r\n });\r\n // Ensure first existing session has (1) if it doesn't yet\r\n for (const s of existing) {\r\n if (!s.displayName?.match(/\\(\\d+\\)$/)) {\r\n s.displayName = `${baseName} (1)`;\r\n }\r\n }\r\n const nextNum = Math.max(...usedNums, 1) + 1;\r\n session.displayName = `${baseName} (${nextNum})`;\r\n } else {\r\n session.displayName = baseName;\r\n }\r\n this.sessions.set(session.id, { ...session });\r\n }\r\n\r\n unregister(sessionId: string): SessionInfo | undefined {\r\n const session = this.sessions.get(sessionId);\r\n this.sessions.delete(sessionId);\r\n return session;\r\n }\r\n\r\n updateStatus(sessionId: string, status: SessionStatus): SessionInfo | undefined {\r\n const session = this.sessions.get(sessionId);\r\n if (session) {\r\n session.status = status;\r\n session.lastActivity = Date.now();\r\n }\r\n return session;\r\n }\r\n\r\n updateActivity(sessionId: string): void {\r\n const session = this.sessions.get(sessionId);\r\n if (session) {\r\n session.lastActivity = Date.now();\r\n }\r\n }\r\n\r\n get(sessionId: string): SessionInfo | undefined {\r\n return this.sessions.get(sessionId);\r\n }\r\n\r\n getAll(): SessionInfo[] {\r\n return Array.from(this.sessions.values());\r\n }\r\n\r\n count(): number {\r\n return this.sessions.size;\r\n }\r\n}\r\n","import notifier from 'node-notifier';\r\nimport { execFile } from 'node:child_process';\r\nimport { logger } from '../shared/logger.js';\r\nimport type { NotifyLevel, WebhookConfig } from '../shared/types.js';\r\nimport type { TelegramBot } from './telegram.js';\r\n\r\nexport class Notifier {\r\n private webhooks: WebhookConfig[] = [];\r\n private desktopEnabled = true;\r\n private notificationSettingsOpened = false;\r\n private dashboardUrl?: string;\r\n private telegramBot?: TelegramBot;\r\n\r\n configure(options: { desktop?: boolean; webhooks?: WebhookConfig[]; dashboardUrl?: string; telegramBot?: TelegramBot }): void {\r\n if (options.dashboardUrl) this.dashboardUrl = options.dashboardUrl;\r\n if (options.desktop !== undefined) this.desktopEnabled = options.desktop;\r\n if (options.webhooks) this.webhooks = options.webhooks;\r\n if (options.telegramBot) this.telegramBot = options.telegramBot;\r\n }\r\n\r\n async notify(title: string, message: string, level: NotifyLevel = 'info'): Promise<void> {\r\n await this.notifyWithSession(undefined, undefined, title, message, level);\r\n }\r\n\r\n async notifyWithSession(sessionId: string | undefined, sessionLabel: string | undefined, title: string, message: string, level: NotifyLevel = 'info'): Promise<void> {\r\n const promises: Promise<void>[] = [];\r\n\r\n if (this.desktopEnabled) {\r\n promises.push(this.sendDesktop(title, message, level));\r\n }\r\n\r\n for (const webhook of this.webhooks) {\r\n promises.push(this.sendWebhook(webhook, title, message, level));\r\n }\r\n\r\n if (this.telegramBot && sessionId && sessionLabel) {\r\n promises.push(this.telegramBot.sendNotification(sessionId, sessionLabel, title, message));\r\n }\r\n\r\n await Promise.allSettled(promises);\r\n }\r\n\r\n private async sendDesktop(title: string, message: string, _level: NotifyLevel): Promise<void> {\r\n if (process.platform === 'win32') {\r\n // Check if notifications are enabled by running snoretoast directly\r\n const enabled = await this.checkWindowsNotifications();\r\n if (!enabled) {\r\n this.openNotificationSettings();\r\n return;\r\n }\r\n }\r\n\r\n return new Promise((resolve) => {\r\n const notification = (notifier as any).notify(\r\n {\r\n title: `Claude Alarm: ${title}`,\r\n message,\r\n sound: true,\r\n wait: true,\r\n },\r\n (err: Error | null) => {\r\n if (err) {\r\n logger.warn(`Desktop notification failed: ${err.message}`);\r\n }\r\n resolve();\r\n },\r\n );\r\n\r\n // No click handler - dashboard is already open and notifications\r\n // can be clicked there to focus the relevant session\r\n });\r\n }\r\n\r\n private checkWindowsNotifications(): Promise<boolean> {\r\n return new Promise((resolve) => {\r\n execFile(\r\n 'powershell',\r\n ['-Command', '(Get-ItemProperty -Path \"HKCU:\\\\Software\\\\Microsoft\\\\Windows\\\\CurrentVersion\\\\PushNotifications\" -Name ToastEnabled -ErrorAction SilentlyContinue).ToastEnabled'],\r\n (err, stdout) => {\r\n if (err) { resolve(true); return; } // assume enabled on error\r\n const value = stdout.trim();\r\n resolve(value !== '0');\r\n },\r\n );\r\n });\r\n }\r\n\r\n private openNotificationSettings(): void {\r\n if (this.notificationSettingsOpened) return;\r\n this.notificationSettingsOpened = true;\r\n\r\n logger.warn('Windows notifications are disabled. Opening notification settings...');\r\n logger.warn('Please enable notifications for this app, then try again.');\r\n\r\n if (process.platform === 'win32') {\r\n execFile('powershell', ['-Command', 'Start-Process ms-settings:notifications']);\r\n }\r\n\r\n // Allow re-opening after 5 minutes\r\n setTimeout(() => { this.notificationSettingsOpened = false; }, 5 * 60 * 1000);\r\n }\r\n\r\n private async sendWebhook(\r\n webhook: WebhookConfig,\r\n title: string,\r\n message: string,\r\n level: NotifyLevel,\r\n ): Promise<void> {\r\n try {\r\n const response = await fetch(webhook.url, {\r\n method: 'POST',\r\n headers: {\r\n 'Content-Type': 'application/json',\r\n ...webhook.headers,\r\n },\r\n body: JSON.stringify({\r\n title,\r\n message,\r\n level,\r\n timestamp: Date.now(),\r\n source: 'claude-alarm',\r\n }),\r\n });\r\n\r\n if (!response.ok) {\r\n logger.warn(`Webhook ${webhook.url} returned ${response.status}`);\r\n }\r\n } catch (err) {\r\n logger.warn(`Webhook ${webhook.url} failed: ${(err as Error).message}`);\r\n }\r\n }\r\n}\r\n","import fs from 'node:fs';\r\nimport path from 'node:path';\r\nimport { randomUUID } from 'node:crypto';\r\nimport { logger } from '../shared/logger.js';\r\nimport { UPLOADS_DIR } from '../shared/constants.js';\r\nimport type { TelegramConfig, SessionInfo } from '../shared/types.js';\r\n\r\nconst TELEGRAM_API = 'https://api.telegram.org/bot';\r\n\r\ninterface TelegramPhotoSize {\r\n file_id: string;\r\n file_unique_id: string;\r\n width: number;\r\n height: number;\r\n file_size?: number;\r\n}\r\n\r\ninterface TelegramMessage {\r\n message_id: number;\r\n chat: { id: number };\r\n text?: string;\r\n caption?: string;\r\n photo?: TelegramPhotoSize[];\r\n reply_to_message?: { message_id: number };\r\n}\r\n\r\ninterface TelegramCallbackQuery {\r\n id: string;\r\n from: { id: number };\r\n message?: TelegramMessage;\r\n data?: string;\r\n}\r\n\r\ninterface TelegramUpdate {\r\n update_id: number;\r\n message?: TelegramMessage;\r\n callback_query?: TelegramCallbackQuery;\r\n}\r\n\r\nexport class TelegramBot {\r\n private config: TelegramConfig;\r\n private offset = 0;\r\n private polling = false;\r\n private pollTimer: ReturnType<typeof setTimeout> | null = null;\r\n\r\n // message_id -> sessionId mapping for reply-based routing\r\n private messageSessionMap = new Map<number, string>();\r\n\r\n // Callback: when a text message arrives from Telegram for a session\r\n public onMessageToSession?: (sessionId: string, content: string) => void;\r\n // Callback: when an image arrives from Telegram for a session\r\n public onImageToSession?: (sessionId: string, imagePath: string, mimeType: string, caption?: string) => void;\r\n // Callback: when a permission verdict arrives from Telegram\r\n public onPermissionVerdict?: (sessionId: string, requestId: string, behavior: 'allow' | 'deny') => void;\r\n // Callback: get current sessions list\r\n public getSessions?: () => SessionInfo[];\r\n // Pending messages for session selection\r\n private pendingMessages = new Map<number, { text?: string; photoFileId?: string; caption?: string }>(); // chatId -> pending\r\n\r\n constructor(config: TelegramConfig) {\r\n this.config = config;\r\n }\r\n\r\n private get apiUrl(): string {\r\n return `${TELEGRAM_API}${this.config.botToken}`;\r\n }\r\n\r\n /** Send a notification message to Telegram */\r\n async sendNotification(sessionId: string, _sessionLabel: string, title: string, message: string): Promise<void> {\r\n const text = `<b>${this.escHtml(title)}</b>\\n${this.mdToHtml(message)}`;\r\n const result = await this.sendMessage(text);\r\n if (result?.message_id) {\r\n this.messageSessionMap.set(result.message_id, sessionId);\r\n // Cleanup old mappings (keep last 200)\r\n if (this.messageSessionMap.size > 200) {\r\n const keys = [...this.messageSessionMap.keys()];\r\n for (let i = 0; i < keys.length - 200; i++) {\r\n this.messageSessionMap.delete(keys[i]);\r\n }\r\n }\r\n }\r\n }\r\n\r\n /** Send a text message to the configured chat */\r\n private async sendMessage(text: string, replyToMessageId?: number, replyMarkup?: unknown): Promise<{ message_id: number } | null> {\r\n try {\r\n const body: Record<string, unknown> = {\r\n chat_id: this.config.chatId,\r\n text,\r\n parse_mode: 'HTML',\r\n };\r\n if (replyToMessageId) body.reply_to_message_id = replyToMessageId;\r\n if (replyMarkup) body.reply_markup = replyMarkup;\r\n\r\n const res = await fetch(`${this.apiUrl}/sendMessage`, {\r\n method: 'POST',\r\n headers: { 'Content-Type': 'application/json' },\r\n body: JSON.stringify(body),\r\n });\r\n\r\n if (!res.ok) {\r\n const err = await res.text();\r\n logger.warn(`Telegram sendMessage failed: ${res.status} ${err}`);\r\n return null;\r\n }\r\n\r\n const data = await res.json() as { ok: boolean; result: { message_id: number } };\r\n return data.ok ? data.result : null;\r\n } catch (err) {\r\n logger.warn(`Telegram sendMessage error: ${(err as Error).message}`);\r\n return null;\r\n }\r\n }\r\n\r\n /** Start long polling for incoming messages */\r\n startPolling(): void {\r\n if (this.polling) return;\r\n this.polling = true;\r\n logger.info('Telegram bot polling started');\r\n this.poll();\r\n }\r\n\r\n /** Stop polling */\r\n stopPolling(): void {\r\n this.polling = false;\r\n if (this.pollTimer) {\r\n clearTimeout(this.pollTimer);\r\n this.pollTimer = null;\r\n }\r\n logger.info('Telegram bot polling stopped');\r\n }\r\n\r\n private async poll(): Promise<void> {\r\n if (!this.polling) return;\r\n\r\n try {\r\n const res = await fetch(`${this.apiUrl}/getUpdates?offset=${this.offset}&timeout=30`, {\r\n signal: AbortSignal.timeout(35000),\r\n });\r\n\r\n if (!res.ok) {\r\n logger.warn(`Telegram getUpdates failed: ${res.status}`);\r\n this.scheduleNextPoll(5000);\r\n return;\r\n }\r\n\r\n const data = await res.json() as { ok: boolean; result: TelegramUpdate[] };\r\n if (data.ok && data.result.length > 0) {\r\n for (const update of data.result) {\r\n this.offset = update.update_id + 1;\r\n if (update.callback_query) {\r\n this.handleCallbackQuery(update.callback_query);\r\n } else if (update.message) {\r\n this.handleIncomingMessage(update.message);\r\n }\r\n }\r\n }\r\n } catch (err) {\r\n if ((err as Error).name !== 'AbortError') {\r\n logger.warn(`Telegram poll error: ${(err as Error).message}`);\r\n }\r\n }\r\n\r\n this.scheduleNextPoll(1000);\r\n }\r\n\r\n private scheduleNextPoll(delay: number): void {\r\n if (!this.polling) return;\r\n this.pollTimer = setTimeout(() => this.poll(), delay);\r\n }\r\n\r\n private async handleIncomingMessage(msg: TelegramMessage): Promise<void> {\r\n // Only process messages from the configured chat\r\n if (String(msg.chat.id) !== String(this.config.chatId)) return;\r\n\r\n const hasPhoto = msg.photo && msg.photo.length > 0;\r\n const text = (msg.text || msg.caption || '').trim();\r\n\r\n if (!text && !hasPhoto) return;\r\n\r\n // Check if it's a reply to a known message\r\n if (msg.reply_to_message) {\r\n const sessionId = this.messageSessionMap.get(msg.reply_to_message.message_id);\r\n if (sessionId) {\r\n if (hasPhoto) {\r\n await this.deliverPhotoToSession(sessionId, msg.photo!, text);\r\n } else {\r\n this.deliverToSession(sessionId, text);\r\n }\r\n return;\r\n }\r\n }\r\n\r\n // Check if it's a session selection command: /s_<index>\r\n if (text) {\r\n const selectMatch = text.match(/^\\/s_(\\d+)$/);\r\n if (selectMatch) {\r\n const pending = this.pendingMessages.get(msg.chat.id);\r\n if (pending) {\r\n this.pendingMessages.delete(msg.chat.id);\r\n const sessions = this.getSessions?.() ?? [];\r\n const idx = parseInt(selectMatch[1], 10) - 1;\r\n if (idx >= 0 && idx < sessions.length) {\r\n if (pending.photoFileId) {\r\n await this.deliverPhotoToSessionByFileId(sessions[idx].id, pending.photoFileId, pending.caption);\r\n } else if (pending.text) {\r\n this.deliverToSession(sessions[idx].id, pending.text);\r\n }\r\n this.sendMessage(`Sent to [${this.getLabel(sessions[idx])}]`);\r\n } else {\r\n this.sendMessage('Invalid session number.');\r\n }\r\n return;\r\n }\r\n }\r\n }\r\n\r\n // No reply context — check sessions count\r\n const sessions = this.getSessions?.() ?? [];\r\n\r\n if (sessions.length === 0) {\r\n this.sendMessage('No active sessions connected.');\r\n return;\r\n }\r\n\r\n if (sessions.length === 1) {\r\n if (hasPhoto) {\r\n await this.deliverPhotoToSession(sessions[0].id, msg.photo!, text);\r\n } else {\r\n this.deliverToSession(sessions[0].id, text);\r\n }\r\n return;\r\n }\r\n\r\n // Multiple sessions — ask user to pick with inline buttons\r\n if (hasPhoto) {\r\n const largest = msg.photo![msg.photo!.length - 1];\r\n this.pendingMessages.set(msg.chat.id, { photoFileId: largest.file_id, caption: text });\r\n } else {\r\n this.pendingMessages.set(msg.chat.id, { text });\r\n }\r\n const buttons = sessions.map((s, i) => ({\r\n text: this.getLabel(s),\r\n callback_data: `sess:${i}:${msg.chat.id}`,\r\n }));\r\n // Arrange buttons in rows of 2\r\n const rows: Array<typeof buttons> = [];\r\n for (let i = 0; i < buttons.length; i += 2) {\r\n rows.push(buttons.slice(i, i + 2));\r\n }\r\n this.sendMessage('Multiple sessions active. Select one:', undefined, { inline_keyboard: rows });\r\n }\r\n\r\n private deliverToSession(sessionId: string, content: string): void {\r\n if (this.onMessageToSession) {\r\n this.onMessageToSession(sessionId, content);\r\n }\r\n }\r\n\r\n private async deliverPhotoToSession(sessionId: string, photos: TelegramPhotoSize[], caption?: string): Promise<void> {\r\n // Get the largest photo (last in array)\r\n const largest = photos[photos.length - 1];\r\n await this.deliverPhotoToSessionByFileId(sessionId, largest.file_id, caption);\r\n }\r\n\r\n private async deliverPhotoToSessionByFileId(sessionId: string, fileId: string, caption?: string): Promise<void> {\r\n try {\r\n // Get file path from Telegram\r\n const fileRes = await fetch(`${this.apiUrl}/getFile?file_id=${fileId}`);\r\n if (!fileRes.ok) { logger.warn('Failed to get Telegram file info'); return; }\r\n const fileData = await fileRes.json() as { ok: boolean; result: { file_path: string } };\r\n if (!fileData.ok) return;\r\n\r\n // Download the file\r\n const downloadUrl = `https://api.telegram.org/file/bot${this.config.botToken}/${fileData.result.file_path}`;\r\n const imgRes = await fetch(downloadUrl);\r\n if (!imgRes.ok) { logger.warn('Failed to download Telegram photo'); return; }\r\n const buffer = Buffer.from(await imgRes.arrayBuffer());\r\n\r\n // Determine extension\r\n const ext = fileData.result.file_path.split('.').pop() || 'jpg';\r\n const mimeType = ext === 'png' ? 'image/png' : ext === 'gif' ? 'image/gif' : ext === 'webp' ? 'image/webp' : 'image/jpeg';\r\n\r\n // Save to uploads dir\r\n fs.mkdirSync(UPLOADS_DIR, { recursive: true });\r\n const filename = `${randomUUID()}.${ext}`;\r\n const filePath = path.join(UPLOADS_DIR, filename);\r\n fs.writeFileSync(filePath, buffer);\r\n logger.info(`Telegram photo saved: ${filename} (${buffer.length} bytes)`);\r\n\r\n // Deliver to session\r\n if (this.onImageToSession) {\r\n this.onImageToSession(sessionId, filePath, mimeType, caption);\r\n }\r\n\r\n // Cleanup after 5 minutes\r\n setTimeout(() => { try { fs.unlinkSync(filePath); } catch {} }, 5 * 60 * 1000);\r\n } catch (err) {\r\n logger.warn(`Telegram photo download failed: ${(err as Error).message}`);\r\n }\r\n }\r\n\r\n private getLabel(session: SessionInfo): string {\r\n return session.displayName || session.cwd?.replace(/^.*[/\\\\]/, '') || session.name;\r\n }\r\n\r\n /** Send a permission request with inline buttons */\r\n async sendPermissionRequest(sessionId: string, sessionLabel: string, requestId: string, toolName: string, description: string, inputPreview: string): Promise<void> {\r\n // Parse inputPreview for readable display\r\n let preview = inputPreview;\r\n let truncated = false;\r\n try {\r\n const p = JSON.parse(inputPreview);\r\n if (p.command) preview = `$ ${p.command}`;\r\n else if (p.title && p.message) preview = p.message;\r\n else if (p.file_path) {\r\n preview = p.file_path;\r\n if (p.content) { preview += '\\n' + p.content.slice(0, 3000); if (p.content.length > 500) truncated = true; }\r\n } else if (p.content && typeof p.content === 'string') {\r\n preview = p.content.slice(0, 3000);\r\n if (p.content.length > 500) truncated = true;\r\n }\r\n } catch {\r\n truncated = true;\r\n const cmdMatch = inputPreview.match(/\"command\"\\s*:\\s*\"((?:[^\"\\\\]|\\\\.)*)\"/);\r\n const contentMatch = inputPreview.match(/\"content\"\\s*:\\s*\"((?:[^\"\\\\]|\\\\.)*)\"/);\r\n if (cmdMatch) preview = `$ ${cmdMatch[1]}`;\r\n else if (contentMatch) preview = contentMatch[1].slice(0, 3000);\r\n }\r\n\r\n const truncNote = truncated ? '\\n\\n<i>...truncated</i>' : '';\r\n const previewSlice = preview.slice(0, 3000);\r\n // Use <code> for short single-line (commands), plain text for longer content\r\n const isShort = !previewSlice.includes('\\n') && previewSlice.length < 100;\r\n const previewHtml = isShort\r\n ? `<code>${this.escHtml(previewSlice)}</code>`\r\n : this.escHtml(previewSlice);\r\n // Friendly tool name for Telegram — use title for notify tools\r\n let displayTool = toolName;\r\n if ((toolName.endsWith('__notify') || toolName === 'notify') && inputPreview) {\r\n try { const pp = JSON.parse(inputPreview); if (pp.title) displayTool = pp.title; } catch {\r\n const tm = inputPreview.match(/\"title\"\\s*:\\s*\"((?:[^\"\\\\]|\\\\.)*)\"/);\r\n if (tm) displayTool = tm[1];\r\n }\r\n } else {\r\n const mcpMatch = toolName.match(/__([^_]+)$/);\r\n if (mcpMatch) displayTool = mcpMatch[1].charAt(0).toUpperCase() + mcpMatch[1].slice(1);\r\n }\r\n\r\n const text = `⚠️ <b>Permission Request</b> — ${this.escHtml(sessionLabel)}\\n\\n` +\r\n `🔧 <b>${this.escHtml(displayTool)}</b>\\n` +\r\n `${previewHtml}${truncNote}`;\r\n\r\n const replyMarkup = {\r\n inline_keyboard: [[\r\n { text: '✅ Allow', callback_data: `perm:allow:${sessionId}:${requestId}` },\r\n { text: '❌ Deny', callback_data: `perm:deny:${sessionId}:${requestId}` },\r\n ]],\r\n };\r\n\r\n await this.sendMessage(text, undefined, replyMarkup);\r\n }\r\n\r\n private async handleCallbackQuery(query: TelegramCallbackQuery): Promise<void> {\r\n if (!query.data) return;\r\n\r\n if (query.data.startsWith('sess:')) {\r\n await this.handleSessionSelectCallback(query);\r\n return;\r\n }\r\n\r\n if (!query.data.startsWith('perm:')) return;\r\n\r\n const parts = query.data.split(':');\r\n if (parts.length < 4) return;\r\n const [, action, sessionId, requestId] = parts;\r\n const behavior = action === 'allow' ? 'allow' : 'deny';\r\n\r\n // Send verdict\r\n if (this.onPermissionVerdict) {\r\n this.onPermissionVerdict(sessionId, requestId, behavior as 'allow' | 'deny');\r\n }\r\n\r\n // Answer callback to remove loading state\r\n await this.answerCallbackQuery(query.id, behavior === 'allow' ? '✅ Allowed' : '❌ Denied');\r\n\r\n // Update message to show result (use escaped original text since it's plain)\r\n if (query.message) {\r\n const label = behavior === 'allow' ? '✅ <b>Allowed</b>' : '❌ <b>Denied</b>';\r\n const original = this.escHtml(query.message.text || '');\r\n await this.editMessageText(query.message.chat.id, query.message.message_id, original + `\\n\\n${label}`);\r\n }\r\n }\r\n\r\n private async handleSessionSelectCallback(query: TelegramCallbackQuery): Promise<void> {\r\n const parts = query.data!.split(':');\r\n if (parts.length < 3) return;\r\n const [, idxStr, chatIdStr] = parts;\r\n const idx = parseInt(idxStr, 10);\r\n const chatId = parseInt(chatIdStr, 10);\r\n\r\n const sessions = this.getSessions?.() ?? [];\r\n if (idx < 0 || idx >= sessions.length) {\r\n await this.answerCallbackQuery(query.id, 'Session not found');\r\n return;\r\n }\r\n\r\n const session = sessions[idx];\r\n const pending = this.pendingMessages.get(chatId);\r\n this.pendingMessages.delete(chatId);\r\n\r\n if (pending) {\r\n if (pending.photoFileId) {\r\n await this.deliverPhotoToSessionByFileId(session.id, pending.photoFileId, pending.caption);\r\n } else if (pending.text) {\r\n this.deliverToSession(session.id, pending.text);\r\n }\r\n }\r\n\r\n await this.answerCallbackQuery(query.id, `Sent to ${this.getLabel(session)}`);\r\n // Update message to show which session was selected\r\n if (query.message) {\r\n await this.editMessageText(query.message.chat.id, query.message.message_id, `✅ Sent to <b>${this.escHtml(this.getLabel(session))}</b>`);\r\n }\r\n }\r\n\r\n private async answerCallbackQuery(callbackQueryId: string, text: string): Promise<void> {\r\n try {\r\n await fetch(`${this.apiUrl}/answerCallbackQuery`, {\r\n method: 'POST',\r\n headers: { 'Content-Type': 'application/json' },\r\n body: JSON.stringify({ callback_query_id: callbackQueryId, text }),\r\n });\r\n } catch (err) {\r\n logger.warn(`Telegram answerCallbackQuery error: ${(err as Error).message}`);\r\n }\r\n }\r\n\r\n private async editMessageText(chatId: number, messageId: number, text: string): Promise<void> {\r\n try {\r\n await fetch(`${this.apiUrl}/editMessageText`, {\r\n method: 'POST',\r\n headers: { 'Content-Type': 'application/json' },\r\n body: JSON.stringify({ chat_id: chatId, message_id: messageId, text, parse_mode: 'HTML' }),\r\n });\r\n } catch (err) {\r\n logger.warn(`Telegram editMessageText error: ${(err as Error).message}`);\r\n }\r\n }\r\n\r\n private escHtml(s: string): string {\r\n return s.replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>');\r\n }\r\n\r\n /** Convert markdown to Telegram HTML (escape first, then apply formatting) */\r\n private mdToHtml(s: string): string {\r\n // Handle tables before escaping (convert to clean text format)\r\n let text = s.replace(\r\n /^(\\|.+\\|)\\n\\|[-| :]+\\|\\n((?:\\|.+\\|\\n?)*)/gm,\r\n (_match, header: string, body: string) => {\r\n const headerCells = header.split('|').filter((c: string) => c.trim()).map((c: string) => c.trim());\r\n const headerLine = headerCells.join(' | ');\r\n const bodyLines = body.trim().split('\\n').map((row: string) => {\r\n return row.split('|').filter((c: string) => c.trim()).map((c: string) => c.trim()).join(' | ');\r\n });\r\n return `**${headerLine}**\\n${bodyLines.join('\\n')}`;\r\n },\r\n );\r\n let html = this.escHtml(text);\r\n // Code blocks: ```...```\r\n html = html.replace(/```(?:\\w*)\\n?([\\s\\S]*?)```/g, '<pre>$1</pre>');\r\n // Inline code: `...`\r\n html = html.replace(/`([^`]+)`/g, '<code>$1</code>');\r\n // Headings: # / ## / ### → bold (Telegram has no heading tags)\r\n html = html.replace(/^#{1,3}\\s+(.+)$/gm, '<b>$1</b>');\r\n // Bold: **...**\r\n html = html.replace(/\\*\\*(.+?)\\*\\*/g, '<b>$1</b>');\r\n // Italic: *...*\r\n html = html.replace(/\\*(.+?)\\*/g, '<i>$1</i>');\r\n return html;\r\n }\r\n\r\n /** Update config (e.g., from dashboard settings) */\r\n updateConfig(config: TelegramConfig): void {\r\n const wasPolling = this.polling;\r\n if (wasPolling) this.stopPolling();\r\n this.config = config;\r\n if (wasPolling && config.enabled) this.startPolling();\r\n }\r\n}\r\n","import fs from 'node:fs';\r\nimport path from 'node:path';\r\nimport { randomUUID } from 'node:crypto';\r\nimport { CONFIG_DIR, CONFIG_FILE, DEFAULT_HUB_HOST, DEFAULT_HUB_PORT } from './constants.js';\r\nimport type { AppConfig } from './types.js';\r\n\r\nconst DEFAULT_CONFIG: AppConfig = {\r\n hub: {\r\n host: DEFAULT_HUB_HOST,\r\n port: DEFAULT_HUB_PORT,\r\n },\r\n notifications: {\r\n desktop: true,\r\n sound: true,\r\n },\r\n webhooks: [],\r\n};\r\n\r\nexport function ensureConfigDir(): void {\r\n if (!fs.existsSync(CONFIG_DIR)) {\r\n fs.mkdirSync(CONFIG_DIR, { recursive: true });\r\n }\r\n}\r\n\r\nexport function loadConfig(): AppConfig {\r\n ensureConfigDir();\r\n let config: AppConfig;\r\n if (!fs.existsSync(CONFIG_FILE)) {\r\n config = { ...DEFAULT_CONFIG, hub: { ...DEFAULT_CONFIG.hub } };\r\n } else {\r\n try {\r\n const raw = fs.readFileSync(CONFIG_FILE, 'utf-8');\r\n const parsed = JSON.parse(raw);\r\n config = { ...DEFAULT_CONFIG, ...parsed, hub: { ...DEFAULT_CONFIG.hub, ...parsed.hub }, ...(parsed.telegram ? { telegram: parsed.telegram } : {}) };\r\n } catch {\r\n config = { ...DEFAULT_CONFIG, hub: { ...DEFAULT_CONFIG.hub } };\r\n }\r\n }\r\n\r\n // Auto-generate token if missing\r\n if (!config.hub.token) {\r\n config.hub.token = randomUUID();\r\n saveConfig(config);\r\n }\r\n\r\n return config;\r\n}\r\n\r\n/** Get the current token, generating one if needed */\r\nexport function getOrCreateToken(): string {\r\n const config = loadConfig();\r\n return config.hub.token!;\r\n}\r\n\r\nexport function saveConfig(config: AppConfig): void {\r\n ensureConfigDir();\r\n fs.writeFileSync(CONFIG_FILE, JSON.stringify(config, null, 2), { encoding: 'utf-8', mode: 0o600 });\r\n}\r\n\r\n/**\r\n * Add claude-alarm as an MCP channel server to .mcp.json\r\n */\r\nexport function setupMcpConfig(targetDir?: string): string {\r\n const dir = targetDir ?? process.cwd();\r\n const mcpPath = path.join(dir, '.mcp.json');\r\n\r\n let mcpConfig: Record<string, any> = {};\r\n if (fs.existsSync(mcpPath)) {\r\n try {\r\n mcpConfig = JSON.parse(fs.readFileSync(mcpPath, 'utf-8'));\r\n } catch {\r\n mcpConfig = {};\r\n }\r\n }\r\n\r\n if (!mcpConfig.mcpServers) {\r\n mcpConfig.mcpServers = {};\r\n }\r\n\r\n mcpConfig.mcpServers['claude-alarm'] = {\r\n command: 'npx',\r\n args: ['-y', '@delt/claude-alarm', 'serve'],\r\n env: {\r\n CLAUDE_ALARM_SESSION_NAME: path.basename(dir),\r\n },\r\n };\r\n\r\n fs.writeFileSync(mcpPath, JSON.stringify(mcpConfig, null, 2), 'utf-8');\r\n return mcpPath;\r\n}\r\n","import WebSocket from 'ws';\r\nimport { logger } from '../shared/logger.js';\r\nimport { DEFAULT_HUB_HOST, DEFAULT_HUB_PORT, WS_PATH_CHANNEL } from '../shared/constants.js';\r\nimport type { ChannelMessage, SessionInfo } from '../shared/types.js';\r\n\r\nexport class HubClient {\r\n private ws: WebSocket | null = null;\r\n private reconnectTimer: ReturnType<typeof setTimeout> | null = null;\r\n private messageHandlers: Array<(msg: ChannelMessage) => void> = [];\r\n private queue: ChannelMessage[] = [];\r\n private connected = false;\r\n\r\n constructor(\r\n private sessionId: string,\r\n private sessionName: string,\r\n private hubHost = DEFAULT_HUB_HOST,\r\n private hubPort = DEFAULT_HUB_PORT,\r\n private token?: string,\r\n ) {}\r\n\r\n connect(): void {\r\n const tokenQuery = this.token ? `?token=${encodeURIComponent(this.token)}` : '';\r\n const url = `ws://${this.hubHost}:${this.hubPort}${WS_PATH_CHANNEL}${tokenQuery}`;\r\n logger.debug(`Connecting to hub at ${url}`);\r\n\r\n try {\r\n this.ws = new WebSocket(url);\r\n\r\n this.ws.on('open', () => {\r\n logger.info('Connected to hub');\r\n this.connected = true;\r\n\r\n // Register this session\r\n const registration: ChannelMessage = {\r\n type: 'register',\r\n session: {\r\n id: this.sessionId,\r\n name: this.sessionName,\r\n status: 'idle',\r\n connectedAt: Date.now(),\r\n lastActivity: Date.now(),\r\n cwd: process.cwd(),\r\n channelEnabled: true,\r\n },\r\n };\r\n this.ws!.send(JSON.stringify(registration));\r\n\r\n // Flush queued messages\r\n for (const msg of this.queue) {\r\n this.ws!.send(JSON.stringify(msg));\r\n }\r\n this.queue = [];\r\n });\r\n\r\n this.ws.on('message', (data) => {\r\n try {\r\n const msg = JSON.parse(data.toString()) as ChannelMessage;\r\n for (const handler of this.messageHandlers) {\r\n handler(msg);\r\n }\r\n } catch (err) {\r\n logger.warn('Failed to parse hub message:', err);\r\n }\r\n });\r\n\r\n this.ws.on('close', () => {\r\n logger.info('Disconnected from hub');\r\n this.connected = false;\r\n this.scheduleReconnect();\r\n });\r\n\r\n this.ws.on('error', (err) => {\r\n logger.debug(`Hub connection error: ${err.message}`);\r\n this.connected = false;\r\n });\r\n } catch {\r\n logger.debug('Failed to connect to hub, will retry');\r\n this.scheduleReconnect();\r\n }\r\n }\r\n\r\n send(msg: ChannelMessage): void {\r\n if (this.connected && this.ws?.readyState === WebSocket.OPEN) {\r\n this.ws.send(JSON.stringify(msg));\r\n } else {\r\n if (this.queue.length < 100) {\r\n this.queue.push(msg);\r\n }\r\n logger.debug('Hub not connected, message queued');\r\n }\r\n }\r\n\r\n onMessage(handler: (msg: ChannelMessage) => void): void {\r\n this.messageHandlers.push(handler);\r\n }\r\n\r\n disconnect(): void {\r\n if (this.reconnectTimer) {\r\n clearTimeout(this.reconnectTimer);\r\n this.reconnectTimer = null;\r\n }\r\n if (this.ws) {\r\n this.ws.close();\r\n this.ws = null;\r\n }\r\n this.connected = false;\r\n }\r\n\r\n private scheduleReconnect(): void {\r\n if (this.reconnectTimer) return;\r\n this.reconnectTimer = setTimeout(() => {\r\n this.reconnectTimer = null;\r\n this.connect();\r\n }, 5000);\r\n }\r\n}\r\n"],"mappings":"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","names":["fs","path","randomUUID","path","sessions","path","fs","path","randomUUID","fs","randomUUID","fs","path","path","fs","randomUUID","WebSocket","WebSocket"]}
|