@hopper-agent/core 0.1.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (55) hide show
  1. package/LICENSE +179 -0
  2. package/dist/.tsbuildinfo +1 -0
  3. package/dist/src/channel-registry.d.ts +26 -0
  4. package/dist/src/channel-registry.d.ts.map +1 -0
  5. package/dist/src/channel-registry.js +100 -0
  6. package/dist/src/channel-registry.js.map +1 -0
  7. package/dist/src/channel-router.d.ts +28 -0
  8. package/dist/src/channel-router.d.ts.map +1 -0
  9. package/dist/src/channel-router.js +52 -0
  10. package/dist/src/channel-router.js.map +1 -0
  11. package/dist/src/cron-manager.d.ts +70 -0
  12. package/dist/src/cron-manager.d.ts.map +1 -0
  13. package/dist/src/cron-manager.js +301 -0
  14. package/dist/src/cron-manager.js.map +1 -0
  15. package/dist/src/event-bus.d.ts +99 -0
  16. package/dist/src/event-bus.d.ts.map +1 -0
  17. package/dist/src/event-bus.js +56 -0
  18. package/dist/src/event-bus.js.map +1 -0
  19. package/dist/src/heartbeat-delivery.d.ts +27 -0
  20. package/dist/src/heartbeat-delivery.d.ts.map +1 -0
  21. package/dist/src/heartbeat-delivery.js +37 -0
  22. package/dist/src/heartbeat-delivery.js.map +1 -0
  23. package/dist/src/heartbeat-executor.d.ts +35 -0
  24. package/dist/src/heartbeat-executor.d.ts.map +1 -0
  25. package/dist/src/heartbeat-executor.js +174 -0
  26. package/dist/src/heartbeat-executor.js.map +1 -0
  27. package/dist/src/index.d.ts +26 -0
  28. package/dist/src/index.d.ts.map +1 -0
  29. package/dist/src/index.js +15 -0
  30. package/dist/src/index.js.map +1 -0
  31. package/dist/src/permission.d.ts +21 -0
  32. package/dist/src/permission.d.ts.map +1 -0
  33. package/dist/src/permission.js +69 -0
  34. package/dist/src/permission.js.map +1 -0
  35. package/dist/src/reminder-parser.d.ts +12 -0
  36. package/dist/src/reminder-parser.d.ts.map +1 -0
  37. package/dist/src/reminder-parser.js +164 -0
  38. package/dist/src/reminder-parser.js.map +1 -0
  39. package/dist/src/session-store.d.ts +32 -0
  40. package/dist/src/session-store.d.ts.map +1 -0
  41. package/dist/src/session-store.js +112 -0
  42. package/dist/src/session-store.js.map +1 -0
  43. package/dist/src/settings.d.ts +41 -0
  44. package/dist/src/settings.d.ts.map +1 -0
  45. package/dist/src/settings.js +128 -0
  46. package/dist/src/settings.js.map +1 -0
  47. package/dist/src/telegram-gateway.d.ts +35 -0
  48. package/dist/src/telegram-gateway.d.ts.map +1 -0
  49. package/dist/src/telegram-gateway.js +145 -0
  50. package/dist/src/telegram-gateway.js.map +1 -0
  51. package/dist/src/turn-loop.d.ts +106 -0
  52. package/dist/src/turn-loop.d.ts.map +1 -0
  53. package/dist/src/turn-loop.js +288 -0
  54. package/dist/src/turn-loop.js.map +1 -0
  55. package/package.json +33 -0
