claude-pager 0.1.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.
Files changed (143) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +207 -0
  3. package/dist/channels/channel.d.ts +27 -0
  4. package/dist/channels/channel.d.ts.map +1 -0
  5. package/dist/channels/channel.js +3 -0
  6. package/dist/channels/channel.js.map +1 -0
  7. package/dist/channels/factory.d.ts +4 -0
  8. package/dist/channels/factory.d.ts.map +1 -0
  9. package/dist/channels/factory.js +22 -0
  10. package/dist/channels/factory.js.map +1 -0
  11. package/dist/channels/ntfy/__tests__/provider.test.d.ts +2 -0
  12. package/dist/channels/ntfy/__tests__/provider.test.d.ts.map +1 -0
  13. package/dist/channels/ntfy/__tests__/provider.test.js +29 -0
  14. package/dist/channels/ntfy/__tests__/provider.test.js.map +1 -0
  15. package/dist/channels/ntfy/provider.d.ts +17 -0
  16. package/dist/channels/ntfy/provider.d.ts.map +1 -0
  17. package/dist/channels/ntfy/provider.js +142 -0
  18. package/dist/channels/ntfy/provider.js.map +1 -0
  19. package/dist/channels/telegram/provider.d.ts +27 -0
  20. package/dist/channels/telegram/provider.d.ts.map +1 -0
  21. package/dist/channels/telegram/provider.js +312 -0
  22. package/dist/channels/telegram/provider.js.map +1 -0
  23. package/dist/channels/telegram/voice-handler.d.ts +22 -0
  24. package/dist/channels/telegram/voice-handler.d.ts.map +1 -0
  25. package/dist/channels/telegram/voice-handler.js +68 -0
  26. package/dist/channels/telegram/voice-handler.js.map +1 -0
  27. package/dist/cli/index.d.ts +3 -0
  28. package/dist/cli/index.d.ts.map +1 -0
  29. package/dist/cli/index.js +99 -0
  30. package/dist/cli/index.js.map +1 -0
  31. package/dist/cli/recover.d.ts +2 -0
  32. package/dist/cli/recover.d.ts.map +1 -0
  33. package/dist/cli/recover.js +45 -0
  34. package/dist/cli/recover.js.map +1 -0
  35. package/dist/cli/run.d.ts +2 -0
  36. package/dist/cli/run.d.ts.map +1 -0
  37. package/dist/cli/run.js +32 -0
  38. package/dist/cli/run.js.map +1 -0
  39. package/dist/cli/setup.d.ts +8 -0
  40. package/dist/cli/setup.d.ts.map +1 -0
  41. package/dist/cli/setup.js +238 -0
  42. package/dist/cli/setup.js.map +1 -0
  43. package/dist/config/index.d.ts +6 -0
  44. package/dist/config/index.d.ts.map +1 -0
  45. package/dist/config/index.js +40 -0
  46. package/dist/config/index.js.map +1 -0
  47. package/dist/daemon/__tests__/handlers.test.d.ts +2 -0
  48. package/dist/daemon/__tests__/handlers.test.d.ts.map +1 -0
  49. package/dist/daemon/__tests__/handlers.test.js +99 -0
  50. package/dist/daemon/__tests__/handlers.test.js.map +1 -0
  51. package/dist/daemon/__tests__/server.test.d.ts +2 -0
  52. package/dist/daemon/__tests__/server.test.d.ts.map +1 -0
  53. package/dist/daemon/__tests__/server.test.js +118 -0
  54. package/dist/daemon/__tests__/server.test.js.map +1 -0
  55. package/dist/daemon/handlers.d.ts +4 -0
  56. package/dist/daemon/handlers.d.ts.map +1 -0
  57. package/dist/daemon/handlers.js +153 -0
  58. package/dist/daemon/handlers.js.map +1 -0
  59. package/dist/daemon/index.d.ts +7 -0
  60. package/dist/daemon/index.d.ts.map +1 -0
  61. package/dist/daemon/index.js +83 -0
  62. package/dist/daemon/index.js.map +1 -0
  63. package/dist/daemon/server.d.ts +11 -0
  64. package/dist/daemon/server.d.ts.map +1 -0
  65. package/dist/daemon/server.js +115 -0
  66. package/dist/daemon/server.js.map +1 -0
  67. package/dist/hooks/index.d.ts +3 -0
  68. package/dist/hooks/index.d.ts.map +1 -0
  69. package/dist/hooks/index.js +172 -0
  70. package/dist/hooks/index.js.map +1 -0
  71. package/dist/index.d.ts +4 -0
  72. package/dist/index.d.ts.map +1 -0
  73. package/dist/index.js +3 -0
  74. package/dist/index.js.map +1 -0
  75. package/dist/injectors/__tests__/tmux.test.d.ts +2 -0
  76. package/dist/injectors/__tests__/tmux.test.d.ts.map +1 -0
  77. package/dist/injectors/__tests__/tmux.test.js +38 -0
  78. package/dist/injectors/__tests__/tmux.test.js.map +1 -0
  79. package/dist/injectors/factory.d.ts +3 -0
  80. package/dist/injectors/factory.d.ts.map +1 -0
  81. package/dist/injectors/factory.js +22 -0
  82. package/dist/injectors/factory.js.map +1 -0
  83. package/dist/injectors/injector.d.ts +7 -0
  84. package/dist/injectors/injector.d.ts.map +1 -0
  85. package/dist/injectors/injector.js +3 -0
  86. package/dist/injectors/injector.js.map +1 -0
  87. package/dist/injectors/tmux/injector.d.ts +9 -0
  88. package/dist/injectors/tmux/injector.d.ts.map +1 -0
  89. package/dist/injectors/tmux/injector.js +55 -0
  90. package/dist/injectors/tmux/injector.js.map +1 -0
  91. package/dist/injectors/xdotool/injector.d.ts +9 -0
  92. package/dist/injectors/xdotool/injector.d.ts.map +1 -0
  93. package/dist/injectors/xdotool/injector.js +59 -0
  94. package/dist/injectors/xdotool/injector.js.map +1 -0
  95. package/dist/sessions/__tests__/events.test.d.ts +2 -0
  96. package/dist/sessions/__tests__/events.test.d.ts.map +1 -0
  97. package/dist/sessions/__tests__/events.test.js +103 -0
  98. package/dist/sessions/__tests__/events.test.js.map +1 -0
  99. package/dist/sessions/__tests__/tracker.test.d.ts +2 -0
  100. package/dist/sessions/__tests__/tracker.test.d.ts.map +1 -0
  101. package/dist/sessions/__tests__/tracker.test.js +24 -0
  102. package/dist/sessions/__tests__/tracker.test.js.map +1 -0
  103. package/dist/sessions/events.d.ts +11 -0
  104. package/dist/sessions/events.d.ts.map +1 -0
  105. package/dist/sessions/events.js +73 -0
  106. package/dist/sessions/events.js.map +1 -0
  107. package/dist/sessions/tracker.d.ts +7 -0
  108. package/dist/sessions/tracker.d.ts.map +1 -0
  109. package/dist/sessions/tracker.js +83 -0
  110. package/dist/sessions/tracker.js.map +1 -0
  111. package/dist/types.d.ts +55 -0
  112. package/dist/types.d.ts.map +1 -0
  113. package/dist/types.js +3 -0
  114. package/dist/types.js.map +1 -0
  115. package/dist/utils/__tests__/html.test.d.ts +2 -0
  116. package/dist/utils/__tests__/html.test.d.ts.map +1 -0
  117. package/dist/utils/__tests__/html.test.js +42 -0
  118. package/dist/utils/__tests__/html.test.js.map +1 -0
  119. package/dist/utils/__tests__/json.test.d.ts +2 -0
  120. package/dist/utils/__tests__/json.test.d.ts.map +1 -0
  121. package/dist/utils/__tests__/json.test.js +27 -0
  122. package/dist/utils/__tests__/json.test.js.map +1 -0
  123. package/dist/utils/__tests__/validation.test.d.ts +2 -0
  124. package/dist/utils/__tests__/validation.test.d.ts.map +1 -0
  125. package/dist/utils/__tests__/validation.test.js +38 -0
  126. package/dist/utils/__tests__/validation.test.js.map +1 -0
  127. package/dist/utils/html.d.ts +3 -0
  128. package/dist/utils/html.d.ts.map +1 -0
  129. package/dist/utils/html.js +43 -0
  130. package/dist/utils/html.js.map +1 -0
  131. package/dist/utils/json.d.ts +2 -0
  132. package/dist/utils/json.d.ts.map +1 -0
  133. package/dist/utils/json.js +13 -0
  134. package/dist/utils/json.js.map +1 -0
  135. package/dist/utils/validation.d.ts +4 -0
  136. package/dist/utils/validation.d.ts.map +1 -0
  137. package/dist/utils/validation.js +16 -0
  138. package/dist/utils/validation.js.map +1 -0
  139. package/dist/voice/transcribe.d.ts +7 -0
  140. package/dist/voice/transcribe.d.ts.map +1 -0
  141. package/dist/voice/transcribe.js +66 -0
  142. package/dist/voice/transcribe.js.map +1 -0
  143. package/package.json +50 -0
