@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.
- package/LICENSE +179 -0
- package/dist/.tsbuildinfo +1 -0
- package/dist/src/channel-registry.d.ts +26 -0
- package/dist/src/channel-registry.d.ts.map +1 -0
- package/dist/src/channel-registry.js +100 -0
- package/dist/src/channel-registry.js.map +1 -0
- package/dist/src/channel-router.d.ts +28 -0
- package/dist/src/channel-router.d.ts.map +1 -0
- package/dist/src/channel-router.js +52 -0
- package/dist/src/channel-router.js.map +1 -0
- package/dist/src/cron-manager.d.ts +70 -0
- package/dist/src/cron-manager.d.ts.map +1 -0
- package/dist/src/cron-manager.js +301 -0
- package/dist/src/cron-manager.js.map +1 -0
- package/dist/src/event-bus.d.ts +99 -0
- package/dist/src/event-bus.d.ts.map +1 -0
- package/dist/src/event-bus.js +56 -0
- package/dist/src/event-bus.js.map +1 -0
- package/dist/src/heartbeat-delivery.d.ts +27 -0
- package/dist/src/heartbeat-delivery.d.ts.map +1 -0
- package/dist/src/heartbeat-delivery.js +37 -0
- package/dist/src/heartbeat-delivery.js.map +1 -0
- package/dist/src/heartbeat-executor.d.ts +35 -0
- package/dist/src/heartbeat-executor.d.ts.map +1 -0
- package/dist/src/heartbeat-executor.js +174 -0
- package/dist/src/heartbeat-executor.js.map +1 -0
- package/dist/src/index.d.ts +26 -0
- package/dist/src/index.d.ts.map +1 -0
- package/dist/src/index.js +15 -0
- package/dist/src/index.js.map +1 -0
- package/dist/src/permission.d.ts +21 -0
- package/dist/src/permission.d.ts.map +1 -0
- package/dist/src/permission.js +69 -0
- package/dist/src/permission.js.map +1 -0
- package/dist/src/reminder-parser.d.ts +12 -0
- package/dist/src/reminder-parser.d.ts.map +1 -0
- package/dist/src/reminder-parser.js +164 -0
- package/dist/src/reminder-parser.js.map +1 -0
- package/dist/src/session-store.d.ts +32 -0
- package/dist/src/session-store.d.ts.map +1 -0
- package/dist/src/session-store.js +112 -0
- package/dist/src/session-store.js.map +1 -0
- package/dist/src/settings.d.ts +41 -0
- package/dist/src/settings.d.ts.map +1 -0
- package/dist/src/settings.js +128 -0
- package/dist/src/settings.js.map +1 -0
- package/dist/src/telegram-gateway.d.ts +35 -0
- package/dist/src/telegram-gateway.d.ts.map +1 -0
- package/dist/src/telegram-gateway.js +145 -0
- package/dist/src/telegram-gateway.js.map +1 -0
- package/dist/src/turn-loop.d.ts +106 -0
- package/dist/src/turn-loop.d.ts.map +1 -0
- package/dist/src/turn-loop.js +288 -0
- package/dist/src/turn-loop.js.map +1 -0
- 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, '&').replace(/</g, '<').replace(/>/g, '>');
|
|
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":"AAAA,OAAO,EAAE,GAAG,EAAW,MAAM,QAAQ,CAAC;AAmBtC,MAAM,OAAO,eAAe;IAClB,GAAG,CAAe;IAClB,SAAS,GAAG,KAAK,CAAC;IAClB,MAAM,CAAwB;IAC9B,YAAY,GAAgC,IAAI,GAAG,EAAE,CAAC;IAE9D,YAAY,MAA6B;QACvC,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;QACrB,IAAI,CAAC,GAAG,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;QACpC,IAAI,CAAC,aAAa,EAAE,CAAC;IACvB,CAAC;IAEO,aAAa;QACnB,6BAA6B;QAC7B,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,cAAc,EAAE,KAAK,EAAE,GAAG,EAAE,EAAE;YACxC,MAAM,GAAG,GAAG,GAAG,CAAC,OAAO,CAAC,IAAI,CAAC;YAC7B,MAAM,MAAM,GAAG,MAAM,CAAC,GAAG,CAAC,IAAI,EAAE,EAAE,IAAI,EAAE,CAAC,CAAC;YAC1C,MAAM,MAAM,GAAG,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACnC,MAAM,QAAQ,GAAG,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC;YAC/B,MAAM,SAAS,GAAG,GAAG,CAAC,IAAI,CAAC,KAAK,CAAC;YAEjC,uCAAuC;YACvC,IAAI,MAAM,KAAK,IAAI,CAAC,MAAM,CAAC,aAAa,EAAE,CAAC;gBACzC,OAAO;YACT,CAAC;YAED,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC;gBACxB,IAAI,EAAE,GAAG;gBACT,MAAM;gBACN,MAAM;gBACN,OAAO,EAAE,QAAQ,KAAK,OAAO,IAAI,QAAQ,KAAK,YAAY;gBAC1D,SAAS;aACV,CAAC,CAAC;YACH,kEAAkE;YAClE,4DAA4D;YAC5D,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;QAC3B,CAAC,CAAC,CAAC;QAEH,4DAA4D;QAC5D,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,gBAAgB,EAAE,KAAK,EAAE,GAAG,EAAE,EAAE;YAC1C,MAAM,IAAI,GAAG,GAAG,CAAC,aAAa,CAAC,IAAI,CAAC;YACpC,IAAI,CAAC,IAAI,EAAE,CAAC;gBACV,MAAM,GAAG,CAAC,mBAAmB,EAAE,CAAC;gBAChC,OAAO;YACT,CAAC;YACD,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;YACpC,IAAI,SAAS,KAAK,CAAC,CAAC,EAAE,CAAC;gBACrB,MAAM,GAAG,CAAC,mBAAmB,EAAE,CAAC;gBAChC,OAAO;YACT,CAAC;YACD,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,SAAS,CAAC,CAAC;YACxC,MAAM,UAAU,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC;YAC7C,MAAM,QAAQ,GAAG,MAAM,KAAK,SAAS,CAAC;YACtC,IAAI,CAAC,MAAM,CAAC,kBAAkB,EAAE,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC;YACvD,MAAM,GAAG,CAAC,mBAAmB,EAAE,CAAC,CAAC,8BAA8B;QACjE,CAAC,CAAC,CAAC;QAEH,0BAA0B;QAC1B,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE;YACrB,OAAO,CAAC,KAAK,CAAC,2BAA2B,EAAE,GAAG,CAAC,KAAK,CAAC,CAAC;YACtD,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC,GAAG,CAAC,KAAc,CAAC,CAAC;QAC5C,CAAC,CAAC,CAAC;IACL,CAAC;IAED,KAAK,CAAC,WAAW,CAAC,MAAc,EAAE,IAAY;QAC5C,IAAI,CAAC;YACH,MAAM,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,WAAW,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC;QAC/C,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,GAAG,GAAG,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;YAC7D,OAAO,CAAC,KAAK,CAAC,kDAAkD,MAAM,KAAK,GAAG,EAAE,CAAC,CAAC;YAClF,MAAM,GAAG,CAAC;QACZ,CAAC;IACH,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,mBAAmB,CACvB,MAAc,EACd,UAAkB,EAClB,QAAgB,EAChB,YAAoB;QAEpB,6EAA6E;QAC7E,MAAM,IAAI,GAAG,CAAC,CAAS,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,eAAe,EAAE,EAAE,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;QAC9H,MAAM,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC,CAAC,OAAO,CAAC,SAAS,EAAE,GAAG,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;QAC9E,MAAM,IAAI,GAAG,+BAA+B,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,QAAQ,GAAG,YAAY,CAAC;QAExF,IAAI,CAAC;YACH,MAAM,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,WAAW,CAAC,MAAM,EAAE,IAAI,EAAE;gBAC3C,YAAY,EAAE;oBACZ,eAAe,EAAE;wBACf;4BACE,EAAE,IAAI,EAAE,OAAO,EAAE,aAAa,EAAE,UAAU,GAAG,UAAU,EAAE;4BACzD,EAAE,IAAI,EAAE,MAAM,EAAE,aAAa,EAAE,OAAO,GAAG,UAAU,EAAE;yBACtD;qBACF;iBACF;gBACD,UAAU,EAAE,MAAM;aACnB,CAAC,CAAC;QACL,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,GAAG,GAAG,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;YAC7D,OAAO,CAAC,KAAK,CAAC,yDAAyD,GAAG,MAAM,GAAG,IAAI,GAAG,GAAG,CAAC,CAAC;YAC/F,MAAM,GAAG,CAAC;QACZ,CAAC;IACH,CAAC;IAED,KAAK,CAAC,KAAK;QACT,IAAI,IAAI,CAAC,SAAS;YAAE,OAAO;QAC3B,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QACtB,MAAM,IAAI,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC;QACvB,OAAO,CAAC,GAAG,CAAC,0CAA0C,CAAC,CAAC;IAC1D,CAAC;IAED,KAAK,CAAC,IAAI;QACR,IAAI,CAAC,IAAI,CAAC,SAAS;YAAE,OAAO;QAC5B,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;QACvB,0BAA0B;QAC1B,KAAK,MAAM,KAAK,IAAI,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE,EAAE,CAAC;YAC/C,aAAa,CAAC,KAAK,CAAC,CAAC;QACvB,CAAC;QACD,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,CAAC;QAC1B,MAAM,IAAI,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC;QACtB,OAAO,CAAC,GAAG,CAAC,gCAAgC,CAAC,CAAC;IAChD,CAAC;IAED,qBAAqB,CAAC,EAAmD;QACvE,IAAI,CAAC,MAAM,CAAC,kBAAkB,GAAG,EAAE,CAAC;IACtC,CAAC;IAED,IAAI,OAAO;QACT,OAAO,IAAI,CAAC,SAAS,CAAC;IACxB,CAAC;IAED,WAAW,CAAC,MAAc;QACxB,+CAA+C;QAC/C,IAAI,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,MAAM,CAAC;YAAE,OAAO;QAE1C,OAAO,CAAC,KAAK,CAAC,sCAAsC,MAAM,eAAe,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC;QAE3F,MAAM,IAAI,GAAG,GAAG,EAAE;YAChB,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;gBACnB,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,cAAc,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE;oBAC1D,OAAO,CAAC,KAAK,CAAC,gDAAgD,MAAM,GAAG,EAAE,GAAG,EAAE,OAAO,IAAI,GAAG,CAAC,CAAC;gBAChG,CAAC,CAAC,CAAC;YACL,CAAC;QACH,CAAC,CAAC;QAEF,IAAI,EAAE,CAAC,CAAC,mBAAmB;QAC3B,MAAM,QAAQ,GAAG,WAAW,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;QACzC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAC;IAC1C,CAAC;IAED,UAAU,CAAC,MAAc;QACvB,MAAM,KAAK,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QAC5C,IAAI,KAAK,EAAE,CAAC;YACV,aAAa,CAAC,KAAK,CAAC,CAAC;YACrB,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QACnC,CAAC;IACH,CAAC;CACF"}
|
|
@@ -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":"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"}
|