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.
- package/LICENSE +21 -0
- package/README.md +224 -0
- package/camelagi.mjs +2 -0
- package/config.example.yaml +107 -0
- package/dist/agent/agent-openai.js +206 -0
- package/dist/agent/agent-openai.js.map +1 -0
- package/dist/agent/agent-sdk.js +209 -0
- package/dist/agent/agent-sdk.js.map +1 -0
- package/dist/agent/tool-adapter.js +31 -0
- package/dist/agent/tool-adapter.js.map +1 -0
- package/dist/agent/types.js +3 -0
- package/dist/agent/types.js.map +1 -0
- package/dist/agent.js +17 -0
- package/dist/agent.js.map +1 -0
- package/dist/approval-forward.js +42 -0
- package/dist/approval-forward.js.map +1 -0
- package/dist/approvals.js +151 -0
- package/dist/approvals.js.map +1 -0
- package/dist/boot.js +34 -0
- package/dist/boot.js.map +1 -0
- package/dist/bootstrap.js +451 -0
- package/dist/bootstrap.js.map +1 -0
- package/dist/camelagi-gateway.mjs +93611 -0
- package/dist/camelagi-gateway.mjs.map +7 -0
- package/dist/channels/adapter.js +10 -0
- package/dist/channels/adapter.js.map +1 -0
- package/dist/channels/discord.js +232 -0
- package/dist/channels/discord.js.map +1 -0
- package/dist/channels/handler.js +349 -0
- package/dist/channels/handler.js.map +1 -0
- package/dist/channels/index.js +19 -0
- package/dist/channels/index.js.map +1 -0
- package/dist/channels/registry.js +71 -0
- package/dist/channels/registry.js.map +1 -0
- package/dist/channels/telegram.js +83 -0
- package/dist/channels/telegram.js.map +1 -0
- package/dist/channels/types.js +3 -0
- package/dist/channels/types.js.map +1 -0
- package/dist/chunker.js +102 -0
- package/dist/chunker.js.map +1 -0
- package/dist/cli/cmd-agents.js +65 -0
- package/dist/cli/cmd-agents.js.map +1 -0
- package/dist/cli/cmd-bootstrap.js +10 -0
- package/dist/cli/cmd-bootstrap.js.map +1 -0
- package/dist/cli/cmd-chat.js +32 -0
- package/dist/cli/cmd-chat.js.map +1 -0
- package/dist/cli/cmd-config.js +88 -0
- package/dist/cli/cmd-config.js.map +1 -0
- package/dist/cli/cmd-cron.js +120 -0
- package/dist/cli/cmd-cron.js.map +1 -0
- package/dist/cli/cmd-daemon.js +37 -0
- package/dist/cli/cmd-daemon.js.map +1 -0
- package/dist/cli/cmd-doctor.js +18 -0
- package/dist/cli/cmd-doctor.js.map +1 -0
- package/dist/cli/cmd-logs.js +30 -0
- package/dist/cli/cmd-logs.js.map +1 -0
- package/dist/cli/cmd-pairing.js +41 -0
- package/dist/cli/cmd-pairing.js.map +1 -0
- package/dist/cli/cmd-reset.js +39 -0
- package/dist/cli/cmd-reset.js.map +1 -0
- package/dist/cli/cmd-serve.js +30 -0
- package/dist/cli/cmd-serve.js.map +1 -0
- package/dist/cli/cmd-sessions.js +56 -0
- package/dist/cli/cmd-sessions.js.map +1 -0
- package/dist/cli/cmd-setup.js +11 -0
- package/dist/cli/cmd-setup.js.map +1 -0
- package/dist/cli/cmd-soul.js +43 -0
- package/dist/cli/cmd-soul.js.map +1 -0
- package/dist/cli/parse.js +50 -0
- package/dist/cli/parse.js.map +1 -0
- package/dist/cli/registry.js +15 -0
- package/dist/cli/registry.js.map +1 -0
- package/dist/cli.js +103 -0
- package/dist/cli.js.map +1 -0
- package/dist/compact.js +92 -0
- package/dist/compact.js.map +1 -0
- package/dist/config.js +153 -0
- package/dist/config.js.map +1 -0
- package/dist/constants.js +21 -0
- package/dist/constants.js.map +1 -0
- package/dist/core/config.js +212 -0
- package/dist/core/config.js.map +1 -0
- package/dist/core/constants.js +21 -0
- package/dist/core/constants.js.map +1 -0
- package/dist/core/errors.js +5 -0
- package/dist/core/errors.js.map +1 -0
- package/dist/core/log.js +41 -0
- package/dist/core/log.js.map +1 -0
- package/dist/core/models.js +123 -0
- package/dist/core/models.js.map +1 -0
- package/dist/core/types.js +3 -0
- package/dist/core/types.js.map +1 -0
- package/dist/core/update-check.js +51 -0
- package/dist/core/update-check.js.map +1 -0
- package/dist/cron.js +81 -0
- package/dist/cron.js.map +1 -0
- package/dist/daemon.js +109 -0
- package/dist/daemon.js.map +1 -0
- package/dist/doctor.js +194 -0
- package/dist/doctor.js.map +1 -0
- package/dist/errors.js +5 -0
- package/dist/errors.js.map +1 -0
- package/dist/extensions/approval-forward.js +42 -0
- package/dist/extensions/approval-forward.js.map +1 -0
- package/dist/extensions/approvals.js +144 -0
- package/dist/extensions/approvals.js.map +1 -0
- package/dist/extensions/cron.js +306 -0
- package/dist/extensions/cron.js.map +1 -0
- package/dist/extensions/hooks.js +72 -0
- package/dist/extensions/hooks.js.map +1 -0
- package/dist/extensions/skills.js +97 -0
- package/dist/extensions/skills.js.map +1 -0
- package/dist/gateway/csrf.js +44 -0
- package/dist/gateway/csrf.js.map +1 -0
- package/dist/gateway/logger.js +81 -0
- package/dist/gateway/logger.js.map +1 -0
- package/dist/gateway/rate-limit.js +33 -0
- package/dist/gateway/rate-limit.js.map +1 -0
- package/dist/gateway/routes.js +315 -0
- package/dist/gateway/routes.js.map +1 -0
- package/dist/gateway/state.js +54 -0
- package/dist/gateway/state.js.map +1 -0
- package/dist/gateway/ws-handler.js +200 -0
- package/dist/gateway/ws-handler.js.map +1 -0
- package/dist/gateway-entry.js +16 -0
- package/dist/gateway-entry.js.map +1 -0
- package/dist/hooks.js +72 -0
- package/dist/hooks.js.map +1 -0
- package/dist/lanes.js +62 -0
- package/dist/lanes.js.map +1 -0
- package/dist/model.js +30 -0
- package/dist/model.js.map +1 -0
- package/dist/policy.js +22 -0
- package/dist/policy.js.map +1 -0
- package/dist/queue.js +45 -0
- package/dist/queue.js.map +1 -0
- package/dist/retry.js +96 -0
- package/dist/retry.js.map +1 -0
- package/dist/runs.js +83 -0
- package/dist/runs.js.map +1 -0
- package/dist/runtime/compact.js +99 -0
- package/dist/runtime/compact.js.map +1 -0
- package/dist/runtime/lanes.js +66 -0
- package/dist/runtime/lanes.js.map +1 -0
- package/dist/runtime/orchestrate.js +121 -0
- package/dist/runtime/orchestrate.js.map +1 -0
- package/dist/runtime/queue.js +50 -0
- package/dist/runtime/queue.js.map +1 -0
- package/dist/runtime/retry.js +127 -0
- package/dist/runtime/retry.js.map +1 -0
- package/dist/runtime/runs.js +105 -0
- package/dist/runtime/runs.js.map +1 -0
- package/dist/serve.js +209 -0
- package/dist/serve.js.map +1 -0
- package/dist/session.js +75 -0
- package/dist/session.js.map +1 -0
- package/dist/setup.js +254 -0
- package/dist/setup.js.map +1 -0
- package/dist/skills.js +89 -0
- package/dist/skills.js.map +1 -0
- package/dist/subagent.js +71 -0
- package/dist/subagent.js.map +1 -0
- package/dist/system-prompt.js +157 -0
- package/dist/system-prompt.js.map +1 -0
- package/dist/telegram/admin-bot.js +705 -0
- package/dist/telegram/admin-bot.js.map +1 -0
- package/dist/telegram/agent-bot.js +551 -0
- package/dist/telegram/agent-bot.js.map +1 -0
- package/dist/telegram/bot-approval.js +63 -0
- package/dist/telegram/bot-approval.js.map +1 -0
- package/dist/telegram/draft-stream.js +86 -0
- package/dist/telegram/draft-stream.js.map +1 -0
- package/dist/telegram/format.js +106 -0
- package/dist/telegram/format.js.map +1 -0
- package/dist/telegram/helpers.js +87 -0
- package/dist/telegram/helpers.js.map +1 -0
- package/dist/telegram/pairing-notify.js +52 -0
- package/dist/telegram/pairing-notify.js.map +1 -0
- package/dist/telegram/pairing.js +138 -0
- package/dist/telegram/pairing.js.map +1 -0
- package/dist/telegram/resolve.js +33 -0
- package/dist/telegram/resolve.js.map +1 -0
- package/dist/telegram/transcribe.js +77 -0
- package/dist/telegram/transcribe.js.map +1 -0
- package/dist/telegram/types.js +3 -0
- package/dist/telegram/types.js.map +1 -0
- package/dist/telegram/voice-wizard.js +84 -0
- package/dist/telegram/voice-wizard.js.map +1 -0
- package/dist/telegram/wizard.js +89 -0
- package/dist/telegram/wizard.js.map +1 -0
- package/dist/telegram/wizards.js +297 -0
- package/dist/telegram/wizards.js.map +1 -0
- package/dist/telegram-admin.js +800 -0
- package/dist/telegram-admin.js.map +1 -0
- package/dist/telegram.js +118 -0
- package/dist/telegram.js.map +1 -0
- package/dist/tools/cron.js +94 -0
- package/dist/tools/cron.js.map +1 -0
- package/dist/tools/edit.js +29 -0
- package/dist/tools/edit.js.map +1 -0
- package/dist/tools/exec.js +38 -0
- package/dist/tools/exec.js.map +1 -0
- package/dist/tools/fetch.js +28 -0
- package/dist/tools/fetch.js.map +1 -0
- package/dist/tools/index.js +16 -0
- package/dist/tools/index.js.map +1 -0
- package/dist/tools/memory.js +164 -0
- package/dist/tools/memory.js.map +1 -0
- package/dist/tools/patch.js +284 -0
- package/dist/tools/patch.js.map +1 -0
- package/dist/tools/read.js +26 -0
- package/dist/tools/read.js.map +1 -0
- package/dist/tools/search.js +62 -0
- package/dist/tools/search.js.map +1 -0
- package/dist/tools/subagent.js +48 -0
- package/dist/tools/subagent.js.map +1 -0
- package/dist/tools/write.js +22 -0
- package/dist/tools/write.js.map +1 -0
- package/dist/tui/commands.js +450 -0
- package/dist/tui/commands.js.map +1 -0
- package/dist/tui/components/assistant-message.js +26 -0
- package/dist/tui/components/assistant-message.js.map +1 -0
- package/dist/tui/components/chat-log.js +94 -0
- package/dist/tui/components/chat-log.js.map +1 -0
- package/dist/tui/components/custom-editor.js +40 -0
- package/dist/tui/components/custom-editor.js.map +1 -0
- package/dist/tui/components/hint-bar.js +13 -0
- package/dist/tui/components/hint-bar.js.map +1 -0
- package/dist/tui/components/tool-execution.js +73 -0
- package/dist/tui/components/tool-execution.js.map +1 -0
- package/dist/tui/components/user-message.js +19 -0
- package/dist/tui/components/user-message.js.map +1 -0
- package/dist/tui/components/welcome.js +147 -0
- package/dist/tui/components/welcome.js.map +1 -0
- package/dist/tui/context.js +3 -0
- package/dist/tui/context.js.map +1 -0
- package/dist/tui/theme.js +91 -0
- package/dist/tui/theme.js.map +1 -0
- package/dist/tui/tui.js +389 -0
- package/dist/tui/tui.js.map +1 -0
- package/dist/tui/ws-handler.js +154 -0
- package/dist/tui/ws-handler.js.map +1 -0
- package/dist/types.js +3 -0
- package/dist/types.js.map +1 -0
- package/dist/usage.js +88 -0
- package/dist/usage.js.map +1 -0
- package/dist/workspace.js +245 -0
- package/dist/workspace.js.map +1 -0
- 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":"AAAA,oDAAoD;AAYpD,MAAM,UAAU,aAAa,CAAC,GAAU;IACtC,MAAM,GAAG,GAAG,GAAG,CAAC,OAAO,CAAC,WAAW,EAAE,CAAC;IAEtC,qDAAqD;IACrD,IAAI,GAAG,KAAK,SAAS,IAAI,GAAG,KAAK,2BAA2B,IAAI,GAAG,KAAK,4BAA4B,EAAE,CAAC;QACrG,OAAO,OAAO,CAAC;IACjB,CAAC;IAED,cAAc;IACd,IAAI,GAAG,CAAC,QAAQ,CAAC,KAAK,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,KAAK,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,cAAc,CAAC;WACzE,GAAG,CAAC,QAAQ,CAAC,iBAAiB,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,eAAe,CAAC,EAAE,CAAC;QACtE,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,UAAU;IACV,IAAI,GAAG,CAAC,QAAQ,CAAC,KAAK,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,cAAc,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,kBAAkB,CAAC;WACtF,GAAG,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC;QAC7B,OAAO,SAAS,CAAC;IACnB,CAAC;IAED,aAAa;IACb,IAAI,GAAG,CAAC,QAAQ,CAAC,KAAK,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,YAAY,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,mBAAmB,CAAC;WACrF,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,oBAAoB,CAAC,EAAE,CAAC;QACjE,OAAO,YAAY,CAAC;IACtB,CAAC;IAED,mBAAmB;IACnB,IAAI,GAAG,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,UAAU,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC;WACjF,GAAG,CAAC,QAAQ,CAAC,oBAAoB,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,mBAAmB,CAAC;WACvE,GAAG,CAAC,QAAQ,CAAC,wBAAwB,CAAC,EAAE,CAAC;QAC5C,OAAO,UAAU,CAAC;IACpB,CAAC;IAED,UAAU;IACV,IAAI,GAAG,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,mBAAmB,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,SAAS,CAAC;WACtF,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;QAC3B,OAAO,SAAS,CAAC;IACnB,CAAC;IAED,oCAAoC;IACpC,IAAI,GAAG,CAAC,QAAQ,CAAC,KAAK,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,iBAAiB,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,YAAY,CAAC,EAAE,CAAC;QACzF,OAAO,QAAQ,CAAC;IAClB,CAAC;IAED,sCAAsC;IACtC,IAAI,GAAG,CAAC,QAAQ,CAAC,KAAK,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,KAAK,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,KAAK,CAAC;WAChE,GAAG,CAAC,QAAQ,CAAC,qBAAqB,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,uBAAuB,CAAC,EAAE,CAAC;QAClF,OAAO,YAAY,CAAC,CAAC,qBAAqB;IAC5C,CAAC;IAED,OAAO,SAAS,CAAC;AACnB,CAAC;AAED,MAAM,UAAU,WAAW,CAAC,IAAe;IACzC,OAAO,IAAI,KAAK,YAAY,IAAI,IAAI,KAAK,SAAS,CAAC;AACrD,CAAC;AASD;;;;;GAKG;AACH,MAAM,CAAC,KAAK,UAAU,SAAS,CAC7B,EAAoB,EACpB,IAAe;IAEf,IAAI,SAA4B,CAAC;IACjC,IAAI,eAAe,GAAG,KAAK,CAAC;IAE5B,KAAK,IAAI,OAAO,GAAG,CAAC,EAAE,OAAO,IAAI,IAAI,CAAC,UAAU,EAAE,OAAO,EAAE,EAAE,CAAC;QAC5D,IAAI,CAAC;YACH,OAAO,MAAM,EAAE,EAAE,CAAC;QACpB,CAAC;QAAC,OAAO,GAAY,EAAE,CAAC;YACtB,SAAS,GAAG,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;YAChE,MAAM,IAAI,GAAG,aAAa,CAAC,SAAS,CAAC,CAAC;YAEtC,uBAAuB;YACvB,IAAI,IAAI,KAAK,MAAM,IAAI,IAAI,KAAK,SAAS,IAAI,IAAI,KAAK,QAAQ,IAAI,IAAI,KAAK,OAAO,EAAE,CAAC;gBACnF,MAAM,SAAS,CAAC;YAClB,CAAC;YAED,mCAAmC;YACnC,IAAI,IAAI,KAAK,UAAU,IAAI,CAAC,eAAe,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;gBAC9D,eAAe,GAAG,IAAI,CAAC;gBACvB,IAAI,CAAC,OAAO,EAAE,CAAC,OAAO,EAAE,IAAI,EAAE,SAAS,CAAC,CAAC;gBACzC,MAAM,IAAI,CAAC,SAAS,EAAE,CAAC;gBACvB,SAAS;YACX,CAAC;YAED,6BAA6B;YAC7B,IAAI,OAAO,KAAK,IAAI,CAAC,UAAU;gBAAE,MAAM;YAEvC,qBAAqB;YACrB,IAAI,WAAW,CAAC,IAAI,CAAC,EAAE,CAAC;gBACtB,IAAI,CAAC,OAAO,EAAE,CAAC,OAAO,EAAE,IAAI,EAAE,SAAS,CAAC,CAAC;gBACzC,MAAM,KAAK,GAAG,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC;gBACpD,MAAM,IAAI,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,UAAU,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,CAAC;gBAC/C,SAAS;YACX,CAAC;YAED,uCAAuC;YACvC,IAAI,IAAI,KAAK,SAAS,IAAI,OAAO,KAAK,CAAC,EAAE,CAAC;gBACxC,IAAI,CAAC,OAAO,EAAE,CAAC,OAAO,EAAE,IAAI,EAAE,SAAS,CAAC,CAAC;gBACzC,MAAM,IAAI,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,UAAU,CAAC,CAAC,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC;gBACxD,SAAS;YACX,CAAC;YAED,MAAM;QACR,CAAC;IACH,CAAC;IAED,MAAM,SAAU,CAAC;AACnB,CAAC"}
|