@@ -0,0 +1,112 @@
1
+ import fs from 'node:fs';
2
+ import path from 'node:path';
3
+ import os from 'node:os';
4
+ const ULID_ALPHABET = '0123456789ABCDEFGHJKMNPQRSTVWXYZ';
5
+ export function ulid(timestamp) {
6
+ const ts = timestamp ?? Date.now();
7
+ const timeBuf = new BigUint64Array(1);
8
+ const randBuf = new Uint8Array(16);
9
+ crypto.getRandomValues(randBuf);
10
+ const view = new DataView(timeBuf.buffer);
11
+ view.setBigUint64(0, BigInt(ts), false);
12
+ let result = '';
13
+ const timeBytes = new Uint8Array(timeBuf.buffer);
14
+ for (let i = 0; i < 10; i++) {
15
+ const idx = 9 - i;
16
+ const b = timeBytes[idx];
17
+ result += ULID_ALPHABET[b & 0x1f];
18
+ }
19
+ for (let i = 0; i < 16; i++) {
20
+ const b = randBuf[i];
21
+ result += ULID_ALPHABET[b & 0x1f];
22
+ }
23
+ return result;
24
+ }
25
+ export class SessionStore {
26
+ baseDir;
27
+ constructor(baseDir = path.join(os.homedir(), '.hopper-agent', 'sessions')) {
28
+ this.baseDir = baseDir;
29
+ }
30
+ sessionPath(id) {
31
+ return path.join(this.baseDir, id);
32
+ }
33
+ eventsPath(id) {
34
+ return path.join(this.sessionPath(id), 'events.jsonl');
35
+ }
36
+ metadataPath(id) {
37
+ return path.join(this.sessionPath(id), 'metadata.json');
38
+ }
39
+ getFiles(id) {
40
+ return {
41
+ path: this.sessionPath(id),
42
+ events: this.eventsPath(id),
43
+ metadata: this.metadataPath(id),
44
+ };
45
+ }
46
+ create(cwd, model, provider) {
47
+ const id = ulid();
48
+ const now = Date.now();
49
+ const info = { id, cwd, model, provider, createdAt: now, updatedAt: now };
50
+ const dir = this.sessionPath(id);
51
+ fs.mkdirSync(dir, { recursive: true });
52
+ fs.writeFileSync(this.metadataPath(id), JSON.stringify(info, null, 2) + '\n');
53
+ fs.closeSync(fs.openSync(this.eventsPath(id), 'a'));
54
+ return info;
55
+ }
56
+ load(id) {
57
+ const metaPath = this.metadataPath(id);
58
+ if (!fs.existsSync(metaPath))
59
+ return undefined;
60
+ return JSON.parse(fs.readFileSync(metaPath, 'utf-8'));
61
+ }
62
+ loadEvents(id) {
63
+ const eventsPath = this.eventsPath(id);
64
+ if (!fs.existsSync(eventsPath))
65
+ return [];
66
+ const content = fs.readFileSync(eventsPath, 'utf-8').trim();
67
+ if (!content)
68
+ return [];
69
+ return content
70
+ .split('\n')
71
+ .filter(Boolean)
72
+ .map((line) => JSON.parse(line));
73
+ }
74
+ appendEvent(id, event) {
75
+ this.appendEvents(id, [event]);
76
+ }
77
+ appendEvents(id, events) {
78
+ const eventsPath = this.eventsPath(id);
79
+ const data = events.map((e) => JSON.stringify(e)).join('\n') + '\n';
80
+ fs.appendFileSync(eventsPath, data, 'utf-8');
81
+ this.touch(id);
82
+ }
83
+ list() {
84
+ if (!fs.existsSync(this.baseDir))
85
+ return [];
86
+ const entries = fs.readdirSync(this.baseDir, { withFileTypes: true });
87
+ const sessions = [];
88
+ for (const entry of entries) {
89
+ if (!entry.isDirectory())
90
+ continue;
91
+ const info = this.load(entry.name);
92
+ if (info)
93
+ sessions.push(info);
94
+ }
95
+ return sessions.sort((a, b) => b.updatedAt - a.updatedAt);
96
+ }
97
+ remove(id) {
98
+ const dir = this.sessionPath(id);
99
+ if (fs.existsSync(dir)) {
100
+ fs.rmSync(dir, { recursive: true, force: true });
101
+ }
102
+ }
103
+ touch(id) {
104
+ const metaPath = this.metadataPath(id);
105
+ if (fs.existsSync(metaPath)) {
106
+ const info = JSON.parse(fs.readFileSync(metaPath, 'utf-8'));
107
+ info.updatedAt = Date.now();
108
+ fs.writeFileSync(metaPath, JSON.stringify(info, null, 2) + '\n');
109
+ }
110
+ }
111
+ }
112
+ //# sourceMappingURL=session-store.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"session-store.js","sourceRoot":"","sources":["../../src/session-store.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,41 @@
1
+ export interface HopperSettings {
2
+ model: string;
3
+ effort: 'low' | 'medium' | 'high' | 'max' | 'auto';
4
+ mode: 'plan' | 'edit' | 'auto' | 'yolo';
5
+ theme: string;
6
+ statusline: boolean;
7
+ debug: boolean;
8
+ MODEL_PROVIDER?: string;
9
+ MODEL_URL?: string;
10
+ MODEL_API_KEY?: string;
11
+ OPENAI_API_KEY?: string;
12
+ OPENAI_URL?: string;
13
+ ANTHROPIC_API_KEY?: string;
14
+ ANTHROPIC_URL?: string;
15
+ OPENROUTER_API_KEY?: string;
16
+ OPENROUTER_URL?: string;
17
+ TELEGRAM_BOT_TOKEN?: string;
18
+ TELEGRAM_USER_ID?: string;
19
+ TELEGRAM_CHAT_ID?: string;
20
+ TELEGRAM_ALLOW_GROUPS?: boolean;
21
+ CHANNEL_TAB_ACTIVE?: string;
22
+ BRAVE_SEARCH_API_KEY?: string;
23
+ MCP_SERVERS?: Record<string, unknown>;
24
+ TOOLS_PER_CALL?: number;
25
+ WEBSEARCH_PER_CALL?: number;
26
+ HEARTBEAT?: 'on' | 'off';
27
+ HEARTBEAT_TIMES?: string;
28
+ HEARTBEAT_DAILY?: string;
29
+ HEARTBEAT_WEEKLY?: string;
30
+ HEARTBEAT_MONTHLY?: string;
31
+ }
32
+ export declare const DEFAULT_SETTINGS: HopperSettings;
33
+ export declare class SettingsManager {
34
+ private settings;
35
+ constructor();
36
+ load(): HopperSettings;
37
+ save(): void;
38
+ update(partial: Partial<HopperSettings>): HopperSettings;
39
+ get(): HopperSettings;
40
+ }
41
+ //# sourceMappingURL=settings.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"settings.d.ts","sourceRoot":"","sources":["../../src/settings.ts"],"names":[],"mappings":"AAIA,MAAM,WAAW,cAAc;IAC7B,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,EAAE,KAAK,GAAG,QAAQ,GAAG,MAAM,GAAG,KAAK,GAAG,MAAM,CAAC;IACnD,IAAI,EAAE,MAAM,GAAG,MAAM,GAAG,MAAM,GAAG,MAAM,CAAC;IACxC,KAAK,EAAE,MAAM,CAAC;IACd,UAAU,EAAE,OAAO,CAAC;IACpB,KAAK,EAAE,OAAO,CAAC;IAEf,cAAc,CAAC,EAAE,MAAM,CAAC;IAExB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,aAAa,CAAC,EAAE,MAAM,CAAC;IAEvB,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,UAAU,CAAC,EAAE,MAAM,CAAC;IAEpB,iBAAiB,CAAC,EAAE,MAAM,CAAC;IAC3B,aAAa,CAAC,EAAE,MAAM,CAAC;IAEvB,kBAAkB,CAAC,EAAE,MAAM,CAAC;IAC5B,cAAc,CAAC,EAAE,MAAM,CAAC;IAExB,kBAAkB,CAAC,EAAE,MAAM,CAAC;IAC5B,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,qBAAqB,CAAC,EAAE,OAAO,CAAC;IAChC,kBAAkB,CAAC,EAAE,MAAM,CAAC;IAE5B,oBAAoB,CAAC,EAAE,MAAM,CAAC;IAE9B,WAAW,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAEtC,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,kBAAkB,CAAC,EAAE,MAAM,CAAC;IAE5B,SAAS,CAAC,EAAE,IAAI,GAAG,KAAK,CAAC;IACzB,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,iBAAiB,CAAC,EAAE,MAAM,CAAC;CAC5B;AAED,eAAO,MAAM,gBAAgB,EAAE,cAoB9B,CAAC;AAgBF,qBAAa,eAAe;IAC1B,OAAO,CAAC,QAAQ,CAAiB;;IAMjC,IAAI,IAAI,cAAc;IAiEtB,IAAI,IAAI,IAAI;IAOZ,MAAM,CAAC,OAAO,EAAE,OAAO,CAAC,cAAc,CAAC,GAAG,cAAc;IAMxD,GAAG,IAAI,cAAc;CAGtB"}
@@ -0,0 +1,128 @@
1
+ import { existsSync, mkdirSync, readFileSync, writeFileSync } from 'node:fs';
2
+ import { homedir } from 'node:os';
3
+ import { join } from 'node:path';
4
+ export const DEFAULT_SETTINGS = {
5
+ model: 'claude-sonnet-4-6',
6
+ effort: 'auto',
7
+ mode: 'auto',
8
+ theme: 'tokyo-night',
9
+ statusline: true,
10
+ debug: false,
11
+ BRAVE_SEARCH_API_KEY: '',
12
+ TELEGRAM_BOT_TOKEN: '',
13
+ TELEGRAM_USER_ID: '',
14
+ TELEGRAM_CHAT_ID: '',
15
+ TELEGRAM_ALLOW_GROUPS: false,
16
+ CHANNEL_TAB_ACTIVE: 'main',
17
+ TOOLS_PER_CALL: 10,
18
+ WEBSEARCH_PER_CALL: 5,
19
+ HEARTBEAT: 'off',
20
+ HEARTBEAT_TIMES: '',
21
+ HEARTBEAT_DAILY: '6:00',
22
+ HEARTBEAT_WEEKLY: 'monday@6:00',
23
+ HEARTBEAT_MONTHLY: '1@6:00',
24
+ };
25
+ function migrateMode(raw) {
26
+ if (typeof raw !== 'string')
27
+ return DEFAULT_SETTINGS.mode;
28
+ switch (raw) {
29
+ case 'plan':
30
+ case 'edit':
31
+ case 'auto':
32
+ case 'yolo': return raw;
33
+ case 'auto-edit': return 'edit';
34
+ case 'manual': return 'auto';
35
+ case 'full-auto': return 'yolo';
36
+ default: return DEFAULT_SETTINGS.mode;
37
+ }
38
+ }
39
+ const CONFIG_DIR = join(homedir(), '.hopper-agent');
40
+ const SETTINGS_FILE = join(CONFIG_DIR, 'settings.json');
41
+ export class SettingsManager {
42
+ settings;
43
+ constructor() {
44
+ this.settings = { ...DEFAULT_SETTINGS };
45
+ }
46
+ load() {
47
+ this.settings = { ...DEFAULT_SETTINGS };
48
+ if (existsSync(SETTINGS_FILE)) {
49
+ try {
50
+ const raw = readFileSync(SETTINGS_FILE, 'utf-8');
51
+ const parsed = JSON.parse(raw);
52
+ if (parsed && typeof parsed === 'object') {
53
+ // Accept MODEL_URL / MODEL_API_KEY regardless of casing (users
54
+ // commonly write model_url, ModelUrl, MODEL_API, etc.).
55
+ const pick = (...keys) => {
56
+ for (const key of keys) {
57
+ const direct = parsed[key];
58
+ if (typeof direct === 'string' && direct.length > 0)
59
+ return direct;
60
+ }
61
+ const normalize = (s) => s.replace(/[_-]/g, '').toLowerCase();
62
+ const targets = keys.map(normalize);
63
+ for (const [k, v] of Object.entries(parsed)) {
64
+ if (typeof v !== 'string' || v.length === 0)
65
+ continue;
66
+ if (targets.includes(normalize(k)))
67
+ return v;
68
+ }
69
+ return undefined;
70
+ };
71
+ const merged = {
72
+ model: parsed.model ?? DEFAULT_SETTINGS.model,
73
+ effort: parsed.effort ?? DEFAULT_SETTINGS.effort,
74
+ mode: migrateMode(parsed.mode),
75
+ theme: parsed.theme ?? DEFAULT_SETTINGS.theme,
76
+ statusline: parsed.statusline ?? DEFAULT_SETTINGS.statusline,
77
+ debug: parsed.debug ?? DEFAULT_SETTINGS.debug,
78
+ BRAVE_SEARCH_API_KEY: pick('BRAVE_SEARCH_API_KEY', 'brave_search_api_key', 'braveSearchApiKey', 'BRAVE_API_KEY'),
79
+ MODEL_URL: pick('MODEL_URL', 'model_url', 'modelUrl', 'MODEL_BASE_URL', 'baseUrl'),
80
+ MODEL_API_KEY: pick('MODEL_API_KEY', 'model_api_key', 'MODEL_API', 'model_api', 'modelApiKey', 'apiKey', 'API_KEY'),
81
+ MODEL_PROVIDER: pick('MODEL_PROVIDER', 'model_provider', 'modelProvider'),
82
+ OPENAI_API_KEY: pick('OPENAI_API_KEY', 'openai_api_key', 'openaiApiKey'),
83
+ OPENAI_URL: pick('OPENAI_URL', 'openai_url', 'openaiBaseUrl', 'OPENAI_BASE_URL', 'openai_base_url'),
84
+ ANTHROPIC_API_KEY: pick('ANTHROPIC_API_KEY', 'anthropic_api_key', 'anthropicApiKey'),
85
+ ANTHROPIC_URL: pick('ANTHROPIC_URL', 'anthropic_url', 'anthropicUrl'),
86
+ OPENROUTER_API_KEY: pick('OPENROUTER_API_KEY', 'openrouter_api_key', 'openrouterApiKey'),
87
+ OPENROUTER_URL: pick('OPENROUTER_URL', 'openrouter_url', 'openrouterUrl'),
88
+ TELEGRAM_BOT_TOKEN: pick('TELEGRAM_BOT_TOKEN', 'telegram_bot_token', 'telegramBotToken'),
89
+ TELEGRAM_USER_ID: pick('TELEGRAM_USER_ID', 'telegram_user_id', 'telegramUserId'),
90
+ TELEGRAM_CHAT_ID: pick('TELEGRAM_CHAT_ID', 'telegram_chat_id', 'telegramChatId'),
91
+ TELEGRAM_ALLOW_GROUPS: parsed.TELEGRAM_ALLOW_GROUPS ?? false,
92
+ CHANNEL_TAB_ACTIVE: parsed.CHANNEL_TAB_ACTIVE ?? 'main',
93
+ MCP_SERVERS: parsed.MCP_SERVERS ?? parsed.mcp_servers,
94
+ TOOLS_PER_CALL: (typeof parsed.TOOLS_PER_CALL === 'number' && parsed.TOOLS_PER_CALL > 0)
95
+ ? parsed.TOOLS_PER_CALL : DEFAULT_SETTINGS.TOOLS_PER_CALL,
96
+ WEBSEARCH_PER_CALL: (typeof parsed.WEBSEARCH_PER_CALL === 'number' && parsed.WEBSEARCH_PER_CALL > 0)
97
+ ? parsed.WEBSEARCH_PER_CALL : DEFAULT_SETTINGS.WEBSEARCH_PER_CALL,
98
+ HEARTBEAT: parsed.HEARTBEAT ?? DEFAULT_SETTINGS.HEARTBEAT,
99
+ HEARTBEAT_TIMES: parsed.HEARTBEAT_TIMES ?? parsed.HEARTBEAT_HOURLY ?? DEFAULT_SETTINGS.HEARTBEAT_TIMES,
100
+ HEARTBEAT_DAILY: parsed.HEARTBEAT_DAILY ?? DEFAULT_SETTINGS.HEARTBEAT_DAILY,
101
+ HEARTBEAT_WEEKLY: parsed.HEARTBEAT_WEEKLY ?? DEFAULT_SETTINGS.HEARTBEAT_WEEKLY,
102
+ HEARTBEAT_MONTHLY: parsed.HEARTBEAT_MONTHLY ?? DEFAULT_SETTINGS.HEARTBEAT_MONTHLY,
103
+ };
104
+ this.settings = merged;
105
+ }
106
+ }
107
+ catch {
108
+ // ignore parse errors, use defaults
109
+ }
110
+ }
111
+ return this.settings;
112
+ }
113
+ save() {
114
+ if (!existsSync(CONFIG_DIR)) {
115
+ mkdirSync(CONFIG_DIR, { recursive: true });
116
+ }
117
+ writeFileSync(SETTINGS_FILE, JSON.stringify(this.settings, null, 2) + '\n');
118
+ }
119
+ update(partial) {
120
+ this.settings = { ...this.settings, ...partial };
121
+ this.save();
122
+ return this.settings;
123
+ }
124
+ get() {
125
+ return { ...this.settings };
126
+ }
127
+ }
128
+ //# sourceMappingURL=settings.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"settings.js","sourceRoot":"","sources":["../../src/settings.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,35 @@
1
+ export interface TelegramMessageContext {
2
+ text: string;
3
+ chatId: string;
4
+ userId: string;
5
+ isGroup: boolean;
6
+ chatTitle?: string;
7
+ }
8
+ export interface TelegramGatewayConfig {
9
+ botToken: string;
10
+ allowedUserId: string;
11
+ onUserMessage: (ctx: TelegramMessageContext) => void;
12
+ /** Called when user taps Approve/Deny on an approval inline keyboard. */
13
+ onApprovalDecision?: (toolCallId: string, approved: boolean) => void;
14
+ onError?: (error: Error) => void;
15
+ }
16
+ export declare class TelegramGateway {
17
+ private bot;
18
+ private isRunning;
19
+ private config;
20
+ private typingTimers;
21
+ constructor(config: TelegramGatewayConfig);
22
+ private setupHandlers;
23
+ sendMessage(chatId: string, text: string): Promise<void>;
24
+ /**
25
+ * Send an inline keyboard approval request to a Telegram chat.
26
+ */
27
+ sendApprovalRequest(chatId: string, toolCallId: string, toolName: string, inputSummary: string): Promise<void>;
28
+ start(): Promise<void>;
29
+ stop(): Promise<void>;
30
+ setOnApprovalDecision(fn: (toolCallId: string, approved: boolean) => void): void;
31
+ get isAlive(): boolean;
32
+ startTyping(chatId: string): void;
33
+ stopTyping(chatId: string): void;
34
+ }
35
+ //# sourceMappingURL=telegram-gateway.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"telegram-gateway.d.ts","sourceRoot":"","sources":["../../src/telegram-gateway.ts"],"names":[],"mappings":"AAEA,MAAM,WAAW,sBAAsB;IACrC,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC;IACf,MAAM,EAAE,MAAM,CAAC;IACf,OAAO,EAAE,OAAO,CAAC;IACjB,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB;AAED,MAAM,WAAW,qBAAqB;IACpC,QAAQ,EAAE,MAAM,CAAC;IACjB,aAAa,EAAE,MAAM,CAAC;IACtB,aAAa,EAAE,CAAC,GAAG,EAAE,sBAAsB,KAAK,IAAI,CAAC;IACrD,yEAAyE;IACzE,kBAAkB,CAAC,EAAE,CAAC,UAAU,EAAE,MAAM,EAAE,QAAQ,EAAE,OAAO,KAAK,IAAI,CAAC;IACrE,OAAO,CAAC,EAAE,CAAC,KAAK,EAAE,KAAK,KAAK,IAAI,CAAC;CAClC;AAED,qBAAa,eAAe;IAC1B,OAAO,CAAC,GAAG,CAAe;IAC1B,OAAO,CAAC,SAAS,CAAS;IAC1B,OAAO,CAAC,MAAM,CAAwB;IACtC,OAAO,CAAC,YAAY,CAA0C;gBAElD,MAAM,EAAE,qBAAqB;IAMzC,OAAO,CAAC,aAAa;IAoDf,WAAW,CAAC,MAAM,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAU9D;;OAEG;IACG,mBAAmB,CACvB,MAAM,EAAE,MAAM,EACd,UAAU,EAAE,MAAM,EAClB,QAAQ,EAAE,MAAM,EAChB,YAAY,EAAE,MAAM,GACnB,OAAO,CAAC,IAAI,CAAC;IAyBV,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAOtB,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAY3B,qBAAqB,CAAC,EAAE,EAAE,CAAC,UAAU,EAAE,MAAM,EAAE,QAAQ,EAAE,OAAO,KAAK,IAAI,GAAG,IAAI;IAIhF,IAAI,OAAO,IAAI,OAAO,CAErB;IAED,WAAW,CAAC,MAAM,EAAE,MAAM,GAAG,IAAI;IAmBjC,UAAU,CAAC,MAAM,EAAE,MAAM,GAAG,IAAI;CAOjC"}
@@ -0,0 +1,145 @@
1
+ import { Bot } from 'grammy';
2
+ export class TelegramGateway {
3
+ bot;
4
+ isRunning = false;
5
+ config;
6
+ typingTimers = new Map();
7
+ constructor(config) {
8
+ this.config = config;
9
+ this.bot = new Bot(config.botToken);
10
+ this.setupHandlers();
11
+ }
12
+ setupHandlers() {
13
+ // Only process text messages
14
+ this.bot.on('message:text', async (ctx) => {
15
+ const msg = ctx.message.text;
16
+ const userId = String(ctx.from?.id ?? '');
17
+ const chatId = String(ctx.chat.id);
18
+ const chatType = ctx.chat.type;
19
+ const chatTitle = ctx.chat.title;
20
+ // User ID filtering — gatekeeper logic
21
+ if (userId !== this.config.allowedUserId) {
22
+ return;
23
+ }
24
+ this.config.onUserMessage({
25
+ text: msg,
26
+ chatId,
27
+ userId,
28
+ isGroup: chatType === 'group' || chatType === 'supergroup',
29
+ chatTitle,
30
+ });
31
+ // Show typing indicator — Telegram's indicator expires after ~5s,
32
+ // so we keep resending every 4s until stopTyping is called.
33
+ this.startTyping(chatId);
34
+ });
35
+ // Handle inline keyboard button taps for approval decisions
36
+ this.bot.on('callback_query', async (ctx) => {
37
+ const data = ctx.callbackQuery.data;
38
+ if (!data) {
39
+ await ctx.answerCallbackQuery();
40
+ return;
41
+ }
42
+ const pipeIndex = data.indexOf('|');
43
+ if (pipeIndex === -1) {
44
+ await ctx.answerCallbackQuery();
45
+ return;
46
+ }
47
+ const action = data.slice(0, pipeIndex);
48
+ const toolCallId = data.slice(pipeIndex + 1);
49
+ const approved = action === 'approve';
50
+ this.config.onApprovalDecision?.(toolCallId, approved);
51
+ await ctx.answerCallbackQuery(); // dismiss the loading spinner
52
+ });
53
+ // Catch-all error handler
54
+ this.bot.catch((err) => {
55
+ console.error('[telegram-gateway] Error:', err.error);
56
+ this.config.onError?.(err.error);
57
+ });
58
+ }
59
+ async sendMessage(chatId, text) {
60
+ try {
61
+ await this.bot.api.sendMessage(chatId, text);
62
+ }
63
+ catch (err) {
64
+ const msg = err instanceof Error ? err.message : String(err);
65
+ console.error(`[telegram-gateway] sendMessage failed for chat ${chatId}: ${msg}`);
66
+ throw err;
67
+ }
68
+ }
69
+ /**
70
+ * Send an inline keyboard approval request to a Telegram chat.
71
+ */
72
+ async sendApprovalRequest(chatId, toolCallId, toolName, inputSummary) {
73
+ // Strip non-printable chars and escape HTML to avoid Telegram parse failures
74
+ const safe = (s) => s.replace(/[^\x20-\x7E]/g, '').replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;');
75
+ const cleanSummary = safe(inputSummary).replace(/[\r\n]/g, ' ').slice(0, 200);
76
+ const text = '[Tool approval needed]\n\n<b>' + safe(toolName) + '</b>\n' + cleanSummary;
77
+ try {
78
+ await this.bot.api.sendMessage(chatId, text, {
79
+ reply_markup: {
80
+ inline_keyboard: [
81
+ [
82
+ { text: 'Allow', callback_data: 'approve|' + toolCallId },
83
+ { text: 'Deny', callback_data: 'deny|' + toolCallId },
84
+ ],
85
+ ],
86
+ },
87
+ parse_mode: 'HTML',
88
+ });
89
+ }
90
+ catch (err) {
91
+ const msg = err instanceof Error ? err.message : String(err);
92
+ console.error('[telegram-gateway] sendApprovalRequest failed for chat ' + chatId + ': ' + msg);
93
+ throw err;
94
+ }
95
+ }
96
+ async start() {
97
+ if (this.isRunning)
98
+ return;
99
+ this.isRunning = true;
100
+ await this.bot.start();
101
+ console.log('[telegram-gateway] Bot started (polling)');
102
+ }
103
+ async stop() {
104
+ if (!this.isRunning)
105
+ return;
106
+ this.isRunning = false;
107
+ // Clear all typing timers
108
+ for (const timer of this.typingTimers.values()) {
109
+ clearInterval(timer);
110
+ }
111
+ this.typingTimers.clear();
112
+ await this.bot.stop();
113
+ console.log('[telegram-gateway] Bot stopped');
114
+ }
115
+ setOnApprovalDecision(fn) {
116
+ this.config.onApprovalDecision = fn;
117
+ }
118
+ get isAlive() {
119
+ return this.isRunning;
120
+ }
121
+ startTyping(chatId) {
122
+ // If already typing, don't start another timer
123
+ if (this.typingTimers.has(chatId))
124
+ return;
125
+ console.error(`[telegram-gateway] startTyping for ${chatId}, isRunning=${this.isRunning}`);
126
+ const send = () => {
127
+ if (this.isRunning) {
128
+ this.bot.api.sendChatAction(chatId, 'typing').catch((err) => {
129
+ console.error(`[telegram-gateway] sendChatAction failed for ${chatId}:`, err?.message || err);
130
+ });
131
+ }
132
+ };
133
+ send(); // Send immediately
134
+ const interval = setInterval(send, 4000);
135
+ this.typingTimers.set(chatId, interval);
136
+ }
137
+ stopTyping(chatId) {
138
+ const timer = this.typingTimers.get(chatId);
139
+ if (timer) {
140
+ clearInterval(timer);
141
+ this.typingTimers.delete(chatId);
142
+ }
143
+ }
144
+ }
145
+ //# sourceMappingURL=telegram-gateway.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"telegram-gateway.js","sourceRoot":"","sources":["../../src/telegram-gateway.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,106 @@
1
+ import { EventBus, type Event } from './event-bus.js';
2
+ import { SessionStore } from './session-store.js';
3
+ import { type ApprovalMode } from './permission.js';
4
+ import type { HopperSettings } from './settings.js';
5
+ export type ReasoningEffort = 'low' | 'medium' | 'high' | 'max' | 'auto';
6
+ export interface ProviderStream {
7
+ name: string;
8
+ stream(args: {
9
+ messages: Array<{
10
+ role: string;
11
+ content: unknown;
12
+ }>;
13
+ tools?: Array<{
14
+ name: string;
15
+ description: string;
16
+ inputSchema: unknown;
17
+ }>;
18
+ model?: string;
19
+ system?: string;
20
+ signal?: AbortSignal;
21
+ effort?: ReasoningEffort;
22
+ }): AsyncIterable<ProviderEvent>;
23
+ /** Non-streaming call — used for quick evaluations (e.g. harm-check). */
24
+ check(prompt: string, args?: {
25
+ model?: string;
26
+ system?: string;
27
+ }): Promise<string>;
28
+ }
29
+ export interface Tool {
30
+ definition: {
31
+ name: string;
32
+ description: string;
33
+ inputSchema: unknown;
34
+ };
35
+ execute: (input: Record<string, unknown>, settings: any) => Promise<{
36
+ output: unknown;
37
+ error?: string;
38
+ }>;
39
+ }
40
+ export type ProviderEvent = {
41
+ type: 'text-delta';
42
+ text: string;
43
+ } | {
44
+ type: 'thinking-block';
45
+ thinking: string;
46
+ signature: string;
47
+ } | {
48
+ type: 'tool-call';
49
+ id: string;
50
+ name: string;
51
+ input: Record<string, unknown>;
52
+ } | {
53
+ type: 'tool-result-request';
54
+ callId: string;
55
+ } | {
56
+ type: 'usage';
57
+ inputTokens: number;
58
+ outputTokens: number;
59
+ cacheReadTokens?: number;
60
+ cacheWriteTokens?: number;
61
+ } | {
62
+ type: 'stop';
63
+ reason: string;
64
+ };
65
+ export interface TurnLoopConfig {
66
+ provider: ProviderStream;
67
+ model: string;
68
+ tools: Tool[];
69
+ cwd: string;
70
+ settings: HopperSettings;
71
+ approvalMode?: ApprovalMode;
72
+ permissions?: Record<string, string[]>;
73
+ maxTurns?: number;
74
+ system?: string;
75
+ effort?: ReasoningEffort;
76
+ /** If provided, resume this session instead of creating a new one. */
77
+ sessionId?: string;
78
+ /** Called when a tool call needs interactive approval. Return true to allow, false to deny. */
79
+ onApprovalAsk?: (req: {
80
+ name: string;
81
+ input: Record<string, unknown>;
82
+ reason: string;
83
+ }) => Promise<boolean>;
84
+ }
85
+ export interface TurnLoopResult {
86
+ events: Event[];
87
+ sessionId: string;
88
+ reason: string;
89
+ }
90
+ export declare class TurnLoop {
91
+ private bus;
92
+ private store;
93
+ private permission;
94
+ private config;
95
+ private abort;
96
+ private abortController;
97
+ private messages;
98
+ private activeSessionId;
99
+ constructor(config: TurnLoopConfig, store?: SessionStore);
100
+ get eventBus(): EventBus;
101
+ getSessionId(): string | undefined;
102
+ cancel(): void;
103
+ private evaluateHarm;
104
+ run(prompt: string): Promise<TurnLoopResult>;
105
+ }
106
+ //# sourceMappingURL=turn-loop.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"turn-loop.d.ts","sourceRoot":"","sources":["../../src/turn-loop.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,KAAK,KAAK,EAAE,MAAM,gBAAgB,CAAC;AACtD,OAAO,EAAE,YAAY,EAAoB,MAAM,oBAAoB,CAAC;AACpE,OAAO,EAAoB,KAAK,YAAY,EAAE,MAAM,iBAAiB,CAAC;AACtE,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,eAAe,CAAC;AAEpD,MAAM,MAAM,eAAe,GAAG,KAAK,GAAG,QAAQ,GAAG,MAAM,GAAG,KAAK,GAAG,MAAM,CAAC;AAEzE,MAAM,WAAW,cAAc;IAC7B,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,CAAC,IAAI,EAAE;QACX,QAAQ,EAAE,KAAK,CAAC;YAAE,IAAI,EAAE,MAAM,CAAC;YAAC,OAAO,EAAE,OAAO,CAAA;SAAE,CAAC,CAAC;QACpD,KAAK,CAAC,EAAE,KAAK,CAAC;YAAE,IAAI,EAAE,MAAM,CAAC;YAAC,WAAW,EAAE,MAAM,CAAC;YAAC,WAAW,EAAE,OAAO,CAAA;SAAE,CAAC,CAAC;QAC3E,KAAK,CAAC,EAAE,MAAM,CAAC;QACf,MAAM,CAAC,EAAE,MAAM,CAAC;QAChB,MAAM,CAAC,EAAE,WAAW,CAAC;QACrB,MAAM,CAAC,EAAE,eAAe,CAAC;KAC1B,GAAG,aAAa,CAAC,aAAa,CAAC,CAAC;IACjC,yEAAyE;IACzE,KAAK,CAAC,MAAM,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE;QAAE,KAAK,CAAC,EAAE,MAAM,CAAC;QAAC,MAAM,CAAC,EAAE,MAAM,CAAA;KAAE,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC;CACpF;AAED,MAAM,WAAW,IAAI;IACnB,UAAU,EAAE;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,WAAW,EAAE,MAAM,CAAC;QAAC,WAAW,EAAE,OAAO,CAAA;KAAE,CAAC;IACxE,OAAO,EAAE,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,QAAQ,EAAE,GAAG,KAAK,OAAO,CAAC;QAAE,MAAM,EAAE,OAAO,CAAC;QAAC,KAAK,CAAC,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;CAC1G;AAED,MAAM,MAAM,aAAa,GACrB;IAAE,IAAI,EAAE,YAAY,CAAC;IAAC,IAAI,EAAE,MAAM,CAAA;CAAE,GACpC;IAAE,IAAI,EAAE,gBAAgB,CAAC;IAAC,QAAQ,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAA;CAAE,GAC/D;IAAE,IAAI,EAAE,WAAW,CAAC;IAAC,EAAE,EAAE,MAAM,CAAC;IAAC,IAAI,EAAE,MAAM,CAAC;IAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;CAAE,GAC/E;IAAE,IAAI,EAAE,qBAAqB,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAAE,GAC/C;IAAE,IAAI,EAAE,OAAO,CAAC;IAAC,WAAW,EAAE,MAAM,CAAC;IAAC,YAAY,EAAE,MAAM,CAAC;IAAC,eAAe,CAAC,EAAE,MAAM,CAAC;IAAC,gBAAgB,CAAC,EAAE,MAAM,CAAA;CAAE,GACjH;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAAE,CAAC;AAErC,MAAM,WAAW,cAAc;IAC7B,QAAQ,EAAE,cAAc,CAAC;IACzB,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,IAAI,EAAE,CAAC;IACd,GAAG,EAAE,MAAM,CAAC;IACZ,QAAQ,EAAE,cAAc,CAAC;IACzB,YAAY,CAAC,EAAE,YAAY,CAAC;IAC5B,WAAW,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,EAAE,CAAC,CAAC;IACvC,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,MAAM,CAAC,EAAE,eAAe,CAAC;IACzB,sEAAsE;IACtE,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,+FAA+F;IAC/F,aAAa,CAAC,EAAE,CAAC,GAAG,EAAE;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;QAAC,MAAM,EAAE,MAAM,CAAA;KAAE,KAAK,OAAO,CAAC,OAAO,CAAC,CAAC;CAC7G;AAED,MAAM,WAAW,cAAc;IAC7B,MAAM,EAAE,KAAK,EAAE,CAAC;IAChB,SAAS,EAAE,MAAM,CAAC;IAClB,MAAM,EAAE,MAAM,CAAC;CAChB;AAYD,qBAAa,QAAQ;IACnB,OAAO,CAAC,GAAG,CAAW;IACtB,OAAO,CAAC,KAAK,CAAe;IAC5B,OAAO,CAAC,UAAU,CAAmB;IACrC,OAAO,CAAC,MAAM,CAAiB;IAC/B,OAAO,CAAC,KAAK,CAAS;IACtB,OAAO,CAAC,eAAe,CAAgC;IACvD,OAAO,CAAC,QAAQ,CAAiB;IACjC,OAAO,CAAC,eAAe,CAAqB;gBAEhC,MAAM,EAAE,cAAc,EAAE,KAAK,CAAC,EAAE,YAAY;IAOxD,IAAI,QAAQ,IAAI,QAAQ,CAEvB;IAED,YAAY,IAAI,MAAM,GAAG,SAAS;IAIlC,MAAM,IAAI,IAAI;YAKA,YAAY;IAkBpB,GAAG,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC,cAAc,CAAC;CA+QnD"}