camelagi 0.5.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 (249) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +224 -0
  3. package/camelagi.mjs +2 -0
  4. package/config.example.yaml +107 -0
  5. package/dist/agent/agent-openai.js +206 -0
  6. package/dist/agent/agent-openai.js.map +1 -0
  7. package/dist/agent/agent-sdk.js +209 -0
  8. package/dist/agent/agent-sdk.js.map +1 -0
  9. package/dist/agent/tool-adapter.js +31 -0
  10. package/dist/agent/tool-adapter.js.map +1 -0
  11. package/dist/agent/types.js +3 -0
  12. package/dist/agent/types.js.map +1 -0
  13. package/dist/agent.js +17 -0
  14. package/dist/agent.js.map +1 -0
  15. package/dist/approval-forward.js +42 -0
  16. package/dist/approval-forward.js.map +1 -0
  17. package/dist/approvals.js +151 -0
  18. package/dist/approvals.js.map +1 -0
  19. package/dist/boot.js +34 -0
  20. package/dist/boot.js.map +1 -0
  21. package/dist/bootstrap.js +451 -0
  22. package/dist/bootstrap.js.map +1 -0
  23. package/dist/camelagi-gateway.mjs +93611 -0
  24. package/dist/camelagi-gateway.mjs.map +7 -0
  25. package/dist/channels/adapter.js +10 -0
  26. package/dist/channels/adapter.js.map +1 -0
  27. package/dist/channels/discord.js +232 -0
  28. package/dist/channels/discord.js.map +1 -0
  29. package/dist/channels/handler.js +349 -0
  30. package/dist/channels/handler.js.map +1 -0
  31. package/dist/channels/index.js +19 -0
  32. package/dist/channels/index.js.map +1 -0
  33. package/dist/channels/registry.js +71 -0
  34. package/dist/channels/registry.js.map +1 -0
  35. package/dist/channels/telegram.js +83 -0
  36. package/dist/channels/telegram.js.map +1 -0
  37. package/dist/channels/types.js +3 -0
  38. package/dist/channels/types.js.map +1 -0
  39. package/dist/chunker.js +102 -0
  40. package/dist/chunker.js.map +1 -0
  41. package/dist/cli/cmd-agents.js +65 -0
  42. package/dist/cli/cmd-agents.js.map +1 -0
  43. package/dist/cli/cmd-bootstrap.js +10 -0
  44. package/dist/cli/cmd-bootstrap.js.map +1 -0
  45. package/dist/cli/cmd-chat.js +32 -0
  46. package/dist/cli/cmd-chat.js.map +1 -0
  47. package/dist/cli/cmd-config.js +88 -0
  48. package/dist/cli/cmd-config.js.map +1 -0
  49. package/dist/cli/cmd-cron.js +120 -0
  50. package/dist/cli/cmd-cron.js.map +1 -0
  51. package/dist/cli/cmd-daemon.js +37 -0
  52. package/dist/cli/cmd-daemon.js.map +1 -0
  53. package/dist/cli/cmd-doctor.js +18 -0
  54. package/dist/cli/cmd-doctor.js.map +1 -0
  55. package/dist/cli/cmd-logs.js +30 -0
  56. package/dist/cli/cmd-logs.js.map +1 -0
  57. package/dist/cli/cmd-pairing.js +41 -0
  58. package/dist/cli/cmd-pairing.js.map +1 -0
  59. package/dist/cli/cmd-reset.js +39 -0
  60. package/dist/cli/cmd-reset.js.map +1 -0
  61. package/dist/cli/cmd-serve.js +30 -0
  62. package/dist/cli/cmd-serve.js.map +1 -0
  63. package/dist/cli/cmd-sessions.js +56 -0
  64. package/dist/cli/cmd-sessions.js.map +1 -0
  65. package/dist/cli/cmd-setup.js +11 -0
  66. package/dist/cli/cmd-setup.js.map +1 -0
  67. package/dist/cli/cmd-soul.js +43 -0
  68. package/dist/cli/cmd-soul.js.map +1 -0
  69. package/dist/cli/parse.js +50 -0
  70. package/dist/cli/parse.js.map +1 -0
  71. package/dist/cli/registry.js +15 -0
  72. package/dist/cli/registry.js.map +1 -0
  73. package/dist/cli.js +103 -0
  74. package/dist/cli.js.map +1 -0
  75. package/dist/compact.js +92 -0
  76. package/dist/compact.js.map +1 -0
  77. package/dist/config.js +153 -0
  78. package/dist/config.js.map +1 -0
  79. package/dist/constants.js +21 -0
  80. package/dist/constants.js.map +1 -0
  81. package/dist/core/config.js +212 -0
  82. package/dist/core/config.js.map +1 -0
  83. package/dist/core/constants.js +21 -0
  84. package/dist/core/constants.js.map +1 -0
  85. package/dist/core/errors.js +5 -0
  86. package/dist/core/errors.js.map +1 -0
  87. package/dist/core/log.js +41 -0
  88. package/dist/core/log.js.map +1 -0
  89. package/dist/core/models.js +123 -0
  90. package/dist/core/models.js.map +1 -0
  91. package/dist/core/types.js +3 -0
  92. package/dist/core/types.js.map +1 -0
  93. package/dist/core/update-check.js +51 -0
  94. package/dist/core/update-check.js.map +1 -0
  95. package/dist/cron.js +81 -0
  96. package/dist/cron.js.map +1 -0
  97. package/dist/daemon.js +109 -0
  98. package/dist/daemon.js.map +1 -0
  99. package/dist/doctor.js +194 -0
  100. package/dist/doctor.js.map +1 -0
  101. package/dist/errors.js +5 -0
  102. package/dist/errors.js.map +1 -0
  103. package/dist/extensions/approval-forward.js +42 -0
  104. package/dist/extensions/approval-forward.js.map +1 -0
  105. package/dist/extensions/approvals.js +144 -0
  106. package/dist/extensions/approvals.js.map +1 -0
  107. package/dist/extensions/cron.js +306 -0
  108. package/dist/extensions/cron.js.map +1 -0
  109. package/dist/extensions/hooks.js +72 -0
  110. package/dist/extensions/hooks.js.map +1 -0
  111. package/dist/extensions/skills.js +97 -0
  112. package/dist/extensions/skills.js.map +1 -0
  113. package/dist/gateway/csrf.js +44 -0
  114. package/dist/gateway/csrf.js.map +1 -0
  115. package/dist/gateway/logger.js +81 -0
  116. package/dist/gateway/logger.js.map +1 -0
  117. package/dist/gateway/rate-limit.js +33 -0
  118. package/dist/gateway/rate-limit.js.map +1 -0
  119. package/dist/gateway/routes.js +315 -0
  120. package/dist/gateway/routes.js.map +1 -0
  121. package/dist/gateway/state.js +54 -0
  122. package/dist/gateway/state.js.map +1 -0
  123. package/dist/gateway/ws-handler.js +200 -0
  124. package/dist/gateway/ws-handler.js.map +1 -0
  125. package/dist/gateway-entry.js +16 -0
  126. package/dist/gateway-entry.js.map +1 -0
  127. package/dist/hooks.js +72 -0
  128. package/dist/hooks.js.map +1 -0
  129. package/dist/lanes.js +62 -0
  130. package/dist/lanes.js.map +1 -0
  131. package/dist/model.js +30 -0
  132. package/dist/model.js.map +1 -0
  133. package/dist/policy.js +22 -0
  134. package/dist/policy.js.map +1 -0
  135. package/dist/queue.js +45 -0
  136. package/dist/queue.js.map +1 -0
  137. package/dist/retry.js +96 -0
  138. package/dist/retry.js.map +1 -0
  139. package/dist/runs.js +83 -0
  140. package/dist/runs.js.map +1 -0
  141. package/dist/runtime/compact.js +99 -0
  142. package/dist/runtime/compact.js.map +1 -0
  143. package/dist/runtime/lanes.js +66 -0
  144. package/dist/runtime/lanes.js.map +1 -0
  145. package/dist/runtime/orchestrate.js +121 -0
  146. package/dist/runtime/orchestrate.js.map +1 -0
  147. package/dist/runtime/queue.js +50 -0
  148. package/dist/runtime/queue.js.map +1 -0
  149. package/dist/runtime/retry.js +127 -0
  150. package/dist/runtime/retry.js.map +1 -0
  151. package/dist/runtime/runs.js +105 -0
  152. package/dist/runtime/runs.js.map +1 -0
  153. package/dist/serve.js +209 -0
  154. package/dist/serve.js.map +1 -0
  155. package/dist/session.js +75 -0
  156. package/dist/session.js.map +1 -0
  157. package/dist/setup.js +254 -0
  158. package/dist/setup.js.map +1 -0
  159. package/dist/skills.js +89 -0
  160. package/dist/skills.js.map +1 -0
  161. package/dist/subagent.js +71 -0
  162. package/dist/subagent.js.map +1 -0
  163. package/dist/system-prompt.js +157 -0
  164. package/dist/system-prompt.js.map +1 -0
  165. package/dist/telegram/admin-bot.js +705 -0
  166. package/dist/telegram/admin-bot.js.map +1 -0
  167. package/dist/telegram/agent-bot.js +551 -0
  168. package/dist/telegram/agent-bot.js.map +1 -0
  169. package/dist/telegram/bot-approval.js +63 -0
  170. package/dist/telegram/bot-approval.js.map +1 -0
  171. package/dist/telegram/draft-stream.js +86 -0
  172. package/dist/telegram/draft-stream.js.map +1 -0
  173. package/dist/telegram/format.js +106 -0
  174. package/dist/telegram/format.js.map +1 -0
  175. package/dist/telegram/helpers.js +87 -0
  176. package/dist/telegram/helpers.js.map +1 -0
  177. package/dist/telegram/pairing-notify.js +52 -0
  178. package/dist/telegram/pairing-notify.js.map +1 -0
  179. package/dist/telegram/pairing.js +138 -0
  180. package/dist/telegram/pairing.js.map +1 -0
  181. package/dist/telegram/resolve.js +33 -0
  182. package/dist/telegram/resolve.js.map +1 -0
  183. package/dist/telegram/transcribe.js +77 -0
  184. package/dist/telegram/transcribe.js.map +1 -0
  185. package/dist/telegram/types.js +3 -0
  186. package/dist/telegram/types.js.map +1 -0
  187. package/dist/telegram/voice-wizard.js +84 -0
  188. package/dist/telegram/voice-wizard.js.map +1 -0
  189. package/dist/telegram/wizard.js +89 -0
  190. package/dist/telegram/wizard.js.map +1 -0
  191. package/dist/telegram/wizards.js +297 -0
  192. package/dist/telegram/wizards.js.map +1 -0
  193. package/dist/telegram-admin.js +800 -0
  194. package/dist/telegram-admin.js.map +1 -0
  195. package/dist/telegram.js +118 -0
  196. package/dist/telegram.js.map +1 -0
  197. package/dist/tools/cron.js +94 -0
  198. package/dist/tools/cron.js.map +1 -0
  199. package/dist/tools/edit.js +29 -0
  200. package/dist/tools/edit.js.map +1 -0
  201. package/dist/tools/exec.js +38 -0
  202. package/dist/tools/exec.js.map +1 -0
  203. package/dist/tools/fetch.js +28 -0
  204. package/dist/tools/fetch.js.map +1 -0
  205. package/dist/tools/index.js +16 -0
  206. package/dist/tools/index.js.map +1 -0
  207. package/dist/tools/memory.js +164 -0
  208. package/dist/tools/memory.js.map +1 -0
  209. package/dist/tools/patch.js +284 -0
  210. package/dist/tools/patch.js.map +1 -0
  211. package/dist/tools/read.js +26 -0
  212. package/dist/tools/read.js.map +1 -0
  213. package/dist/tools/search.js +62 -0
  214. package/dist/tools/search.js.map +1 -0
  215. package/dist/tools/subagent.js +48 -0
  216. package/dist/tools/subagent.js.map +1 -0
  217. package/dist/tools/write.js +22 -0
  218. package/dist/tools/write.js.map +1 -0
  219. package/dist/tui/commands.js +450 -0
  220. package/dist/tui/commands.js.map +1 -0
  221. package/dist/tui/components/assistant-message.js +26 -0
  222. package/dist/tui/components/assistant-message.js.map +1 -0
  223. package/dist/tui/components/chat-log.js +94 -0
  224. package/dist/tui/components/chat-log.js.map +1 -0
  225. package/dist/tui/components/custom-editor.js +40 -0
  226. package/dist/tui/components/custom-editor.js.map +1 -0
  227. package/dist/tui/components/hint-bar.js +13 -0
  228. package/dist/tui/components/hint-bar.js.map +1 -0
  229. package/dist/tui/components/tool-execution.js +73 -0
  230. package/dist/tui/components/tool-execution.js.map +1 -0
  231. package/dist/tui/components/user-message.js +19 -0
  232. package/dist/tui/components/user-message.js.map +1 -0
  233. package/dist/tui/components/welcome.js +147 -0
  234. package/dist/tui/components/welcome.js.map +1 -0
  235. package/dist/tui/context.js +3 -0
  236. package/dist/tui/context.js.map +1 -0
  237. package/dist/tui/theme.js +91 -0
  238. package/dist/tui/theme.js.map +1 -0
  239. package/dist/tui/tui.js +389 -0
  240. package/dist/tui/tui.js.map +1 -0
  241. package/dist/tui/ws-handler.js +154 -0
  242. package/dist/tui/ws-handler.js.map +1 -0
  243. package/dist/types.js +3 -0
  244. package/dist/types.js.map +1 -0
  245. package/dist/usage.js +88 -0
  246. package/dist/usage.js.map +1 -0
  247. package/dist/workspace.js +245 -0
  248. package/dist/workspace.js.map +1 -0
  249. package/package.json +74 -0
