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,105 @@
|
|
|
1
|
+
// Run tracking: prevents concurrent runs on the same session
|
|
2
|
+
import { QUEUE_WAIT_TIMEOUT_MS } from "../core/constants.js";
|
|
3
|
+
export function createRunTracker() {
|
|
4
|
+
// Primary index: runId -> handle
|
|
5
|
+
const runsByRunId = new Map();
|
|
6
|
+
// Secondary index: sessionId -> runId (latest)
|
|
7
|
+
const sessionToRunId = new Map();
|
|
8
|
+
const waiters = new Map();
|
|
9
|
+
let runCounter = 0;
|
|
10
|
+
function generateRunId() {
|
|
11
|
+
return `run-${Date.now()}-${++runCounter}`;
|
|
12
|
+
}
|
|
13
|
+
function setActiveRun(sessionId, handle) {
|
|
14
|
+
// Abort any existing run for this session
|
|
15
|
+
const existingRunId = sessionToRunId.get(sessionId);
|
|
16
|
+
if (existingRunId) {
|
|
17
|
+
const existing = runsByRunId.get(existingRunId);
|
|
18
|
+
if (existing)
|
|
19
|
+
existing.abort();
|
|
20
|
+
runsByRunId.delete(existingRunId);
|
|
21
|
+
}
|
|
22
|
+
runsByRunId.set(handle.runId, handle);
|
|
23
|
+
sessionToRunId.set(sessionId, handle.runId);
|
|
24
|
+
}
|
|
25
|
+
function clearActiveRun(runId) {
|
|
26
|
+
const handle = runsByRunId.get(runId);
|
|
27
|
+
if (!handle)
|
|
28
|
+
return;
|
|
29
|
+
runsByRunId.delete(runId);
|
|
30
|
+
// Only clear session mapping if this is still the latest run
|
|
31
|
+
if (sessionToRunId.get(handle.sessionId) === runId) {
|
|
32
|
+
sessionToRunId.delete(handle.sessionId);
|
|
33
|
+
}
|
|
34
|
+
// Notify waiters for this session
|
|
35
|
+
const sessionWaiters = waiters.get(handle.sessionId);
|
|
36
|
+
if (sessionWaiters) {
|
|
37
|
+
for (const resolve of sessionWaiters)
|
|
38
|
+
resolve(true);
|
|
39
|
+
waiters.delete(handle.sessionId);
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
function isRunActive(sessionId) {
|
|
43
|
+
const runId = sessionToRunId.get(sessionId);
|
|
44
|
+
return runId !== undefined && runsByRunId.has(runId);
|
|
45
|
+
}
|
|
46
|
+
function getActiveRun(sessionId) {
|
|
47
|
+
const runId = sessionToRunId.get(sessionId);
|
|
48
|
+
return runId ? runsByRunId.get(runId) : undefined;
|
|
49
|
+
}
|
|
50
|
+
function abortRun(sessionId) {
|
|
51
|
+
const runId = sessionToRunId.get(sessionId);
|
|
52
|
+
if (!runId)
|
|
53
|
+
return false;
|
|
54
|
+
const handle = runsByRunId.get(runId);
|
|
55
|
+
if (handle) {
|
|
56
|
+
handle.abort();
|
|
57
|
+
clearActiveRun(runId);
|
|
58
|
+
return true;
|
|
59
|
+
}
|
|
60
|
+
return false;
|
|
61
|
+
}
|
|
62
|
+
function waitForRunEnd(sessionId, timeoutMs = QUEUE_WAIT_TIMEOUT_MS) {
|
|
63
|
+
if (!isRunActive(sessionId))
|
|
64
|
+
return Promise.resolve(true);
|
|
65
|
+
return new Promise((resolve) => {
|
|
66
|
+
const timer = setTimeout(() => {
|
|
67
|
+
const set = waiters.get(sessionId);
|
|
68
|
+
if (set)
|
|
69
|
+
set.delete(wrappedResolve);
|
|
70
|
+
resolve(false);
|
|
71
|
+
}, timeoutMs);
|
|
72
|
+
const wrappedResolve = (ended) => {
|
|
73
|
+
clearTimeout(timer);
|
|
74
|
+
resolve(ended);
|
|
75
|
+
};
|
|
76
|
+
if (!waiters.has(sessionId))
|
|
77
|
+
waiters.set(sessionId, new Set());
|
|
78
|
+
waiters.get(sessionId).add(wrappedResolve);
|
|
79
|
+
});
|
|
80
|
+
}
|
|
81
|
+
function getActiveRunCount() {
|
|
82
|
+
return runsByRunId.size;
|
|
83
|
+
}
|
|
84
|
+
function acquireRun(sessionId, handle) {
|
|
85
|
+
if (isRunActive(sessionId))
|
|
86
|
+
return false;
|
|
87
|
+
setActiveRun(sessionId, handle);
|
|
88
|
+
return true;
|
|
89
|
+
}
|
|
90
|
+
function reset() {
|
|
91
|
+
runsByRunId.clear();
|
|
92
|
+
sessionToRunId.clear();
|
|
93
|
+
waiters.clear();
|
|
94
|
+
runCounter = 0;
|
|
95
|
+
}
|
|
96
|
+
return {
|
|
97
|
+
generateRunId, setActiveRun, clearActiveRun, isRunActive,
|
|
98
|
+
getActiveRun, abortRun, waitForRunEnd, getActiveRunCount,
|
|
99
|
+
acquireRun, reset,
|
|
100
|
+
};
|
|
101
|
+
}
|
|
102
|
+
// Backward-compat singleton
|
|
103
|
+
const defaultTracker = createRunTracker();
|
|
104
|
+
export const { generateRunId, setActiveRun, clearActiveRun, isRunActive, getActiveRun, abortRun, waitForRunEnd, getActiveRunCount } = defaultTracker;
|
|
105
|
+
//# sourceMappingURL=runs.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"runs.js","sourceRoot":"","sources":["../../src/runtime/runs.ts"],"names":[],"mappings":"AAAA,6DAA6D;AAE7D,OAAO,EAAE,qBAAqB,EAAE,MAAM,sBAAsB,CAAC;AAyB7D,MAAM,UAAU,gBAAgB;IAC9B,iCAAiC;IACjC,MAAM,WAAW,GAAG,IAAI,GAAG,EAAqB,CAAC;IACjD,+CAA+C;IAC/C,MAAM,cAAc,GAAG,IAAI,GAAG,EAAkB,CAAC;IACjD,MAAM,OAAO,GAAG,IAAI,GAAG,EAAyC,CAAC;IACjE,IAAI,UAAU,GAAG,CAAC,CAAC;IAEnB,SAAS,aAAa;QACpB,OAAO,OAAO,IAAI,CAAC,GAAG,EAAE,IAAI,EAAE,UAAU,EAAE,CAAC;IAC7C,CAAC;IAED,SAAS,YAAY,CAAC,SAAiB,EAAE,MAAiB;QACxD,0CAA0C;QAC1C,MAAM,aAAa,GAAG,cAAc,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QACpD,IAAI,aAAa,EAAE,CAAC;YAClB,MAAM,QAAQ,GAAG,WAAW,CAAC,GAAG,CAAC,aAAa,CAAC,CAAC;YAChD,IAAI,QAAQ;gBAAE,QAAQ,CAAC,KAAK,EAAE,CAAC;YAC/B,WAAW,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC;QACpC,CAAC;QACD,WAAW,CAAC,GAAG,CAAC,MAAM,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC;QACtC,cAAc,CAAC,GAAG,CAAC,SAAS,EAAE,MAAM,CAAC,KAAK,CAAC,CAAC;IAC9C,CAAC;IAED,SAAS,cAAc,CAAC,KAAa;QACnC,MAAM,MAAM,GAAG,WAAW,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QACtC,IAAI,CAAC,MAAM;YAAE,OAAO;QAEpB,WAAW,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QAE1B,6DAA6D;QAC7D,IAAI,cAAc,CAAC,GAAG,CAAC,MAAM,CAAC,SAAS,CAAC,KAAK,KAAK,EAAE,CAAC;YACnD,cAAc,CAAC,MAAM,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;QAC1C,CAAC;QAED,kCAAkC;QAClC,MAAM,cAAc,GAAG,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;QACrD,IAAI,cAAc,EAAE,CAAC;YACnB,KAAK,MAAM,OAAO,IAAI,cAAc;gBAAE,OAAO,CAAC,IAAI,CAAC,CAAC;YACpD,OAAO,CAAC,MAAM,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;QACnC,CAAC;IACH,CAAC;IAED,SAAS,WAAW,CAAC,SAAiB;QACpC,MAAM,KAAK,GAAG,cAAc,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAC5C,OAAO,KAAK,KAAK,SAAS,IAAI,WAAW,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;IACvD,CAAC;IAED,SAAS,YAAY,CAAC,SAAiB;QACrC,MAAM,KAAK,GAAG,cAAc,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAC5C,OAAO,KAAK,CAAC,CAAC,CAAC,WAAW,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;IACpD,CAAC;IAED,SAAS,QAAQ,CAAC,SAAiB;QACjC,MAAM,KAAK,GAAG,cAAc,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAC5C,IAAI,CAAC,KAAK;YAAE,OAAO,KAAK,CAAC;QACzB,MAAM,MAAM,GAAG,WAAW,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QACtC,IAAI,MAAM,EAAE,CAAC;YACX,MAAM,CAAC,KAAK,EAAE,CAAC;YACf,cAAc,CAAC,KAAK,CAAC,CAAC;YACtB,OAAO,IAAI,CAAC;QACd,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAED,SAAS,aAAa,CAAC,SAAiB,EAAE,SAAS,GAAG,qBAAqB;QACzE,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC;YAAE,OAAO,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QAE1D,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;YAC7B,MAAM,KAAK,GAAG,UAAU,CAAC,GAAG,EAAE;gBAC5B,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;gBACnC,IAAI,GAAG;oBAAE,GAAG,CAAC,MAAM,CAAC,cAAc,CAAC,CAAC;gBACpC,OAAO,CAAC,KAAK,CAAC,CAAC;YACjB,CAAC,EAAE,SAAS,CAAC,CAAC;YAEd,MAAM,cAAc,GAAG,CAAC,KAAc,EAAE,EAAE;gBACxC,YAAY,CAAC,KAAK,CAAC,CAAC;gBACpB,OAAO,CAAC,KAAK,CAAC,CAAC;YACjB,CAAC,CAAC;YAEF,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,SAAS,CAAC;gBAAE,OAAO,CAAC,GAAG,CAAC,SAAS,EAAE,IAAI,GAAG,EAAE,CAAC,CAAC;YAC/D,OAAO,CAAC,GAAG,CAAC,SAAS,CAAE,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC;QAC9C,CAAC,CAAC,CAAC;IACL,CAAC;IAED,SAAS,iBAAiB;QACxB,OAAO,WAAW,CAAC,IAAI,CAAC;IAC1B,CAAC;IAED,SAAS,UAAU,CAAC,SAAiB,EAAE,MAAiB;QACtD,IAAI,WAAW,CAAC,SAAS,CAAC;YAAE,OAAO,KAAK,CAAC;QACzC,YAAY,CAAC,SAAS,EAAE,MAAM,CAAC,CAAC;QAChC,OAAO,IAAI,CAAC;IACd,CAAC;IAED,SAAS,KAAK;QACZ,WAAW,CAAC,KAAK,EAAE,CAAC;QACpB,cAAc,CAAC,KAAK,EAAE,CAAC;QACvB,OAAO,CAAC,KAAK,EAAE,CAAC;QAChB,UAAU,GAAG,CAAC,CAAC;IACjB,CAAC;IAED,OAAO;QACL,aAAa,EAAE,YAAY,EAAE,cAAc,EAAE,WAAW;QACxD,YAAY,EAAE,QAAQ,EAAE,aAAa,EAAE,iBAAiB;QACxD,UAAU,EAAE,KAAK;KAClB,CAAC;AACJ,CAAC;AAED,4BAA4B;AAC5B,MAAM,cAAc,GAAG,gBAAgB,EAAE,CAAC;AAC1C,MAAM,CAAC,MAAM,EAAE,aAAa,EAAE,YAAY,EAAE,cAAc,EAAE,WAAW,EACrE,YAAY,EAAE,QAAQ,EAAE,aAAa,EAAE,iBAAiB,EAAE,GAAG,cAAc,CAAC"}
|
package/dist/serve.js
ADDED
|
@@ -0,0 +1,209 @@
|
|
|
1
|
+
// Gateway server: Express + WebSocket — single orchestration point
|
|
2
|
+
// Routes and WS logic are in gateway/*.ts
|
|
3
|
+
import express from "express";
|
|
4
|
+
import { createServer } from "node:http";
|
|
5
|
+
import { WebSocketServer } from "ws";
|
|
6
|
+
import { loadConfig, ensureDirs, onConfigSaved } from "./core/config.js";
|
|
7
|
+
import { createClient } from "./model.js";
|
|
8
|
+
import { seedWorkspace } from "./workspace.js";
|
|
9
|
+
import { buildSystemPrompt } from "./system-prompt.js";
|
|
10
|
+
import { startCronJob, stopAllCronJobs, startRuntimeJobs, setCronContext } from "./extensions/cron.js";
|
|
11
|
+
import { configureLane, Lane } from "./runtime/lanes.js";
|
|
12
|
+
import { runBoot } from "./boot.js";
|
|
13
|
+
import { errorMessage } from "./core/errors.js";
|
|
14
|
+
import { log as slog } from "./core/log.js";
|
|
15
|
+
import { HEARTBEAT_INTERVAL_MS } from "./core/constants.js";
|
|
16
|
+
import { registerRoutes } from "./gateway/routes.js";
|
|
17
|
+
import { registerWsHandler } from "./gateway/ws-handler.js";
|
|
18
|
+
import { requestLogger } from "./gateway/logger.js";
|
|
19
|
+
import { rateLimit } from "./gateway/rate-limit.js";
|
|
20
|
+
import { csrfProtection } from "./gateway/csrf.js";
|
|
21
|
+
import fs from "node:fs";
|
|
22
|
+
export async function startServer(opts = {}) {
|
|
23
|
+
ensureDirs();
|
|
24
|
+
seedWorkspace();
|
|
25
|
+
const config = loadConfig();
|
|
26
|
+
console.log(`[serve] Startup config — agents: [${Object.keys(config.agents)}], apiKey: ${config.apiKey ? "set" : "none"}, hasTgToken: ${!!config.telegram.botToken}`);
|
|
27
|
+
const state = {
|
|
28
|
+
config,
|
|
29
|
+
client: createClient(config),
|
|
30
|
+
systemPrompt: buildSystemPrompt(config.systemPrompt, config.skills),
|
|
31
|
+
token: config.serve.token,
|
|
32
|
+
silent: !!opts.silent,
|
|
33
|
+
clients: new Set(),
|
|
34
|
+
startTime: Date.now(),
|
|
35
|
+
};
|
|
36
|
+
configureLane(Lane.Main, config.lanes.main);
|
|
37
|
+
configureLane(Lane.Cron, config.lanes.cron);
|
|
38
|
+
configureLane(Lane.Subagent, config.lanes.subagent);
|
|
39
|
+
// Immediately sync in-memory state on every saveConfig call (no debounce)
|
|
40
|
+
onConfigSaved((fresh) => {
|
|
41
|
+
state.config = fresh;
|
|
42
|
+
state.systemPrompt = buildSystemPrompt(fresh.systemPrompt, fresh.skills);
|
|
43
|
+
});
|
|
44
|
+
// Set cron context so runtime-added jobs can auto-start
|
|
45
|
+
setCronContext(state.config, state.systemPrompt);
|
|
46
|
+
const requestedPort = opts.port ?? config.serve.port;
|
|
47
|
+
const host = opts.host ?? config.serve.host;
|
|
48
|
+
const log = opts.silent ? (..._a) => { } : console.log;
|
|
49
|
+
const app = express();
|
|
50
|
+
app.use(express.json({ limit: "1mb" }));
|
|
51
|
+
app.use(csrfProtection());
|
|
52
|
+
if (!opts.silent) {
|
|
53
|
+
app.use(requestLogger());
|
|
54
|
+
}
|
|
55
|
+
app.use(rateLimit(config.serve.rateLimit));
|
|
56
|
+
const server = createServer(app);
|
|
57
|
+
const wss = new WebSocketServer({ server });
|
|
58
|
+
// Heartbeat
|
|
59
|
+
const alive = new WeakMap();
|
|
60
|
+
const heartbeat = setInterval(() => {
|
|
61
|
+
for (const ws of state.clients) {
|
|
62
|
+
if (!alive.get(ws)) {
|
|
63
|
+
state.clients.delete(ws);
|
|
64
|
+
ws.terminate();
|
|
65
|
+
continue;
|
|
66
|
+
}
|
|
67
|
+
alive.set(ws, false);
|
|
68
|
+
ws.ping();
|
|
69
|
+
}
|
|
70
|
+
}, HEARTBEAT_INTERVAL_MS);
|
|
71
|
+
// Track pong per-client in WS handler
|
|
72
|
+
wss.on("connection", (ws) => {
|
|
73
|
+
alive.set(ws, true);
|
|
74
|
+
ws.on("pong", () => alive.set(ws, true));
|
|
75
|
+
});
|
|
76
|
+
// Register handlers
|
|
77
|
+
registerRoutes(app, state);
|
|
78
|
+
registerWsHandler(wss, state);
|
|
79
|
+
// Start listening
|
|
80
|
+
const actualPort = await new Promise((resolve) => {
|
|
81
|
+
server.listen(requestedPort, host, () => {
|
|
82
|
+
const addr = server.address();
|
|
83
|
+
resolve(typeof addr === "object" && addr ? addr.port : requestedPort);
|
|
84
|
+
});
|
|
85
|
+
});
|
|
86
|
+
log(`CamelAGI gateway listening on ${host}:${actualPort}`);
|
|
87
|
+
log(` HTTP: http://${host}:${actualPort}/health`);
|
|
88
|
+
log(` WS: ws://${host}:${actualPort}`);
|
|
89
|
+
// Boot script
|
|
90
|
+
if (opts.boot !== false && state.config.boot) {
|
|
91
|
+
try {
|
|
92
|
+
const bootResult = await runBoot(state.config, state.systemPrompt);
|
|
93
|
+
if (bootResult.status === "ran") {
|
|
94
|
+
log(` BOOT.md: ${bootResult.response?.slice(0, 80) ?? "done"}`);
|
|
95
|
+
}
|
|
96
|
+
else if (bootResult.status === "failed") {
|
|
97
|
+
log(` BOOT.md failed: ${bootResult.error}`);
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
catch { /* best effort */ }
|
|
101
|
+
}
|
|
102
|
+
// Channels (Telegram, Discord, Slack, etc.)
|
|
103
|
+
if (opts.channels !== false) {
|
|
104
|
+
try {
|
|
105
|
+
const { loadChannels, startAllChannels } = await import("./channels/index.js");
|
|
106
|
+
await loadChannels(state.config);
|
|
107
|
+
const started = await startAllChannels(() => state.config, () => state.systemPrompt);
|
|
108
|
+
for (const [type, ids] of started) {
|
|
109
|
+
log(` ${type}: ${ids.length} bot(s) started [${ids.join(", ")}]`);
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
catch (err) {
|
|
113
|
+
slog.error("channels", "Failed to start", { error: errorMessage(err) });
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
// Cron (config-defined + runtime-defined)
|
|
117
|
+
if (opts.cron !== false) {
|
|
118
|
+
const cronOpts = {
|
|
119
|
+
onRun: (id, response) => { log(` Cron ${id}: ${response.slice(0, 80)}`); },
|
|
120
|
+
onError: (id, err) => { slog.error("cron", `Job ${id} failed`, { jobId: id, error: err.message }); },
|
|
121
|
+
};
|
|
122
|
+
const enabledJobs = state.config.cron.filter((j) => j.enabled);
|
|
123
|
+
for (const job of enabledJobs) {
|
|
124
|
+
startCronJob({ ...job, source: "config" }, state.config, state.systemPrompt, cronOpts);
|
|
125
|
+
}
|
|
126
|
+
const runtimeCount = startRuntimeJobs(state.config, state.systemPrompt, cronOpts);
|
|
127
|
+
const totalCount = enabledJobs.length + runtimeCount;
|
|
128
|
+
if (totalCount > 0) {
|
|
129
|
+
log(` ${totalCount} cron job(s) started (${enabledJobs.length} config, ${runtimeCount} runtime)`);
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
// Config hot-reload
|
|
133
|
+
const channelsEnabled = opts.channels !== false;
|
|
134
|
+
const configWatcher = watchConfig(state.config, (newConfig) => {
|
|
135
|
+
const oldAgentKeys = Object.keys(state.config.agents);
|
|
136
|
+
const newAgentKeys = Object.keys(newConfig.agents);
|
|
137
|
+
state.config = newConfig;
|
|
138
|
+
state.systemPrompt = buildSystemPrompt(state.config.systemPrompt, state.config.skills);
|
|
139
|
+
console.log(`[serve] Config reloaded — agents: [${oldAgentKeys}] → [${newAgentKeys}]`);
|
|
140
|
+
configureLane(Lane.Main, state.config.lanes.main);
|
|
141
|
+
configureLane(Lane.Cron, state.config.lanes.cron);
|
|
142
|
+
configureLane(Lane.Subagent, state.config.lanes.subagent);
|
|
143
|
+
setCronContext(state.config, state.systemPrompt);
|
|
144
|
+
// Reconcile channels: start new bots, stop removed ones
|
|
145
|
+
if (channelsEnabled) {
|
|
146
|
+
import("./channels/index.js")
|
|
147
|
+
.then(({ reconcileAllChannels }) => reconcileAllChannels(() => state.config, () => state.systemPrompt))
|
|
148
|
+
.catch(() => { });
|
|
149
|
+
}
|
|
150
|
+
});
|
|
151
|
+
// Close handle
|
|
152
|
+
const close = async () => {
|
|
153
|
+
clearInterval(heartbeat);
|
|
154
|
+
configWatcher?.close();
|
|
155
|
+
stopAllCronJobs();
|
|
156
|
+
for (const ws of state.clients)
|
|
157
|
+
ws.close(1001, "Server shutting down");
|
|
158
|
+
try {
|
|
159
|
+
const { stopAllChannels } = await import("./channels/index.js");
|
|
160
|
+
stopAllChannels();
|
|
161
|
+
}
|
|
162
|
+
catch { /* channels not loaded */ }
|
|
163
|
+
await new Promise((resolve) => server.close(() => resolve()));
|
|
164
|
+
};
|
|
165
|
+
if (!opts.silent) {
|
|
166
|
+
const shutdown = async () => {
|
|
167
|
+
console.log("\nShutting down...");
|
|
168
|
+
await close();
|
|
169
|
+
process.exit(0);
|
|
170
|
+
};
|
|
171
|
+
process.on("SIGINT", shutdown);
|
|
172
|
+
process.on("SIGTERM", shutdown);
|
|
173
|
+
}
|
|
174
|
+
return { port: actualPort, close, config: state.config, client: state.client, systemPrompt: state.systemPrompt };
|
|
175
|
+
}
|
|
176
|
+
function watchConfig(_initialConfig, onChange) {
|
|
177
|
+
try {
|
|
178
|
+
const configDir = `${process.env.HOME}/.camelagi`;
|
|
179
|
+
// Watch the DIRECTORY, not the file — so we detect config.yaml being created
|
|
180
|
+
// (e.g. after a reset + onboarding)
|
|
181
|
+
if (!fs.existsSync(configDir)) {
|
|
182
|
+
fs.mkdirSync(configDir, { recursive: true });
|
|
183
|
+
}
|
|
184
|
+
let debounce = null;
|
|
185
|
+
const watcher = fs.watch(configDir, (_event, filename) => {
|
|
186
|
+
if (filename !== "config.yaml")
|
|
187
|
+
return;
|
|
188
|
+
console.log(`[watchConfig] config.yaml changed (event: ${_event})`);
|
|
189
|
+
if (debounce)
|
|
190
|
+
clearTimeout(debounce);
|
|
191
|
+
debounce = setTimeout(() => {
|
|
192
|
+
try {
|
|
193
|
+
const newConfig = loadConfig();
|
|
194
|
+
onChange(newConfig);
|
|
195
|
+
}
|
|
196
|
+
catch (err) {
|
|
197
|
+
console.error(`[watchConfig] Failed to reload config:`, err);
|
|
198
|
+
}
|
|
199
|
+
}, 500);
|
|
200
|
+
});
|
|
201
|
+
console.log(`[watchConfig] Watching ${configDir} for config.yaml changes`);
|
|
202
|
+
return watcher;
|
|
203
|
+
}
|
|
204
|
+
catch (err) {
|
|
205
|
+
console.error(`[watchConfig] FAILED to set up watcher:`, err);
|
|
206
|
+
return null;
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
//# sourceMappingURL=serve.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"serve.js","sourceRoot":"","sources":["../src/serve.ts"],"names":[],"mappings":"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"}
|
package/dist/session.js
ADDED
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
// Simple JSONL session persistence
|
|
2
|
+
import fs from "node:fs";
|
|
3
|
+
import path from "node:path";
|
|
4
|
+
import { paths } from "./core/config.js";
|
|
5
|
+
import { deleteUsage } from "./usage.js";
|
|
6
|
+
function sessionPath(id) {
|
|
7
|
+
return path.join(paths.sessionsDir, `${encodeURIComponent(id)}.jsonl`);
|
|
8
|
+
}
|
|
9
|
+
export function listSessions() {
|
|
10
|
+
if (!fs.existsSync(paths.sessionsDir))
|
|
11
|
+
return [];
|
|
12
|
+
return fs
|
|
13
|
+
.readdirSync(paths.sessionsDir)
|
|
14
|
+
.filter((f) => f.endsWith(".jsonl"))
|
|
15
|
+
.map((f) => {
|
|
16
|
+
const raw = fs.readFileSync(path.join(paths.sessionsDir, f), "utf-8");
|
|
17
|
+
const firstLine = raw.split("\n")[0];
|
|
18
|
+
try {
|
|
19
|
+
return JSON.parse(firstLine);
|
|
20
|
+
}
|
|
21
|
+
catch {
|
|
22
|
+
return null;
|
|
23
|
+
}
|
|
24
|
+
})
|
|
25
|
+
.filter((m) => m !== null)
|
|
26
|
+
.sort((a, b) => b.createdAt - a.createdAt);
|
|
27
|
+
}
|
|
28
|
+
// Map old LangChain type names to new roles for backward compat
|
|
29
|
+
function typeToRole(type) {
|
|
30
|
+
switch (type) {
|
|
31
|
+
case "human":
|
|
32
|
+
case "user":
|
|
33
|
+
return "user";
|
|
34
|
+
case "ai":
|
|
35
|
+
case "assistant":
|
|
36
|
+
return "assistant";
|
|
37
|
+
case "system":
|
|
38
|
+
return "system";
|
|
39
|
+
case "tool":
|
|
40
|
+
return "tool";
|
|
41
|
+
default:
|
|
42
|
+
return "user";
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
export function loadMessages(sessionId) {
|
|
46
|
+
const file = sessionPath(sessionId);
|
|
47
|
+
if (!fs.existsSync(file))
|
|
48
|
+
return [];
|
|
49
|
+
const lines = fs.readFileSync(file, "utf-8").split("\n").filter(Boolean);
|
|
50
|
+
return lines.slice(1).map((line) => {
|
|
51
|
+
const msg = JSON.parse(line);
|
|
52
|
+
return { role: typeToRole(msg.type), content: msg.content };
|
|
53
|
+
});
|
|
54
|
+
}
|
|
55
|
+
export function saveMessage(sessionId, message, model, label) {
|
|
56
|
+
fs.mkdirSync(paths.sessionsDir, { recursive: true });
|
|
57
|
+
const file = sessionPath(sessionId);
|
|
58
|
+
if (!fs.existsSync(file)) {
|
|
59
|
+
const meta = { id: sessionId, createdAt: Date.now(), model, ...(label && { label }) };
|
|
60
|
+
fs.writeFileSync(file, JSON.stringify(meta) + "\n");
|
|
61
|
+
}
|
|
62
|
+
// Save with new role names
|
|
63
|
+
const serialized = {
|
|
64
|
+
type: message.role,
|
|
65
|
+
content: message.content,
|
|
66
|
+
};
|
|
67
|
+
fs.appendFileSync(file, JSON.stringify(serialized) + "\n");
|
|
68
|
+
}
|
|
69
|
+
export function deleteSession(sessionId) {
|
|
70
|
+
const file = sessionPath(sessionId);
|
|
71
|
+
if (fs.existsSync(file))
|
|
72
|
+
fs.unlinkSync(file);
|
|
73
|
+
deleteUsage(sessionId);
|
|
74
|
+
}
|
|
75
|
+
//# sourceMappingURL=session.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"session.js","sourceRoot":"","sources":["../src/session.ts"],"names":[],"mappings":"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"}
|
package/dist/setup.js
ADDED
|
@@ -0,0 +1,254 @@
|
|
|
1
|
+
// Interactive setup wizard
|
|
2
|
+
import readline from "node:readline";
|
|
3
|
+
import { loadConfig, saveConfig, ensureDirs, paths } from "./core/config.js";
|
|
4
|
+
import { seedWorkspace } from "./workspace.js";
|
|
5
|
+
import { PROVIDER_PRESETS } from "./core/models.js";
|
|
6
|
+
function ask(rl, question) {
|
|
7
|
+
return new Promise((resolve) => rl.question(question, resolve));
|
|
8
|
+
}
|
|
9
|
+
function pick(rl, label, options, compact = false) {
|
|
10
|
+
function showList(items, indices) {
|
|
11
|
+
for (let i = 0; i < indices.length; i++) {
|
|
12
|
+
console.log(` \x1b[33m${indices[i] + 1}\x1b[0m) ${items[i]}`);
|
|
13
|
+
}
|
|
14
|
+
}
|
|
15
|
+
if (!compact) {
|
|
16
|
+
return new Promise((resolve) => {
|
|
17
|
+
console.log(`\n\x1b[36m${label}\x1b[0m`);
|
|
18
|
+
showList(options, options.map((_, i) => i));
|
|
19
|
+
rl.question(`\nPick [1-${options.length}]: `, (answer) => {
|
|
20
|
+
const idx = parseInt(answer.trim(), 10) - 1;
|
|
21
|
+
resolve(options[idx] ?? options[0]);
|
|
22
|
+
});
|
|
23
|
+
});
|
|
24
|
+
}
|
|
25
|
+
// Live-filter mode for large lists
|
|
26
|
+
return new Promise((resolve) => {
|
|
27
|
+
console.log(`\n\x1b[36m${label}\x1b[0m`);
|
|
28
|
+
console.log(`\x1b[90m ${options.length} options — start typing to filter, arrows to navigate, enter to select\x1b[0m\n`);
|
|
29
|
+
rl.pause();
|
|
30
|
+
let query = "";
|
|
31
|
+
let cursor = 0;
|
|
32
|
+
let matches = options.map((o, i) => ({ option: o, index: i }));
|
|
33
|
+
const MAX_VISIBLE = 8;
|
|
34
|
+
function getVisible() {
|
|
35
|
+
if (matches.length <= MAX_VISIBLE)
|
|
36
|
+
return matches;
|
|
37
|
+
let start = Math.max(0, cursor - Math.floor(MAX_VISIBLE / 2));
|
|
38
|
+
if (start + MAX_VISIBLE > matches.length)
|
|
39
|
+
start = Math.max(0, matches.length - MAX_VISIBLE);
|
|
40
|
+
return matches.slice(start, start + MAX_VISIBLE);
|
|
41
|
+
}
|
|
42
|
+
function render() {
|
|
43
|
+
process.stdout.write(`\x1b[2K\r`);
|
|
44
|
+
const lines = [];
|
|
45
|
+
lines.push(`\x1b[36m>\x1b[0m ${query}\x1b[90m_\x1b[0m`);
|
|
46
|
+
lines.push("");
|
|
47
|
+
if (matches.length === 0) {
|
|
48
|
+
lines.push(` \x1b[33mNo matches\x1b[0m`);
|
|
49
|
+
}
|
|
50
|
+
else {
|
|
51
|
+
const visible = getVisible();
|
|
52
|
+
const startIdx = matches.indexOf(visible[0]);
|
|
53
|
+
if (startIdx > 0)
|
|
54
|
+
lines.push(` \x1b[90m ↑ ${startIdx} more\x1b[0m`);
|
|
55
|
+
for (let i = 0; i < visible.length; i++) {
|
|
56
|
+
const m = visible[i];
|
|
57
|
+
const globalIdx = startIdx + i;
|
|
58
|
+
const selected = globalIdx === cursor;
|
|
59
|
+
if (selected) {
|
|
60
|
+
lines.push(` \x1b[36m▸ ${m.index + 1}) ${m.option}\x1b[0m`);
|
|
61
|
+
}
|
|
62
|
+
else {
|
|
63
|
+
lines.push(` \x1b[33m${m.index + 1}\x1b[0m) ${m.option}`);
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
const remaining = matches.length - (startIdx + visible.length);
|
|
67
|
+
if (remaining > 0)
|
|
68
|
+
lines.push(` \x1b[90m ↓ ${remaining} more\x1b[0m`);
|
|
69
|
+
}
|
|
70
|
+
if (render._prevLines) {
|
|
71
|
+
process.stdout.write(`\x1b[${render._prevLines}A`);
|
|
72
|
+
}
|
|
73
|
+
for (const line of lines) {
|
|
74
|
+
process.stdout.write(`\x1b[2K${line}\n`);
|
|
75
|
+
}
|
|
76
|
+
const prevCount = render._prevLines ?? 0;
|
|
77
|
+
for (let i = lines.length; i < prevCount; i++) {
|
|
78
|
+
process.stdout.write(`\x1b[2K\n`);
|
|
79
|
+
}
|
|
80
|
+
if (prevCount > lines.length) {
|
|
81
|
+
process.stdout.write(`\x1b[${prevCount - lines.length}A`);
|
|
82
|
+
}
|
|
83
|
+
render._prevLines = lines.length;
|
|
84
|
+
}
|
|
85
|
+
function updateMatches() {
|
|
86
|
+
const q = query.toLowerCase();
|
|
87
|
+
matches = q
|
|
88
|
+
? options.map((o, i) => ({ option: o, index: i })).filter((m) => m.option.toLowerCase().includes(q))
|
|
89
|
+
: options.map((o, i) => ({ option: o, index: i }));
|
|
90
|
+
cursor = 0;
|
|
91
|
+
}
|
|
92
|
+
const stdin = process.stdin;
|
|
93
|
+
stdin.setRawMode(true);
|
|
94
|
+
stdin.resume();
|
|
95
|
+
render();
|
|
96
|
+
const onData = (buf) => {
|
|
97
|
+
const key = buf.toString();
|
|
98
|
+
if (key === "\r" || key === "\n") {
|
|
99
|
+
stdin.removeListener("data", onData);
|
|
100
|
+
stdin.setRawMode(false);
|
|
101
|
+
stdin.pause();
|
|
102
|
+
rl.resume();
|
|
103
|
+
const prevLines = render._prevLines ?? 0;
|
|
104
|
+
process.stdout.write(`\x1b[${prevLines}A`);
|
|
105
|
+
for (let i = 0; i < prevLines; i++)
|
|
106
|
+
process.stdout.write(`\x1b[2K\n`);
|
|
107
|
+
process.stdout.write(`\x1b[${prevLines}A`);
|
|
108
|
+
if (matches.length > 0) {
|
|
109
|
+
const selected = matches[cursor];
|
|
110
|
+
console.log(` \x1b[32m→ ${selected.option}\x1b[0m\n`);
|
|
111
|
+
resolve(selected.option);
|
|
112
|
+
}
|
|
113
|
+
else if (query.trim()) {
|
|
114
|
+
console.log(` \x1b[32m→ ${query.trim()}\x1b[0m\n`);
|
|
115
|
+
resolve(query.trim());
|
|
116
|
+
}
|
|
117
|
+
else {
|
|
118
|
+
console.log(` \x1b[32m→ ${options[0]}\x1b[0m\n`);
|
|
119
|
+
resolve(options[0]);
|
|
120
|
+
}
|
|
121
|
+
return;
|
|
122
|
+
}
|
|
123
|
+
if (key === "\x03") {
|
|
124
|
+
stdin.removeListener("data", onData);
|
|
125
|
+
stdin.setRawMode(false);
|
|
126
|
+
process.exit(0);
|
|
127
|
+
}
|
|
128
|
+
if (key === "\x1b[A") {
|
|
129
|
+
if (cursor > 0)
|
|
130
|
+
cursor--;
|
|
131
|
+
render();
|
|
132
|
+
return;
|
|
133
|
+
}
|
|
134
|
+
if (key === "\x1b[B") {
|
|
135
|
+
if (cursor < matches.length - 1)
|
|
136
|
+
cursor++;
|
|
137
|
+
render();
|
|
138
|
+
return;
|
|
139
|
+
}
|
|
140
|
+
if (key === "\x7f" || key === "\b") {
|
|
141
|
+
if (query.length > 0) {
|
|
142
|
+
query = query.slice(0, -1);
|
|
143
|
+
updateMatches();
|
|
144
|
+
render();
|
|
145
|
+
}
|
|
146
|
+
return;
|
|
147
|
+
}
|
|
148
|
+
if (key.length === 1 && key >= " ") {
|
|
149
|
+
query += key;
|
|
150
|
+
updateMatches();
|
|
151
|
+
render();
|
|
152
|
+
}
|
|
153
|
+
};
|
|
154
|
+
stdin.on("data", onData);
|
|
155
|
+
});
|
|
156
|
+
}
|
|
157
|
+
export async function runSetup() {
|
|
158
|
+
ensureDirs();
|
|
159
|
+
const rl = readline.createInterface({
|
|
160
|
+
input: process.stdin,
|
|
161
|
+
output: process.stdout,
|
|
162
|
+
});
|
|
163
|
+
console.log(`\n\x1b[36m CamelAGI Setup\x1b[0m`);
|
|
164
|
+
console.log(`\x1b[90m Config: ${paths.configFile}\x1b[0m\n`);
|
|
165
|
+
// Show current config if exists
|
|
166
|
+
try {
|
|
167
|
+
const current = loadConfig();
|
|
168
|
+
console.log(`\x1b[90m Current: provider=${current.provider}, model=${current.model}${current.baseUrl ? `, baseUrl=${current.baseUrl}` : ""}, key=${current.apiKey ? "***" + current.apiKey.slice(-4) : "not set"}\x1b[0m`);
|
|
169
|
+
}
|
|
170
|
+
catch { /* no config yet */ }
|
|
171
|
+
// 1. Pick service
|
|
172
|
+
const service = await pick(rl, "Which service?", [
|
|
173
|
+
"anthropic — Claude (direct)",
|
|
174
|
+
"openai — GPT (direct)",
|
|
175
|
+
"openrouter — Any model via OpenRouter",
|
|
176
|
+
"ollama — Local models",
|
|
177
|
+
"custom — Custom OpenAI-compatible endpoint",
|
|
178
|
+
]);
|
|
179
|
+
const serviceKey = service.split(/\s/)[0];
|
|
180
|
+
const preset = PROVIDER_PRESETS[serviceKey] ?? PROVIDER_PRESETS.custom;
|
|
181
|
+
// 2. API key
|
|
182
|
+
let apiKey;
|
|
183
|
+
if (serviceKey !== "ollama") {
|
|
184
|
+
const keyLabel = serviceKey === "anthropic" ? "Anthropic" : serviceKey === "openai" ? "OpenAI" : serviceKey === "openrouter" ? "OpenRouter" : "API";
|
|
185
|
+
apiKey = await ask(rl, `\n\x1b[36m${keyLabel} API key:\x1b[0m `);
|
|
186
|
+
if (!apiKey.trim()) {
|
|
187
|
+
console.log("\x1b[33m No key entered — you can set it later in config.yaml or via env var.\x1b[0m");
|
|
188
|
+
apiKey = undefined;
|
|
189
|
+
}
|
|
190
|
+
}
|
|
191
|
+
// 3. Base URL (custom only)
|
|
192
|
+
let baseUrl = preset.baseUrl;
|
|
193
|
+
if (serviceKey === "custom") {
|
|
194
|
+
baseUrl = await ask(rl, `\n\x1b[36mBase URL:\x1b[0m `) || undefined;
|
|
195
|
+
}
|
|
196
|
+
// 4. Model
|
|
197
|
+
let model;
|
|
198
|
+
if (preset.models.length > 0) {
|
|
199
|
+
const customOption = "(type a custom model name)";
|
|
200
|
+
const choice = await pick(rl, "Which model?", [...preset.models, customOption], true);
|
|
201
|
+
if (choice === customOption) {
|
|
202
|
+
model = await ask(rl, `\n\x1b[36mModel name:\x1b[0m `);
|
|
203
|
+
}
|
|
204
|
+
else {
|
|
205
|
+
model = choice;
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
else {
|
|
209
|
+
model = await ask(rl, `\n\x1b[36mModel name:\x1b[0m `);
|
|
210
|
+
}
|
|
211
|
+
// 5. Telegram (optional)
|
|
212
|
+
const setupTelegram = await ask(rl, `\n\x1b[36mSet up Telegram bot? (y/N)\x1b[0m `);
|
|
213
|
+
let telegramConfig;
|
|
214
|
+
if (setupTelegram.trim().toLowerCase() === "y") {
|
|
215
|
+
const botToken = await ask(rl, `\x1b[36mBot token (from @BotFather):\x1b[0m `);
|
|
216
|
+
const userId = await ask(rl, `\x1b[36mYour Telegram user ID (from @userinfobot):\x1b[0m `);
|
|
217
|
+
if (botToken.trim()) {
|
|
218
|
+
telegramConfig = {
|
|
219
|
+
botToken: botToken.trim(),
|
|
220
|
+
allowedUsers: userId.trim() ? [parseInt(userId.trim(), 10)] : [],
|
|
221
|
+
};
|
|
222
|
+
}
|
|
223
|
+
}
|
|
224
|
+
rl.close();
|
|
225
|
+
// Save
|
|
226
|
+
const values = {
|
|
227
|
+
provider: preset.provider,
|
|
228
|
+
model: model.trim(),
|
|
229
|
+
};
|
|
230
|
+
if (apiKey)
|
|
231
|
+
values.apiKey = apiKey.trim();
|
|
232
|
+
if (baseUrl)
|
|
233
|
+
values.baseUrl = baseUrl.trim();
|
|
234
|
+
if (!baseUrl && serviceKey !== "custom") {
|
|
235
|
+
// Clear baseUrl if switching away from a custom endpoint
|
|
236
|
+
values.baseUrl = undefined;
|
|
237
|
+
}
|
|
238
|
+
if (telegramConfig) {
|
|
239
|
+
values.telegram = telegramConfig;
|
|
240
|
+
}
|
|
241
|
+
saveConfig(values);
|
|
242
|
+
seedWorkspace();
|
|
243
|
+
console.log(`\n\x1b[32m Saved to ${paths.configFile}\x1b[0m`);
|
|
244
|
+
console.log(`\x1b[90m provider: ${values.provider}`);
|
|
245
|
+
console.log(` model: ${values.model}`);
|
|
246
|
+
if (baseUrl)
|
|
247
|
+
console.log(` baseUrl: ${baseUrl}`);
|
|
248
|
+
console.log(` apiKey: ${apiKey ? "***" + apiKey.slice(-4) : "not set"}`);
|
|
249
|
+
if (telegramConfig)
|
|
250
|
+
console.log(` telegram: bot token configured`);
|
|
251
|
+
console.log(`\x1b[0m`);
|
|
252
|
+
console.log(`\n Run \x1b[36mcamelagi chat\x1b[0m to start chatting.\n`);
|
|
253
|
+
}
|
|
254
|
+
//# sourceMappingURL=setup.js.map
|