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,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":"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"}
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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"}
@@ -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