@@ -0,0 +1,312 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.TelegramProvider = void 0;
4
+ const html_js_1 = require("../../utils/html.js");
5
+ const voice_handler_js_1 = require("./voice-handler.js");
6
+ const MAP_MAX_SIZE = 500;
7
+ class TelegramProvider {
8
+ name = 'telegram';
9
+ abortController = null;
10
+ config;
11
+ lastUpdateId = 0;
12
+ // Map telegram message_id → event_id for reply-based routing
13
+ messageToEvent = new Map();
14
+ // Map telegram message_id → session_id for free message routing
15
+ messageToSession = new Map();
16
+ constructor(config) {
17
+ this.config = config;
18
+ }
19
+ apiUrl(method) {
20
+ return `https://api.telegram.org/bot${this.config.botToken}/${method}`;
21
+ }
22
+ async send(event, shortId) {
23
+ const projectName = event.project.split('/').pop() || event.project;
24
+ const isPermission = event.type === 'permission_prompt';
25
+ // Build message text (HTML format)
26
+ const icon = isPermission ? '🔒' : '💬';
27
+ let text = `${icon} <b>#${shortId} ${(0, html_js_1.escapeHtml)(projectName)}</b>\n\n`;
28
+ if (event.toolName) {
29
+ text += `<code>${(0, html_js_1.escapeHtml)(event.toolName)}</code>`;
30
+ if (event.toolInput) {
31
+ const input = event.toolInput.length > 300
32
+ ? event.toolInput.slice(0, 300) + '...'
33
+ : event.toolInput;
34
+ text += `\n<pre>${(0, html_js_1.escapeHtml)(input)}</pre>`;
35
+ }
36
+ }
37
+ else {
38
+ text += (0, html_js_1.markdownToHtml)(event.message);
39
+ }
40
+ // Build inline keyboard
41
+ const keyboard = isPermission
42
+ ? {
43
+ inline_keyboard: [
44
+ [
45
+ { text: '✅ Allow', callback_data: `allow:${event.id}` },
46
+ { text: '❌ Deny', callback_data: `deny:${event.id}` },
47
+ ],
48
+ ],
49
+ }
50
+ : undefined;
51
+ try {
52
+ const res = await fetch(this.apiUrl('sendMessage'), {
53
+ method: 'POST',
54
+ headers: { 'Content-Type': 'application/json' },
55
+ body: JSON.stringify({
56
+ chat_id: this.config.chatId,
57
+ text,
58
+ parse_mode: 'HTML',
59
+ reply_markup: keyboard,
60
+ }),
61
+ signal: AbortSignal.timeout(15000),
62
+ });
63
+ const data = (await res.json());
64
+ if (!data.ok) {
65
+ return { success: false, error: `Telegram API: ${data.description}` };
66
+ }
67
+ const messageId = data.result?.message_id;
68
+ if (messageId) {
69
+ this.capMap(this.messageToEvent);
70
+ this.capMap(this.messageToSession);
71
+ this.messageToEvent.set(messageId, event.id);
72
+ this.messageToSession.set(messageId, event.sessionId);
73
+ }
74
+ return { success: true, messageId: String(messageId) };
75
+ }
76
+ catch (err) {
77
+ return { success: false, error: String(err) };
78
+ }
79
+ }
80
+ async sendRaw(text) {
81
+ await fetch(this.apiUrl('sendMessage'), {
82
+ method: 'POST',
83
+ headers: { 'Content-Type': 'application/json' },
84
+ body: JSON.stringify({
85
+ chat_id: this.config.chatId,
86
+ text,
87
+ parse_mode: 'HTML',
88
+ }),
89
+ signal: AbortSignal.timeout(15000),
90
+ });
91
+ }
92
+ async sendSessionPicker(text, sessions) {
93
+ const keyboard = {
94
+ inline_keyboard: sessions.map(s => [
95
+ { text: s.label, callback_data: `session:${s.id}` },
96
+ ]),
97
+ };
98
+ try {
99
+ const res = await fetch(this.apiUrl('sendMessage'), {
100
+ method: 'POST',
101
+ headers: { 'Content-Type': 'application/json' },
102
+ body: JSON.stringify({
103
+ chat_id: this.config.chatId,
104
+ text,
105
+ parse_mode: 'HTML',
106
+ reply_markup: keyboard,
107
+ }),
108
+ signal: AbortSignal.timeout(15000),
109
+ });
110
+ const data = (await res.json());
111
+ return data.result?.message_id;
112
+ }
113
+ catch {
114
+ return undefined;
115
+ }
116
+ }
117
+ startListening(listeners) {
118
+ this.stopListening();
119
+ this.abortController = new AbortController();
120
+ this.poll(listeners);
121
+ }
122
+ async poll(listeners) {
123
+ const signal = this.abortController.signal;
124
+ while (!signal.aborted) {
125
+ try {
126
+ const res = await fetch(this.apiUrl('getUpdates'), {
127
+ method: 'POST',
128
+ headers: { 'Content-Type': 'application/json' },
129
+ body: JSON.stringify({
130
+ offset: this.lastUpdateId,
131
+ timeout: 25,
132
+ allowed_updates: ['callback_query', 'message'],
133
+ }),
134
+ signal,
135
+ });
136
+ const data = (await res.json());
137
+ if (!data.ok || !data.result)
138
+ continue;
139
+ for (const update of data.result) {
140
+ this.lastUpdateId = update.update_id + 1;
141
+ if (update.callback_query) {
142
+ await this.handleCallback(update.callback_query, listeners);
143
+ }
144
+ else if (update.message?.voice) {
145
+ await this.handleVoice(update.message, listeners);
146
+ }
147
+ else if (update.message?.text) {
148
+ if (update.message.reply_to_message) {
149
+ await this.handleReply(update.message, listeners);
150
+ }
151
+ else if (listeners.onFreeMessage) {
152
+ await this.handleFreeMessage(update.message, listeners.onFreeMessage);
153
+ }
154
+ }
155
+ }
156
+ }
157
+ catch (err) {
158
+ if (signal.aborted)
159
+ return;
160
+ console.error('[telegram] poll error:', err);
161
+ await new Promise(resolve => {
162
+ const timer = setTimeout(resolve, 5000);
163
+ signal.addEventListener('abort', () => { clearTimeout(timer); resolve(); }, { once: true });
164
+ });
165
+ }
166
+ }
167
+ }
168
+ async handleCallback(cb, listeners) {
169
+ // Acknowledge the button press
170
+ try {
171
+ await fetch(this.apiUrl('answerCallbackQuery'), {
172
+ method: 'POST',
173
+ headers: { 'Content-Type': 'application/json' },
174
+ body: JSON.stringify({ callback_query_id: cb.id, text: 'Received!' }),
175
+ });
176
+ }
177
+ catch { /* best effort */ }
178
+ if (!cb.data)
179
+ return;
180
+ const [action, id] = cb.data.split(':', 2);
181
+ if (!id)
182
+ return;
183
+ // Session picker callback
184
+ if (action === 'session') {
185
+ if (cb.message) {
186
+ try {
187
+ await fetch(this.apiUrl('editMessageText'), {
188
+ method: 'POST',
189
+ headers: { 'Content-Type': 'application/json' },
190
+ body: JSON.stringify({
191
+ chat_id: cb.message.chat.id,
192
+ message_id: cb.message.message_id,
193
+ text: `➡️ Sent to session`,
194
+ }),
195
+ });
196
+ }
197
+ catch { /* best effort */ }
198
+ }
199
+ // Forward session pick as a special response
200
+ console.log(`[telegram] session picked: ${id}`);
201
+ try {
202
+ await Promise.resolve(listeners.onResponse(`__session_pick__:${id}`));
203
+ }
204
+ catch (err) {
205
+ console.error('[telegram] session pick handler error:', err);
206
+ }
207
+ return;
208
+ }
209
+ const responseText = action === 'allow' ? 'allow' : action === 'deny' ? 'deny' : action;
210
+ // Update the message to show the action taken
211
+ if (cb.message) {
212
+ try {
213
+ await fetch(this.apiUrl('editMessageReplyMarkup'), {
214
+ method: 'POST',
215
+ headers: { 'Content-Type': 'application/json' },
216
+ body: JSON.stringify({
217
+ chat_id: cb.message.chat.id,
218
+ message_id: cb.message.message_id,
219
+ reply_markup: {
220
+ inline_keyboard: [
221
+ [{ text: action === 'allow' ? '✅ Allowed' : '❌ Denied', callback_data: 'noop' }],
222
+ ],
223
+ },
224
+ }),
225
+ });
226
+ }
227
+ catch { /* best effort */ }
228
+ }
229
+ console.log(`[telegram] callback: ${action} for event ${id}`);
230
+ try {
231
+ await Promise.resolve(listeners.onResponse(`#${id} ${responseText}`));
232
+ }
233
+ catch (err) {
234
+ console.error('[telegram] callback handler error:', err);
235
+ }
236
+ }
237
+ async handleVoice(msg, listeners) {
238
+ await (0, voice_handler_js_1.handleVoiceMessage)(msg, listeners, {
239
+ config: this.config,
240
+ apiUrl: (method) => this.apiUrl(method),
241
+ handleReply: (m, l) => this.handleReply(m, l),
242
+ handleFreeMessage: (m, cb) => this.handleFreeMessage(m, cb),
243
+ });
244
+ }
245
+ async handleFreeMessage(msg, onFreeMessage) {
246
+ if (!msg.text)
247
+ return;
248
+ console.log(`[telegram] free message: "${msg.text}"`);
249
+ try {
250
+ await Promise.resolve(onFreeMessage({
251
+ text: msg.text,
252
+ replyCallback: async (reply) => {
253
+ await this.sendRaw(reply);
254
+ },
255
+ }));
256
+ }
257
+ catch (err) {
258
+ console.error('[telegram] free message handler error:', err);
259
+ }
260
+ }
261
+ async handleReply(msg, listeners) {
262
+ if (!msg.text || !msg.reply_to_message)
263
+ return;
264
+ const replyToId = msg.reply_to_message.message_id;
265
+ const eventId = this.messageToEvent.get(replyToId);
266
+ if (eventId) {
267
+ // Reply to a notification with a pending event
268
+ console.log(`[telegram] reply to event ${eventId}: "${msg.text}"`);
269
+ try {
270
+ await Promise.resolve(listeners.onResponse(`#${eventId} ${msg.text}`));
271
+ }
272
+ catch (err) {
273
+ console.error('[telegram] reply handler error:', err);
274
+ }
275
+ return;
276
+ }
277
+ const sessionId = this.messageToSession.get(replyToId);
278
+ if (sessionId && listeners.onFreeMessage) {
279
+ // Reply to a notification — we know the session, inject directly
280
+ console.log(`[telegram] reply to session ${sessionId}: "${msg.text}"`);
281
+ try {
282
+ await Promise.resolve(listeners.onFreeMessage({
283
+ text: msg.text,
284
+ sessionId,
285
+ replyCallback: async (reply) => { await this.sendRaw(reply); },
286
+ }));
287
+ }
288
+ catch (err) {
289
+ console.error('[telegram] reply handler error:', err);
290
+ }
291
+ return;
292
+ }
293
+ if (listeners.onFreeMessage) {
294
+ // Reply to an unknown message — treat as free message
295
+ console.log(`[telegram] reply as free message: "${msg.text}"`);
296
+ await this.handleFreeMessage(msg, listeners.onFreeMessage);
297
+ }
298
+ }
299
+ capMap(map) {
300
+ if (map.size >= MAP_MAX_SIZE) {
301
+ const first = map.keys().next().value;
302
+ if (first !== undefined)
303
+ map.delete(first);
304
+ }
305
+ }
306
+ stopListening() {
307
+ this.abortController?.abort();
308
+ this.abortController = null;
309
+ }
310
+ }
311
+ exports.TelegramProvider = TelegramProvider;
312
+ //# sourceMappingURL=provider.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"provider.js","sourceRoot":"","sources":["../../../src/channels/telegram/provider.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,22 @@
1
+ import type { ChannelListeners, FreeMessage } from '../channel.js';
2
+ import type { TelegramConfig } from '../../types.js';
3
+ interface TelegramMessage {
4
+ message_id: number;
5
+ text?: string;
6
+ voice?: {
7
+ file_id: string;
8
+ duration: number;
9
+ };
10
+ reply_to_message?: {
11
+ message_id: number;
12
+ };
13
+ }
14
+ interface VoiceHandlerDeps {
15
+ config: TelegramConfig;
16
+ apiUrl: (method: string) => string;
17
+ handleReply: (msg: TelegramMessage, listeners: ChannelListeners) => Promise<void>;
18
+ handleFreeMessage: (msg: TelegramMessage, onFreeMessage: (msg: FreeMessage) => void | Promise<void>) => Promise<void>;
19
+ }
20
+ export declare function handleVoiceMessage(msg: TelegramMessage, listeners: ChannelListeners, deps: VoiceHandlerDeps): Promise<void>;
21
+ export {};
22
+ //# sourceMappingURL=voice-handler.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"voice-handler.d.ts","sourceRoot":"","sources":["../../../src/channels/telegram/voice-handler.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,gBAAgB,EAAE,WAAW,EAAE,MAAM,eAAe,CAAC;AACnE,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,gBAAgB,CAAC;AAGrD,UAAU,eAAe;IACvB,UAAU,EAAE,MAAM,CAAC;IACnB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,KAAK,CAAC,EAAE;QAAE,OAAO,EAAE,MAAM,CAAC;QAAC,QAAQ,EAAE,MAAM,CAAA;KAAE,CAAC;IAC9C,gBAAgB,CAAC,EAAE;QAAE,UAAU,EAAE,MAAM,CAAA;KAAE,CAAC;CAC3C;AAOD,UAAU,gBAAgB;IACxB,MAAM,EAAE,cAAc,CAAC;IACvB,MAAM,EAAE,CAAC,MAAM,EAAE,MAAM,KAAK,MAAM,CAAC;IACnC,WAAW,EAAE,CAAC,GAAG,EAAE,eAAe,EAAE,SAAS,EAAE,gBAAgB,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAClF,iBAAiB,EAAE,CAAC,GAAG,EAAE,eAAe,EAAE,aAAa,EAAE,CAAC,GAAG,EAAE,WAAW,KAAK,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;CACvH;AAED,wBAAsB,kBAAkB,CACtC,GAAG,EAAE,eAAe,EACpB,SAAS,EAAE,gBAAgB,EAC3B,IAAI,EAAE,gBAAgB,GACrB,OAAO,CAAC,IAAI,CAAC,CAmDf"}
@@ -0,0 +1,68 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.handleVoiceMessage = handleVoiceMessage;
4
+ const transcribe_js_1 = require("../../voice/transcribe.js");
5
+ async function handleVoiceMessage(msg, listeners, deps) {
6
+ if (!msg.voice)
7
+ return;
8
+ const { config, apiUrl } = deps;
9
+ console.log(`[telegram] voice message received (${msg.voice.duration}s)`);
10
+ // Send a "transcribing..." feedback
11
+ let statusMsgId;
12
+ try {
13
+ const res = await fetch(apiUrl('sendMessage'), {
14
+ method: 'POST',
15
+ headers: { 'Content-Type': 'application/json' },
16
+ body: JSON.stringify({
17
+ chat_id: config.chatId,
18
+ text: '🎙️ Transcribing...',
19
+ reply_to_message_id: msg.message_id,
20
+ }),
21
+ signal: AbortSignal.timeout(15000),
22
+ });
23
+ const data = (await res.json());
24
+ statusMsgId = data.result?.message_id;
25
+ }
26
+ catch { /* best effort */ }
27
+ let audioPath;
28
+ try {
29
+ audioPath = await (0, transcribe_js_1.downloadTelegramVoice)(config.botToken, msg.voice.file_id);
30
+ const result = await (0, transcribe_js_1.transcribeAudio)(audioPath, config.voiceLanguage || 'fr');
31
+ const text = result.text.trim();
32
+ console.log(`[telegram] transcribed (${result.language}): "${text}"`);
33
+ if (statusMsgId) {
34
+ await updateStatusMessage(apiUrl, config.chatId, statusMsgId, `🎙️ "${text}"`);
35
+ }
36
+ if (!text)
37
+ return;
38
+ const fakeMsg = { ...msg, text };
39
+ if (msg.reply_to_message) {
40
+ await deps.handleReply(fakeMsg, listeners);
41
+ }
42
+ else if (listeners.onFreeMessage) {
43
+ await deps.handleFreeMessage(fakeMsg, listeners.onFreeMessage);
44
+ }
45
+ }
46
+ catch (err) {
47
+ console.error('[telegram] voice transcription error:', err);
48
+ if (statusMsgId) {
49
+ await updateStatusMessage(apiUrl, config.chatId, statusMsgId, `❌ Transcription failed: ${err}`);
50
+ }
51
+ }
52
+ finally {
53
+ if (audioPath)
54
+ (0, transcribe_js_1.cleanupFile)(audioPath);
55
+ }
56
+ }
57
+ async function updateStatusMessage(apiUrl, chatId, messageId, text) {
58
+ try {
59
+ await fetch(apiUrl('editMessageText'), {
60
+ method: 'POST',
61
+ headers: { 'Content-Type': 'application/json' },
62
+ body: JSON.stringify({ chat_id: chatId, message_id: messageId, text }),
63
+ signal: AbortSignal.timeout(15000),
64
+ });
65
+ }
66
+ catch { /* best effort */ }
67
+ }
68
+ //# sourceMappingURL=voice-handler.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"voice-handler.js","sourceRoot":"","sources":["../../../src/channels/telegram/voice-handler.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ #!/usr/bin/env node
2
+ export {};
3
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/cli/index.ts"],"names":[],"mappings":""}
@@ -0,0 +1,99 @@
1
+ #!/usr/bin/env node
2
+ "use strict";
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ const commander_1 = require("commander");
5
+ const node_fs_1 = require("node:fs");
6
+ const node_path_1 = require("node:path");
7
+ const index_js_1 = require("../daemon/index.js");
8
+ const setup_js_1 = require("./setup.js");
9
+ const run_js_1 = require("./run.js");
10
+ const recover_js_1 = require("./recover.js");
11
+ const index_js_2 = require("../config/index.js");
12
+ const pkg = JSON.parse((0, node_fs_1.readFileSync)((0, node_path_1.join)(__dirname, '..', '..', 'package.json'), 'utf-8'));
13
+ const program = new commander_1.Command();
14
+ program
15
+ .name('claude-pager')
16
+ .description('Remote notification and response relay for Claude Code')
17
+ .version(pkg.version)
18
+ .enablePositionalOptions();
19
+ program
20
+ .command('start')
21
+ .description('Start the relay daemon')
22
+ .action(async () => {
23
+ await (0, index_js_1.startDaemon)();
24
+ });
25
+ program
26
+ .command('stop')
27
+ .description('Stop the relay daemon')
28
+ .action(() => {
29
+ (0, index_js_1.stopDaemon)();
30
+ });
31
+ program
32
+ .command('status')
33
+ .description('Show daemon status')
34
+ .action(async () => {
35
+ const { running, pid } = (0, index_js_1.isDaemonRunning)();
36
+ if (running) {
37
+ console.log(`Daemon is running (PID ${pid})`);
38
+ const config = (0, index_js_2.loadConfig)();
39
+ try {
40
+ const res = await fetch(`http://127.0.0.1:${config.port}/api/v1/health`);
41
+ const data = await res.json();
42
+ console.log('Health:', JSON.stringify(data, null, 2));
43
+ }
44
+ catch {
45
+ console.log('Warning: daemon PID exists but HTTP is not responding');
46
+ }
47
+ }
48
+ else {
49
+ console.log('Daemon is not running');
50
+ }
51
+ });
52
+ program
53
+ .command('pending')
54
+ .description('List pending questions')
55
+ .action(async () => {
56
+ const config = (0, index_js_2.loadConfig)();
57
+ try {
58
+ const res = await fetch(`http://127.0.0.1:${config.port}/api/v1/pending`);
59
+ const data = (await res.json());
60
+ if (data.pending.length === 0) {
61
+ console.log('No pending questions');
62
+ return;
63
+ }
64
+ for (const q of data.pending) {
65
+ const ago = Math.round((Date.now() - q.notifiedAt) / 1000);
66
+ console.log(` [${q.event.id.slice(0, 8)}] ${q.event.type} — ${q.event.project.split('/').pop()} (${ago}s ago)`);
67
+ console.log(` ${q.event.message}`);
68
+ }
69
+ }
70
+ catch {
71
+ console.log('Daemon is not running');
72
+ }
73
+ });
74
+ program
75
+ .command('setup')
76
+ .description('Configure relay and install Claude Code hooks')
77
+ .option('--server <url>', 'ntfy server URL')
78
+ .option('--topic <topic>', 'ntfy topic name')
79
+ .option('--user <user>', 'ntfy username')
80
+ .option('--password <password>', 'ntfy password')
81
+ .option('--token <token>', 'ntfy access token')
82
+ .action(async (options) => {
83
+ await (0, setup_js_1.setup)(options);
84
+ });
85
+ program
86
+ .command('recover')
87
+ .description('Recover sessions from existing tmux panes running Claude Code')
88
+ .action(() => {
89
+ (0, recover_js_1.recover)();
90
+ });
91
+ // Handle "run" manually to pass all remaining args to claude
92
+ const runIdx = process.argv.indexOf('run');
93
+ if (runIdx !== -1 && runIdx === 2) {
94
+ // Everything after "run" goes to claude
95
+ (0, run_js_1.run)(process.argv.slice(runIdx + 1));
96
+ process.exit(0);
97
+ }
98
+ program.parse();
99
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/cli/index.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export declare function recover(): void;
2
+ //# sourceMappingURL=recover.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"recover.d.ts","sourceRoot":"","sources":["../../src/cli/recover.ts"],"names":[],"mappings":"AAIA,wBAAgB,OAAO,IAAI,IAAI,CAyC9B"}
@@ -0,0 +1,45 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.recover = recover;
4
+ const node_child_process_1 = require("node:child_process");
5
+ const tracker_js_1 = require("../sessions/tracker.js");
6
+ const index_js_1 = require("../config/index.js");
7
+ function recover() {
8
+ (0, index_js_1.ensureDataDir)();
9
+ let panes;
10
+ try {
11
+ const out = (0, node_child_process_1.execFileSync)('tmux', [
12
+ 'list-panes', '-a', '-F', '#{pane_id}\t#{pane_current_command}\t#{pane_current_path}',
13
+ ], { timeout: 5000 }).toString();
14
+ panes = out.trim().split('\n').filter(Boolean).map(line => {
15
+ const [paneId, command, cwd] = line.split('\t');
16
+ return { paneId, command, cwd };
17
+ });
18
+ }
19
+ catch {
20
+ console.log('No tmux sessions found.');
21
+ return;
22
+ }
23
+ const claudePanes = panes.filter(p => p.command === 'claude');
24
+ if (claudePanes.length === 0) {
25
+ console.log('No Claude Code sessions found in tmux.');
26
+ return;
27
+ }
28
+ let created = 0;
29
+ for (const pane of claudePanes) {
30
+ const sessionId = `recovered-${pane.paneId.replace('%', '')}`;
31
+ (0, tracker_js_1.registerSession)({
32
+ sessionId,
33
+ pid: 0,
34
+ tty: '',
35
+ cwd: pane.cwd,
36
+ tmuxPane: pane.paneId,
37
+ timestamp: Date.now(),
38
+ });
39
+ const project = pane.cwd.split('/').pop();
40
+ console.log(` ${pane.paneId} → ${project} (${sessionId})`);
41
+ created++;
42
+ }
43
+ console.log(`\nRecovered ${created} session(s).`);
44
+ }
45
+ //# sourceMappingURL=recover.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"recover.js","sourceRoot":"","sources":["../../src/cli/recover.ts"],"names":[],"mappings":";;AAIA,0BAyCC;AA7CD,2DAAkD;AAClD,uDAAyD;AACzD,iDAAmD;AAEnD,SAAgB,OAAO;IACrB,IAAA,wBAAa,GAAE,CAAC;IAEhB,IAAI,KAA8D,CAAC;IACnE,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,IAAA,iCAAY,EAAC,MAAM,EAAE;YAC/B,YAAY,EAAE,IAAI,EAAE,IAAI,EAAE,2DAA2D;SACtF,EAAE,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,QAAQ,EAAE,CAAC;QAEjC,KAAK,GAAG,GAAG,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE;YACxD,MAAM,CAAC,MAAM,EAAE,OAAO,EAAE,GAAG,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;YAChD,OAAO,EAAE,MAAM,EAAE,OAAO,EAAE,GAAG,EAAE,CAAC;QAClC,CAAC,CAAC,CAAC;IACL,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,CAAC,GAAG,CAAC,yBAAyB,CAAC,CAAC;QACvC,OAAO;IACT,CAAC;IAED,MAAM,WAAW,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,QAAQ,CAAC,CAAC;IAC9D,IAAI,WAAW,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QAC7B,OAAO,CAAC,GAAG,CAAC,wCAAwC,CAAC,CAAC;QACtD,OAAO;IACT,CAAC;IAED,IAAI,OAAO,GAAG,CAAC,CAAC;IAChB,KAAK,MAAM,IAAI,IAAI,WAAW,EAAE,CAAC;QAC/B,MAAM,SAAS,GAAG,aAAa,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,CAAC,EAAE,CAAC;QAC9D,IAAA,4BAAe,EAAC;YACd,SAAS;YACT,GAAG,EAAE,CAAC;YACN,GAAG,EAAE,EAAE;YACP,GAAG,EAAE,IAAI,CAAC,GAAG;YACb,QAAQ,EAAE,IAAI,CAAC,MAAM;YACrB,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE;SACtB,CAAC,CAAC;QACH,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,CAAC;QAC1C,OAAO,CAAC,GAAG,CAAC,KAAK,IAAI,CAAC,MAAM,MAAM,OAAO,KAAK,SAAS,GAAG,CAAC,CAAC;QAC5D,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,eAAe,OAAO,cAAc,CAAC,CAAC;AACpD,CAAC"}
@@ -0,0 +1,2 @@
1
+ export declare function run(args: string[]): void;
2
+ //# sourceMappingURL=run.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"run.d.ts","sourceRoot":"","sources":["../../src/cli/run.ts"],"names":[],"mappings":"AAGA,wBAAgB,GAAG,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,CA4BxC"}
@@ -0,0 +1,32 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.run = run;
4
+ const node_child_process_1 = require("node:child_process");
5
+ const node_path_1 = require("node:path");
6
+ function run(args) {
7
+ // Generate a session name from cwd
8
+ const project = (0, node_path_1.basename)(process.cwd());
9
+ const sessionName = `claude-${project}-${process.pid}`;
10
+ // Build the claude args array (safe from injection)
11
+ const claudeArgs = ['claude', ...args];
12
+ // Check if we're already inside tmux
13
+ if (process.env.TMUX) {
14
+ // Already in tmux — just run claude directly
15
+ console.log(`Already in tmux (pane ${process.env.TMUX_PANE}), launching Claude Code...`);
16
+ (0, node_child_process_1.execFileSync)('claude', args, { stdio: 'inherit' });
17
+ return;
18
+ }
19
+ console.log(`Launching Claude Code in tmux session "${sessionName}"...`);
20
+ try {
21
+ // Create a new tmux session and run claude inside it, then attach
22
+ (0, node_child_process_1.execFileSync)('tmux', [
23
+ 'new-session',
24
+ '-s', sessionName,
25
+ ...claudeArgs,
26
+ ], { stdio: 'inherit' });
27
+ }
28
+ catch {
29
+ // tmux returns non-zero when the session ends normally
30
+ }
31
+ }
32
+ //# sourceMappingURL=run.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"run.js","sourceRoot":"","sources":["../../src/cli/run.ts"],"names":[],"mappings":";;AAGA,kBA4BC;AA/BD,2DAAkD;AAClD,yCAAqC;AAErC,SAAgB,GAAG,CAAC,IAAc;IAChC,mCAAmC;IACnC,MAAM,OAAO,GAAG,IAAA,oBAAQ,EAAC,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC;IACxC,MAAM,WAAW,GAAG,UAAU,OAAO,IAAI,OAAO,CAAC,GAAG,EAAE,CAAC;IAEvD,oDAAoD;IACpD,MAAM,UAAU,GAAG,CAAC,QAAQ,EAAE,GAAG,IAAI,CAAC,CAAC;IAEvC,qCAAqC;IACrC,IAAI,OAAO,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC;QACrB,6CAA6C;QAC7C,OAAO,CAAC,GAAG,CAAC,yBAAyB,OAAO,CAAC,GAAG,CAAC,SAAS,6BAA6B,CAAC,CAAC;QACzF,IAAA,iCAAY,EAAC,QAAQ,EAAE,IAAI,EAAE,EAAE,KAAK,EAAE,SAAS,EAAE,CAAC,CAAC;QACnD,OAAO;IACT,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,0CAA0C,WAAW,MAAM,CAAC,CAAC;IAEzE,IAAI,CAAC;QACH,kEAAkE;QAClE,IAAA,iCAAY,EAAC,MAAM,EAAE;YACnB,aAAa;YACb,IAAI,EAAE,WAAW;YACjB,GAAG,UAAU;SACd,EAAE,EAAE,KAAK,EAAE,SAAS,EAAE,CAAC,CAAC;IAC3B,CAAC;IAAC,MAAM,CAAC;QACP,uDAAuD;IACzD,CAAC;AACH,CAAC"}