@@ -0,0 +1,200 @@
1
+ // Gateway WebSocket handler
2
+ import { createClient } from "../model.js";
3
+ import { buildSystemPrompt } from "../system-prompt.js";
4
+ import { loadMessages, listSessions, deleteSession } from "../session.js";
5
+ import { getActiveRunCount } from "../runtime/runs.js";
6
+ import { compactHistory } from "../runtime/compact.js";
7
+ import { getLaneStats } from "../runtime/lanes.js";
8
+ import { orchestrate } from "../runtime/orchestrate.js";
9
+ import { getSessionUsage } from "../usage.js";
10
+ import { CHARS_PER_TOKEN } from "../core/constants.js";
11
+ import { errorMessage } from "../core/errors.js";
12
+ import { submitDecision } from "../extensions/approvals.js";
13
+ import { checkAuth, send, logMessage, parseTokenFromUrl } from "./state.js";
14
+ export function registerWsHandler(wss, state) {
15
+ wss.on("connection", (ws, req) => {
16
+ if (!checkAuth(state, req.headers.authorization ?? parseTokenFromUrl(req.url))) {
17
+ ws.close(4001, "Unauthorized");
18
+ return;
19
+ }
20
+ state.clients.add(ws);
21
+ const alive = { value: true };
22
+ ws.on("pong", () => { alive.value = true; });
23
+ let abortController = null;
24
+ let currentRunId = null;
25
+ let sdkSessionId;
26
+ ws.on("message", async (raw) => {
27
+ let msg;
28
+ try {
29
+ msg = JSON.parse(raw.toString());
30
+ }
31
+ catch {
32
+ send(ws, { type: "error", message: "Invalid JSON" });
33
+ return;
34
+ }
35
+ try {
36
+ switch (msg.type) {
37
+ case "chat": {
38
+ const sid = msg.session ?? `ws-${Date.now()}`;
39
+ logMessage(state, "tui", "in", sid, msg.message);
40
+ abortController = new AbortController();
41
+ const resumeId = msg.sdkSessionId ?? sdkSessionId;
42
+ try {
43
+ const result = await orchestrate({
44
+ sessionId: sid,
45
+ message: msg.message,
46
+ config: state.config,
47
+ systemPrompt: state.systemPrompt,
48
+ client: state.client,
49
+ signal: abortController.signal,
50
+ resumeSessionId: resumeId,
51
+ onEvent: (event) => {
52
+ if (event.type === "init") {
53
+ sdkSessionId = event.sessionId;
54
+ }
55
+ send(ws, event);
56
+ },
57
+ onRetry: (attempt, kind) => {
58
+ send(ws, { type: "retry", attempt, kind });
59
+ },
60
+ onCompact: (oldCount, newCount) => {
61
+ send(ws, { type: "compacted", oldCount, newCount });
62
+ },
63
+ });
64
+ if (result.queued) {
65
+ send(ws, { type: "queued", session: sid });
66
+ }
67
+ else {
68
+ if (result.response) {
69
+ logMessage(state, "tui", "out", sid, result.response);
70
+ }
71
+ if (result.sdkSessionId) {
72
+ sdkSessionId = result.sdkSessionId;
73
+ }
74
+ currentRunId = result.runId;
75
+ send(ws, {
76
+ type: "done",
77
+ response: result.response,
78
+ session: sid,
79
+ runId: result.runId,
80
+ usage: result.usage,
81
+ sdkSessionId,
82
+ });
83
+ }
84
+ }
85
+ finally {
86
+ abortController = null;
87
+ currentRunId = null;
88
+ }
89
+ break;
90
+ }
91
+ case "sessions.list":
92
+ send(ws, { type: "sessions", sessions: listSessions() });
93
+ break;
94
+ case "sessions.delete":
95
+ if (msg.id)
96
+ deleteSession(msg.id);
97
+ send(ws, { type: "sessions", sessions: listSessions() });
98
+ break;
99
+ case "sessions.history": {
100
+ const messages = loadMessages((msg.id ?? msg.session ?? ""));
101
+ send(ws, {
102
+ type: "history",
103
+ session: msg.id ?? msg.session,
104
+ messages: messages.map((m) => ({ role: m.role, content: m.content })),
105
+ });
106
+ break;
107
+ }
108
+ case "compact": {
109
+ const compactSid = msg.session;
110
+ if (!compactSid) {
111
+ send(ws, { type: "error", message: "session is required" });
112
+ break;
113
+ }
114
+ const history = loadMessages(compactSid);
115
+ const result = await compactHistory(state.client, state.config.model, history, {
116
+ ...state.config.compaction,
117
+ enabled: true,
118
+ });
119
+ send(ws, {
120
+ type: "compacted",
121
+ session: compactSid,
122
+ oldCount: history.length,
123
+ newCount: result ? result.length : history.length,
124
+ });
125
+ break;
126
+ }
127
+ case "status": {
128
+ const statusSid = (msg.session ?? "");
129
+ const usage = getSessionUsage(statusSid);
130
+ const messages = loadMessages(statusSid);
131
+ const historyChars = messages.reduce((sum, m) => sum + m.content.length, 0);
132
+ send(ws, {
133
+ type: "status",
134
+ session: statusSid,
135
+ model: state.config.model,
136
+ provider: state.config.provider,
137
+ messageCount: messages.length,
138
+ historyTokens: Math.ceil(historyChars / CHARS_PER_TOKEN),
139
+ usage: usage.calls > 0 ? usage : null,
140
+ lanes: getLaneStats(),
141
+ activeRuns: getActiveRunCount(),
142
+ });
143
+ break;
144
+ }
145
+ case "model.switch": {
146
+ try {
147
+ const newModel = msg.model;
148
+ const newThinking = msg.thinking;
149
+ const newEffort = msg.effort;
150
+ if (newModel)
151
+ state.config = { ...state.config, model: newModel };
152
+ if (newThinking)
153
+ state.config = { ...state.config, thinking: newThinking };
154
+ if (newEffort)
155
+ state.config = { ...state.config, effort: newEffort };
156
+ state.client = createClient(state.config);
157
+ state.systemPrompt = buildSystemPrompt(state.config.systemPrompt);
158
+ send(ws, { type: "model.switched", model: state.config.model, thinking: state.config.thinking, effort: state.config.effort });
159
+ }
160
+ catch (err) {
161
+ send(ws, { type: "error", message: `Model switch failed: ${errorMessage(err)}` });
162
+ }
163
+ break;
164
+ }
165
+ case "abort":
166
+ if (abortController) {
167
+ abortController.abort();
168
+ abortController = null;
169
+ send(ws, { type: "aborted" });
170
+ }
171
+ break;
172
+ case "approval.decide": {
173
+ const id = msg.id;
174
+ const decision = msg.decision;
175
+ if (!id || !decision) {
176
+ send(ws, { type: "error", message: "approval.decide requires id and decision" });
177
+ break;
178
+ }
179
+ const resolved = submitDecision(id, decision);
180
+ if (!resolved) {
181
+ send(ws, { type: "error", message: "Approval not found or already resolved" });
182
+ }
183
+ break;
184
+ }
185
+ default:
186
+ send(ws, { type: "error", message: `Unknown type: ${msg.type}` });
187
+ }
188
+ }
189
+ catch (err) {
190
+ send(ws, { type: "error", message: errorMessage(err) });
191
+ }
192
+ });
193
+ ws.on("close", () => {
194
+ state.clients.delete(ws);
195
+ if (abortController)
196
+ abortController.abort();
197
+ });
198
+ });
199
+ }
200
+ //# sourceMappingURL=ws-handler.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ws-handler.js","sourceRoot":"","sources":["../../src/gateway/ws-handler.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,16 @@
1
+ // Standalone entry point for the bundled gateway.
2
+ // Used by the Camel macOS app to spawn the gateway process.
3
+ import fs from "node:fs";
4
+ import { startServer } from "./serve.js";
5
+ // Tee stdout/stderr to a log file so we can debug from the Camel app
6
+ const debugLog = `${process.env.HOME}/.camelagi/logs/gateway-debug.log`;
7
+ fs.mkdirSync(`${process.env.HOME}/.camelagi/logs`, { recursive: true });
8
+ const logStream = fs.createWriteStream(debugLog, { flags: "a" });
9
+ const origStdoutWrite = process.stdout.write.bind(process.stdout);
10
+ const origStderrWrite = process.stderr.write.bind(process.stderr);
11
+ process.stdout.write = ((chunk, ...args) => { logStream.write(chunk); return origStdoutWrite(chunk, ...args); });
12
+ process.stderr.write = ((chunk, ...args) => { logStream.write(chunk); return origStderrWrite(chunk, ...args); });
13
+ const port = parseInt(process.env.PORT ?? "18789", 10);
14
+ const host = process.env.HOST ?? "127.0.0.1";
15
+ await startServer({ port, host, cron: true, boot: false, channels: true });
16
+ //# sourceMappingURL=gateway-entry.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"gateway-entry.js","sourceRoot":"","sources":["../src/gateway-entry.ts"],"names":[],"mappings":"AAAA,kDAAkD;AAClD,4DAA4D;AAE5D,OAAO,EAAE,MAAM,SAAS,CAAC;AACzB,OAAO,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AAEzC,qEAAqE;AACrE,MAAM,QAAQ,GAAG,GAAG,OAAO,CAAC,GAAG,CAAC,IAAI,mCAAmC,CAAC;AACxE,EAAE,CAAC,SAAS,CAAC,GAAG,OAAO,CAAC,GAAG,CAAC,IAAI,iBAAiB,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;AACxE,MAAM,SAAS,GAAG,EAAE,CAAC,iBAAiB,CAAC,QAAQ,EAAE,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC,CAAC;AACjE,MAAM,eAAe,GAAG,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;AAClE,MAAM,eAAe,GAAG,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;AAClE,OAAO,CAAC,MAAM,CAAC,KAAK,GAAG,CAAC,CAAC,KAAU,EAAE,GAAG,IAAW,EAAE,EAAE,GAAG,SAAS,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,OAAO,eAAe,CAAC,KAAK,EAAE,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,CAAQ,CAAC;AACpI,OAAO,CAAC,MAAM,CAAC,KAAK,GAAG,CAAC,CAAC,KAAU,EAAE,GAAG,IAAW,EAAE,EAAE,GAAG,SAAS,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,OAAO,eAAe,CAAC,KAAK,EAAE,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,CAAQ,CAAC;AAEpI,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,IAAI,OAAO,EAAE,EAAE,CAAC,CAAC;AACvD,MAAM,IAAI,GAAG,OAAO,CAAC,GAAG,CAAC,IAAI,IAAI,WAAW,CAAC;AAE7C,MAAM,WAAW,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,KAAK,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC"}
package/dist/hooks.js ADDED
@@ -0,0 +1,72 @@
1
+ // Lifecycle hooks: shell scripts or JS handlers from ~/.opencamel/hooks/
2
+ import fs from "node:fs";
3
+ import path from "node:path";
4
+ import { execSync } from "node:child_process";
5
+ import { paths } from "./core/config.js";
6
+ import { HOOK_TIMEOUT_MS, MAX_STDERR_CHARS } from "./core/constants.js";
7
+ const hooksDir = path.join(paths.configDir, "hooks");
8
+ /**
9
+ * Load all hook scripts from ~/.opencamel/hooks/
10
+ * Naming convention: {point}.{name}.sh or {point}.{name}.js
11
+ * Examples: before_tool.log.sh, after_response.notify.sh
12
+ */
13
+ export function loadHooks() {
14
+ if (!fs.existsSync(hooksDir))
15
+ return [];
16
+ const entries = [];
17
+ const files = fs.readdirSync(hooksDir);
18
+ for (const file of files) {
19
+ if (!file.endsWith(".sh") && !file.endsWith(".js"))
20
+ continue;
21
+ const parts = file.split(".");
22
+ if (parts.length < 3)
23
+ continue;
24
+ const point = parts[0];
25
+ if (!["before_prompt", "after_response", "before_tool", "after_tool"].includes(point))
26
+ continue;
27
+ const name = parts.slice(1, -1).join(".");
28
+ entries.push({
29
+ name,
30
+ point,
31
+ script: path.join(hooksDir, file),
32
+ });
33
+ }
34
+ return entries;
35
+ }
36
+ /**
37
+ * Run all hooks for a given point.
38
+ * Context is passed via environment variables (OPENCAMEL_HOOK_*).
39
+ */
40
+ export async function runHooks(point, context, hooks) {
41
+ const all = hooks ?? loadHooks();
42
+ const matching = all.filter((h) => h.point === point);
43
+ if (matching.length === 0)
44
+ return;
45
+ const env = {
46
+ ...process.env,
47
+ OPENCAMEL_HOOK_POINT: point,
48
+ ...(context.sessionId && { OPENCAMEL_HOOK_SESSION: context.sessionId }),
49
+ ...(context.message && { OPENCAMEL_HOOK_MESSAGE: context.message }),
50
+ ...(context.response && { OPENCAMEL_HOOK_RESPONSE: context.response.slice(0, MAX_STDERR_CHARS) }),
51
+ ...(context.toolName && { OPENCAMEL_HOOK_TOOL: context.toolName }),
52
+ ...(context.toolArgs && { OPENCAMEL_HOOK_TOOL_ARGS: JSON.stringify(context.toolArgs) }),
53
+ ...(context.toolResult && { OPENCAMEL_HOOK_TOOL_RESULT: context.toolResult.slice(0, MAX_STDERR_CHARS) }),
54
+ };
55
+ for (const hook of matching) {
56
+ try {
57
+ execSync(hook.script, {
58
+ env,
59
+ timeout: HOOK_TIMEOUT_MS,
60
+ stdio: "pipe",
61
+ });
62
+ }
63
+ catch (err) {
64
+ const msg = err instanceof Error ? err.message : String(err);
65
+ process.stderr.write(`Hook ${hook.name} failed: ${msg}\n`);
66
+ }
67
+ }
68
+ }
69
+ export function ensureHooksDir() {
70
+ fs.mkdirSync(hooksDir, { recursive: true });
71
+ }
72
+ //# sourceMappingURL=hooks.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hooks.js","sourceRoot":"","sources":["../src/hooks.ts"],"names":[],"mappings":"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"}
package/dist/lanes.js ADDED
@@ -0,0 +1,62 @@
1
+ // Concurrency lanes: limit parallel agent runs by type
2
+ export var Lane;
3
+ (function (Lane) {
4
+ Lane["Main"] = "main";
5
+ Lane["Cron"] = "cron";
6
+ Lane["Subagent"] = "subagent";
7
+ })(Lane || (Lane = {}));
8
+ const lanes = new Map();
9
+ export function configureLane(lane, limit) {
10
+ const existing = lanes.get(lane);
11
+ if (existing) {
12
+ existing.limit = limit;
13
+ }
14
+ else {
15
+ lanes.set(lane, { limit, active: 0, queue: [] });
16
+ }
17
+ }
18
+ /**
19
+ * Acquire a slot in a lane. Returns a release function.
20
+ * If the lane is full, waits until a slot is available.
21
+ */
22
+ export async function acquireLane(lane) {
23
+ let config = lanes.get(lane);
24
+ if (!config) {
25
+ // Default: unlimited
26
+ config = { limit: Infinity, active: 0, queue: [] };
27
+ lanes.set(lane, config);
28
+ }
29
+ if (config.active < config.limit) {
30
+ config.active++;
31
+ return () => releaseLane(lane);
32
+ }
33
+ // Wait for a slot
34
+ await new Promise((resolve) => {
35
+ config.queue.push(resolve);
36
+ });
37
+ config.active++;
38
+ return () => releaseLane(lane);
39
+ }
40
+ function releaseLane(lane) {
41
+ const config = lanes.get(lane);
42
+ if (!config)
43
+ return;
44
+ config.active--;
45
+ // Wake up next waiter
46
+ const next = config.queue.shift();
47
+ if (next)
48
+ next();
49
+ }
50
+ /** Get current lane stats */
51
+ export function getLaneStats() {
52
+ const stats = {};
53
+ for (const [lane, config] of lanes) {
54
+ stats[lane] = {
55
+ active: config.active,
56
+ limit: config.limit === Infinity ? -1 : config.limit,
57
+ queued: config.queue.length,
58
+ };
59
+ }
60
+ return stats;
61
+ }
62
+ //# sourceMappingURL=lanes.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"lanes.js","sourceRoot":"","sources":["../src/lanes.ts"],"names":[],"mappings":"AAAA,uDAAuD;AAEvD,MAAM,CAAN,IAAY,IAIX;AAJD,WAAY,IAAI;IACd,qBAAa,CAAA;IACb,qBAAa,CAAA;IACb,6BAAqB,CAAA;AACvB,CAAC,EAJW,IAAI,KAAJ,IAAI,QAIf;AAQD,MAAM,KAAK,GAAG,IAAI,GAAG,EAAoB,CAAC;AAE1C,MAAM,UAAU,aAAa,CAAC,IAAU,EAAE,KAAa;IACrD,MAAM,QAAQ,GAAG,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IACjC,IAAI,QAAQ,EAAE,CAAC;QACb,QAAQ,CAAC,KAAK,GAAG,KAAK,CAAC;IACzB,CAAC;SAAM,CAAC;QACN,KAAK,CAAC,GAAG,CAAC,IAAI,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,CAAC;IACnD,CAAC;AACH,CAAC;AAED;;;GAGG;AACH,MAAM,CAAC,KAAK,UAAU,WAAW,CAAC,IAAU;IAC1C,IAAI,MAAM,GAAG,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IAC7B,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,qBAAqB;QACrB,MAAM,GAAG,EAAE,KAAK,EAAE,QAAQ,EAAE,MAAM,EAAE,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC;QACnD,KAAK,CAAC,GAAG,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;IAC1B,CAAC;IAED,IAAI,MAAM,CAAC,MAAM,GAAG,MAAM,CAAC,KAAK,EAAE,CAAC;QACjC,MAAM,CAAC,MAAM,EAAE,CAAC;QAChB,OAAO,GAAG,EAAE,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;IACjC,CAAC;IAED,kBAAkB;IAClB,MAAM,IAAI,OAAO,CAAO,CAAC,OAAO,EAAE,EAAE;QAClC,MAAO,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAC9B,CAAC,CAAC,CAAC;IACH,MAAM,CAAC,MAAM,EAAE,CAAC;IAChB,OAAO,GAAG,EAAE,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;AACjC,CAAC;AAED,SAAS,WAAW,CAAC,IAAU;IAC7B,MAAM,MAAM,GAAG,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IAC/B,IAAI,CAAC,MAAM;QAAE,OAAO;IAEpB,MAAM,CAAC,MAAM,EAAE,CAAC;IAEhB,sBAAsB;IACtB,MAAM,IAAI,GAAG,MAAM,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC;IAClC,IAAI,IAAI;QAAE,IAAI,EAAE,CAAC;AACnB,CAAC;AAED,6BAA6B;AAC7B,MAAM,UAAU,YAAY;IAC1B,MAAM,KAAK,GAAsE,EAAE,CAAC;IACpF,KAAK,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,IAAI,KAAK,EAAE,CAAC;QACnC,KAAK,CAAC,IAAI,CAAC,GAAG;YACZ,MAAM,EAAE,MAAM,CAAC,MAAM;YACrB,KAAK,EAAE,MAAM,CAAC,KAAK,KAAK,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK;YACpD,MAAM,EAAE,MAAM,CAAC,KAAK,CAAC,MAAM;SAC5B,CAAC;IACJ,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC"}
package/dist/model.js ADDED
@@ -0,0 +1,30 @@
1
+ // Anthropic SDK client for direct API calls (compaction, doctor)
2
+ import Anthropic from "@anthropic-ai/sdk";
3
+ import { DEFAULT_MAX_TOKENS } from "./core/constants.js";
4
+ export function createClient(config) {
5
+ const apiKey = config.apiKey ?? "not-configured";
6
+ return new Anthropic({ apiKey });
7
+ }
8
+ /** Direct API call — used by compaction and doctor, not by the agent loop */
9
+ export async function chatDirect(client, model, system, userContent) {
10
+ const response = await client.messages.create({
11
+ model,
12
+ max_tokens: DEFAULT_MAX_TOKENS,
13
+ system,
14
+ messages: [{ role: "user", content: userContent }],
15
+ });
16
+ const text = response.content
17
+ .filter((b) => b.type === "text")
18
+ .map((b) => b.text)
19
+ .join("");
20
+ return {
21
+ content: text,
22
+ usage: {
23
+ inputTokens: response.usage.input_tokens,
24
+ outputTokens: response.usage.output_tokens,
25
+ cacheReadTokens: response.usage.cache_read_input_tokens ?? 0,
26
+ cacheWriteTokens: response.usage.cache_creation_input_tokens ?? 0,
27
+ },
28
+ };
29
+ }
30
+ //# sourceMappingURL=model.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"model.js","sourceRoot":"","sources":["../src/model.ts"],"names":[],"mappings":"AAAA,iEAAiE;AAEjE,OAAO,SAAS,MAAM,mBAAmB,CAAC;AAG1C,OAAO,EAAE,kBAAkB,EAAE,MAAM,qBAAqB,CAAC;AAEzD,MAAM,UAAU,YAAY,CAAC,MAAc;IACzC,MAAM,MAAM,GAAG,MAAM,CAAC,MAAM,IAAI,gBAAgB,CAAC;IACjD,OAAO,IAAI,SAAS,CAAC,EAAE,MAAM,EAAE,CAAC,CAAC;AACnC,CAAC;AAOD,6EAA6E;AAC7E,MAAM,CAAC,KAAK,UAAU,UAAU,CAC9B,MAAiB,EACjB,KAAa,EACb,MAAc,EACd,WAAmB;IAEnB,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC;QAC5C,KAAK;QACL,UAAU,EAAE,kBAAkB;QAC9B,MAAM;QACN,QAAQ,EAAE,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,WAAW,EAAE,CAAC;KACnD,CAAC,CAAC;IAEH,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO;SAC1B,MAAM,CAAC,CAAC,CAAC,EAA4B,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,MAAM,CAAC;SAC1D,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC;SAClB,IAAI,CAAC,EAAE,CAAC,CAAC;IAEZ,OAAO;QACL,OAAO,EAAE,IAAI;QACb,KAAK,EAAE;YACL,WAAW,EAAE,QAAQ,CAAC,KAAK,CAAC,YAAY;YACxC,YAAY,EAAE,QAAQ,CAAC,KAAK,CAAC,aAAa;YAC1C,eAAe,EAAG,QAAQ,CAAC,KAAa,CAAC,uBAAuB,IAAI,CAAC;YACrE,gBAAgB,EAAG,QAAQ,CAAC,KAAa,CAAC,2BAA2B,IAAI,CAAC;SAC3E;KACF,CAAC;AACJ,CAAC"}
package/dist/policy.js ADDED
@@ -0,0 +1,22 @@
1
+ // Tool policy: allow/deny filtering for tools
2
+ /**
3
+ * Filter tools based on allow/deny policy.
4
+ * - If deny list is non-empty, remove matching tools first.
5
+ * - If allow list is non-empty, only keep matching tools.
6
+ * - Empty lists = no filtering.
7
+ */
8
+ export function filterTools(tools, policy) {
9
+ let filtered = tools;
10
+ // Apply deny list first
11
+ if (policy.deny.length > 0) {
12
+ const denySet = new Set(policy.deny.map((n) => n.toLowerCase()));
13
+ filtered = filtered.filter((t) => !denySet.has(t.name.toLowerCase()));
14
+ }
15
+ // Apply allow list
16
+ if (policy.allow.length > 0) {
17
+ const allowSet = new Set(policy.allow.map((n) => n.toLowerCase()));
18
+ filtered = filtered.filter((t) => allowSet.has(t.name.toLowerCase()));
19
+ }
20
+ return filtered;
21
+ }
22
+ //# sourceMappingURL=policy.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"policy.js","sourceRoot":"","sources":["../src/policy.ts"],"names":[],"mappings":"AAAA,8CAA8C;AAS9C;;;;;GAKG;AACH,MAAM,UAAU,WAAW,CACzB,KAAgC,EAChC,MAAkB;IAElB,IAAI,QAAQ,GAAG,KAAK,CAAC;IAErB,wBAAwB;IACxB,IAAI,MAAM,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAC3B,MAAM,OAAO,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC;QACjE,QAAQ,GAAG,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC;IACxE,CAAC;IAED,mBAAmB;IACnB,IAAI,MAAM,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAC5B,MAAM,QAAQ,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC;QACnE,QAAQ,GAAG,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC;IACxE,CAAC;IAED,OAAO,QAAQ,CAAC;AAClB,CAAC"}
package/dist/queue.js ADDED
@@ -0,0 +1,45 @@
1
+ // Message queue: per-session queue for inbound messages while agent is running
2
+ import { isRunActive, waitForRunEnd } from "./runs.js";
3
+ const queues = new Map();
4
+ export function enqueueMessage(sessionId, text) {
5
+ return new Promise((resolve, reject) => {
6
+ if (!queues.has(sessionId))
7
+ queues.set(sessionId, []);
8
+ queues.get(sessionId).push({ text, resolve, reject, enqueuedAt: Date.now() });
9
+ });
10
+ }
11
+ export function getQueueLength(sessionId) {
12
+ return queues.get(sessionId)?.length ?? 0;
13
+ }
14
+ export function drainQueue(sessionId) {
15
+ const queue = queues.get(sessionId);
16
+ if (!queue || queue.length === 0)
17
+ return [];
18
+ queues.delete(sessionId);
19
+ return queue;
20
+ }
21
+ export function clearQueue(sessionId) {
22
+ const queue = queues.get(sessionId);
23
+ if (queue) {
24
+ for (const msg of queue) {
25
+ msg.reject(new Error("Queue cleared"));
26
+ }
27
+ queues.delete(sessionId);
28
+ }
29
+ }
30
+ /**
31
+ * If a run is active for this session, enqueue the message and wait for
32
+ * the active run to finish. Returns `{ queued: true }` so the caller
33
+ * knows NOT to start a new run — the message is already stored for
34
+ * later processing. Returns `{ queued: false }` when no run is active.
35
+ */
36
+ export async function queueOrProcess(sessionId, text) {
37
+ if (!isRunActive(sessionId)) {
38
+ return { queued: false };
39
+ }
40
+ // Run is active — enqueue and wait for run to end
41
+ const promise = enqueueMessage(sessionId, text);
42
+ await waitForRunEnd(sessionId);
43
+ return { queued: true, promise };
44
+ }
45
+ //# sourceMappingURL=queue.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"queue.js","sourceRoot":"","sources":["../src/queue.ts"],"names":[],"mappings":"AAAA,+EAA+E;AAE/E,OAAO,EAAE,WAAW,EAAE,aAAa,EAAE,MAAM,WAAW,CAAC;AASvD,MAAM,MAAM,GAAG,IAAI,GAAG,EAA2B,CAAC;AAElD,MAAM,UAAU,cAAc,CAC5B,SAAiB,EACjB,IAAY;IAEZ,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;QACrC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,SAAS,CAAC;YAAE,MAAM,CAAC,GAAG,CAAC,SAAS,EAAE,EAAE,CAAC,CAAC;QACtD,MAAM,CAAC,GAAG,CAAC,SAAS,CAAE,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,EAAE,UAAU,EAAE,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;IACjF,CAAC,CAAC,CAAC;AACL,CAAC;AAED,MAAM,UAAU,cAAc,CAAC,SAAiB;IAC9C,OAAO,MAAM,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE,MAAM,IAAI,CAAC,CAAC;AAC5C,CAAC;AAED,MAAM,UAAU,UAAU,CAAC,SAAiB;IAC1C,MAAM,KAAK,GAAG,MAAM,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;IACpC,IAAI,CAAC,KAAK,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,EAAE,CAAC;IAC5C,MAAM,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;IACzB,OAAO,KAAK,CAAC;AACf,CAAC;AAED,MAAM,UAAU,UAAU,CAAC,SAAiB;IAC1C,MAAM,KAAK,GAAG,MAAM,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;IACpC,IAAI,KAAK,EAAE,CAAC;QACV,KAAK,MAAM,GAAG,IAAI,KAAK,EAAE,CAAC;YACxB,GAAG,CAAC,MAAM,CAAC,IAAI,KAAK,CAAC,eAAe,CAAC,CAAC,CAAC;QACzC,CAAC;QACD,MAAM,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;IAC3B,CAAC;AACH,CAAC;AAED;;;;;GAKG;AACH,MAAM,CAAC,KAAK,UAAU,cAAc,CAClC,SAAiB,EACjB,IAAY;IAEZ,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC,EAAE,CAAC;QAC5B,OAAO,EAAE,MAAM,EAAE,KAAK,EAAE,CAAC;IAC3B,CAAC;IAED,kDAAkD;IAClD,MAAM,OAAO,GAAG,cAAc,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;IAChD,MAAM,aAAa,CAAC,SAAS,CAAC,CAAC;IAC/B,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,OAAO,EAAE,CAAC;AACnC,CAAC"}
package/dist/retry.js ADDED
@@ -0,0 +1,96 @@
1
+ // Error classification + retry logic for agent runs
2
+ export function classifyError(err) {
3
+ const msg = err.message.toLowerCase();
4
+ // User abort (not retryable — distinct from timeout)
5
+ if (msg === "aborted" || msg === "the operation was aborted" || msg === "this operation was aborted") {
6
+ return "abort";
7
+ }
8
+ // Auth errors
9
+ if (msg.includes("401") || msg.includes("403") || msg.includes("unauthorized")
10
+ || msg.includes("invalid api key") || msg.includes("token expired")) {
11
+ return "auth";
12
+ }
13
+ // Billing
14
+ if (msg.includes("402") || msg.includes("insufficient") || msg.includes("payment required")
15
+ || msg.includes("billing")) {
16
+ return "billing";
17
+ }
18
+ // Rate limit
19
+ if (msg.includes("429") || msg.includes("rate limit") || msg.includes("too many requests")
20
+ || msg.includes("quota") || msg.includes("resource exhausted")) {
21
+ return "rate_limit";
22
+ }
23
+ // Context overflow
24
+ if (msg.includes("context") && (msg.includes("exceeded") || msg.includes("too large"))
25
+ || msg.includes("prompt is too long") || msg.includes("request too large")
26
+ || msg.includes("maximum context length")) {
27
+ return "overflow";
28
+ }
29
+ // Timeout
30
+ if (msg.includes("timeout") || msg.includes("deadline exceeded") || msg.includes("aborted")
31
+ || msg.includes("abort")) {
32
+ return "timeout";
33
+ }
34
+ // Format errors (bad request shape)
35
+ if (msg.includes("400") || msg.includes("invalid request") || msg.includes("validation")) {
36
+ return "format";
37
+ }
38
+ // Transient server errors (retryable)
39
+ if (msg.includes("500") || msg.includes("502") || msg.includes("503")
40
+ || msg.includes("service unavailable") || msg.includes("internal server error")) {
41
+ return "rate_limit"; // Treat as retryable
42
+ }
43
+ return "unknown";
44
+ }
45
+ export function isRetryable(kind) {
46
+ return kind === "rate_limit" || kind === "timeout";
47
+ }
48
+ /**
49
+ * Wrap an async function with retry logic.
50
+ * - rate_limit/timeout: retry with exponential backoff
51
+ * - overflow: call onCompact() then retry once
52
+ * - auth/billing/format: fail immediately
53
+ */
54
+ export async function withRetry(fn, opts) {
55
+ let lastError;
56
+ let overflowRetried = false;
57
+ for (let attempt = 0; attempt <= opts.maxRetries; attempt++) {
58
+ try {
59
+ return await fn();
60
+ }
61
+ catch (err) {
62
+ lastError = err instanceof Error ? err : new Error(String(err));
63
+ const kind = classifyError(lastError);
64
+ // Non-retryable errors
65
+ if (kind === "auth" || kind === "billing" || kind === "format" || kind === "abort") {
66
+ throw lastError;
67
+ }
68
+ // Overflow: compact and retry once
69
+ if (kind === "overflow" && !overflowRetried && opts.onCompact) {
70
+ overflowRetried = true;
71
+ opts.onRetry?.(attempt, kind, lastError);
72
+ await opts.onCompact();
73
+ continue;
74
+ }
75
+ // Last attempt — don't retry
76
+ if (attempt === opts.maxRetries)
77
+ break;
78
+ // Retryable: backoff
79
+ if (isRetryable(kind)) {
80
+ opts.onRetry?.(attempt, kind, lastError);
81
+ const delay = opts.backoffMs * Math.pow(2, attempt);
82
+ await new Promise((r) => setTimeout(r, delay));
83
+ continue;
84
+ }
85
+ // Unknown errors: retry once then fail
86
+ if (kind === "unknown" && attempt === 0) {
87
+ opts.onRetry?.(attempt, kind, lastError);
88
+ await new Promise((r) => setTimeout(r, opts.backoffMs));
89
+ continue;
90
+ }
91
+ break;
92
+ }
93
+ }
94
+ throw lastError;
95
+ }
96
+ //# sourceMappingURL=retry.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"retry.js","sourceRoot":"","sources":["../src/retry.ts"],"names":[],"mappings":"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"}