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 @@
1
+ {"version":3,"file":"models.js","sourceRoot":"","sources":["../../src/core/models.ts"],"names":[],"mappings":"AAAA,8DAA8D;AAC9D,EAAE;AACF,+CAA+C;AAQ/C,MAAM,CAAC,MAAM,gBAAgB,GAAmC;IAC9D,SAAS,EAAE;QACT,QAAQ,EAAE,WAAW;QACrB,MAAM,EAAE;YACN,0BAA0B;YAC1B,wBAAwB;YACxB,yBAAyB;SAC1B;KACF;IACD,MAAM,EAAE;QACN,QAAQ,EAAE,QAAQ;QAClB,MAAM,EAAE;YACN,QAAQ;YACR,aAAa;YACb,SAAS;YACT,cAAc;YACd,cAAc;YACd,IAAI;YACJ,SAAS;SACV;KACF;IACD,UAAU,EAAE;QACV,QAAQ,EAAE,QAAQ;QAClB,OAAO,EAAE,8BAA8B;QACvC,MAAM,EAAE;YACN,YAAY;YACZ,oCAAoC;YACpC,kCAAkC;YAClC,mCAAmC;YACnC,SAAS;YACT,eAAe;YACf,oBAAoB;YACpB,gBAAgB;YAChB,qBAAqB;YACrB,qBAAqB;YACrB,WAAW;YACX,gBAAgB;YAChB,SAAS;YACT,uBAAuB;YACvB,yBAAyB;YACzB,yBAAyB;YACzB,WAAW;YACX,sBAAsB;YACtB,gCAAgC;YAChC,2BAA2B;YAC3B,OAAO;YACP,6BAA6B;YAC7B,0BAA0B;YAC1B,mCAAmC;YACnC,OAAO;YACP,sBAAsB;YACtB,gBAAgB;YAChB,oBAAoB;YACpB,UAAU;YACV,8BAA8B;YAC9B,4BAA4B;YAC5B,0CAA0C;YAC1C,0BAA0B;YAC1B,MAAM;YACN,aAAa;YACb,kBAAkB;YAClB,SAAS;YACT,kBAAkB;YAClB,+BAA+B;YAC/B,SAAS;YACT,oBAAoB;YACpB,qBAAqB;SACtB;KACF;IACD,MAAM,EAAE;QACN,QAAQ,EAAE,QAAQ;QAClB,OAAO,EAAE,2BAA2B;QACpC,MAAM,EAAE;YACN,UAAU;YACV,OAAO;YACP,aAAa;YACb,QAAQ;YACR,SAAS;YACT,WAAW;YACX,MAAM;SACP;KACF;IACD,MAAM,EAAE;QACN,QAAQ,EAAE,QAAQ;QAClB,MAAM,EAAE,EAAE;KACX;CACF,CAAC;AAEF,gEAAgE;AAChE,MAAM,UAAU,aAAa,CAAC,QAAgB,EAAE,OAAgB;IAC9D,IAAI,OAAO,EAAE,QAAQ,CAAC,YAAY,CAAC;QAAE,OAAO,gBAAgB,CAAC,UAAU,CAAC;IACxE,IAAI,OAAO,EAAE,QAAQ,CAAC,WAAW,CAAC,IAAI,OAAO,EAAE,QAAQ,CAAC,WAAW,CAAC;QAAE,OAAO,gBAAgB,CAAC,MAAM,CAAC;IACrG,OAAO,gBAAgB,CAAC,QAAQ,CAAC,IAAI,gBAAgB,CAAC,MAAM,CAAC;AAC/D,CAAC;AASD;;;GAGG;AACH,MAAM,CAAC,KAAK,UAAU,qBAAqB,CAAC,MAAe;IACzD,IAAI,CAAC;QACH,MAAM,OAAO,GAA2B,EAAE,cAAc,EAAE,kBAAkB,EAAE,CAAC;QAC/E,IAAI,MAAM;YAAE,OAAO,CAAC,eAAe,CAAC,GAAG,UAAU,MAAM,EAAE,CAAC;QAE1D,MAAM,GAAG,GAAG,MAAM,KAAK,CAAC,qCAAqC,EAAE,EAAE,OAAO,EAAE,MAAM,EAAE,WAAW,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAC/G,IAAI,CAAC,GAAG,CAAC,EAAE;YAAE,OAAO,EAAE,CAAC;QAEvB,MAAM,IAAI,GAAG,MAAM,GAAG,CAAC,IAAI,EAAkC,CAAC;QAC9D,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC;YAAE,OAAO,EAAE,CAAC;QAEzC,uEAAuE;QACvE,OAAO,IAAI,CAAC,IAAI;aACb,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;aAC9C,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,aAAa,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;IAC9C,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,CAAC;IACZ,CAAC;AACH,CAAC"}
@@ -0,0 +1,3 @@
1
+ // Shared types
2
+ export {};
3
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/core/types.ts"],"names":[],"mappings":"AAAA,eAAe"}
@@ -0,0 +1,51 @@
1
+ // Check for newer version on npm registry (non-blocking, fails silently)
2
+ import { createRequire } from "node:module";
3
+ const require = createRequire(import.meta.url);
4
+ /** Compare semver strings. Returns 1 if a > b, -1 if a < b, 0 if equal. */
5
+ function compareSemver(a, b) {
6
+ const pa = a.split(".").map(Number);
7
+ const pb = b.split(".").map(Number);
8
+ for (let i = 0; i < 3; i++) {
9
+ if ((pa[i] ?? 0) > (pb[i] ?? 0))
10
+ return 1;
11
+ if ((pa[i] ?? 0) < (pb[i] ?? 0))
12
+ return -1;
13
+ }
14
+ return 0;
15
+ }
16
+ /**
17
+ * Check npm registry for a newer version.
18
+ * Returns update info or null. Never throws.
19
+ */
20
+ export async function checkForUpdate() {
21
+ try {
22
+ const pkg = require("../../package.json");
23
+ const current = pkg.version;
24
+ const res = await fetch(`https://registry.npmjs.org/${pkg.name}/latest`, {
25
+ signal: AbortSignal.timeout(3000),
26
+ });
27
+ if (!res.ok)
28
+ return null;
29
+ const data = await res.json();
30
+ const latest = data.version;
31
+ if (!latest)
32
+ return null;
33
+ if (compareSemver(latest, current) > 0) {
34
+ return { current, latest };
35
+ }
36
+ return null;
37
+ }
38
+ catch {
39
+ return null;
40
+ }
41
+ }
42
+ /** Print update notice if available (non-blocking) */
43
+ export function printUpdateNotice() {
44
+ checkForUpdate().then((update) => {
45
+ if (update) {
46
+ console.log(`\n\x1b[33m Update available: ${update.current} → ${update.latest}\x1b[0m`);
47
+ console.log(`\x1b[90m Run: npm update -g camelagi\x1b[0m\n`);
48
+ }
49
+ }).catch(() => { });
50
+ }
51
+ //# sourceMappingURL=update-check.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"update-check.js","sourceRoot":"","sources":["../../src/core/update-check.ts"],"names":[],"mappings":"AAAA,yEAAyE;AAEzE,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAE5C,MAAM,OAAO,GAAG,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;AAO/C,2EAA2E;AAC3E,SAAS,aAAa,CAAC,CAAS,EAAE,CAAS;IACzC,MAAM,EAAE,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;IACpC,MAAM,EAAE,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;IACpC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;QAC3B,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;YAAE,OAAO,CAAC,CAAC;QAC1C,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;YAAE,OAAO,CAAC,CAAC,CAAC;IAC7C,CAAC;IACD,OAAO,CAAC,CAAC;AACX,CAAC;AAED;;;GAGG;AACH,MAAM,CAAC,KAAK,UAAU,cAAc;IAClC,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,OAAO,CAAC,oBAAoB,CAAgB,CAAC;QACzD,MAAM,OAAO,GAAG,GAAG,CAAC,OAAO,CAAC;QAE5B,MAAM,GAAG,GAAG,MAAM,KAAK,CAAC,8BAA8B,GAAG,CAAC,IAAI,SAAS,EAAE;YACvE,MAAM,EAAE,WAAW,CAAC,OAAO,CAAC,IAAI,CAAC;SAClC,CAAC,CAAC;QACH,IAAI,CAAC,GAAG,CAAC,EAAE;YAAE,OAAO,IAAI,CAAC;QAEzB,MAAM,IAAI,GAAG,MAAM,GAAG,CAAC,IAAI,EAA0B,CAAC;QACtD,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,CAAC;QAC5B,IAAI,CAAC,MAAM;YAAE,OAAO,IAAI,CAAC;QAEzB,IAAI,aAAa,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC;YACvC,OAAO,EAAE,OAAO,EAAE,MAAM,EAAE,CAAC;QAC7B,CAAC;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAC;IACd,CAAC;AACH,CAAC;AAED,sDAAsD;AACtD,MAAM,UAAU,iBAAiB;IAC/B,cAAc,EAAE,CAAC,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE;QAC/B,IAAI,MAAM,EAAE,CAAC;YACX,OAAO,CAAC,GAAG,CAAC,iCAAiC,MAAM,CAAC,OAAO,MAAM,MAAM,CAAC,MAAM,SAAS,CAAC,CAAC;YACzF,OAAO,CAAC,GAAG,CAAC,gDAAgD,CAAC,CAAC;QAChE,CAAC;IACH,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;AACrB,CAAC"}
package/dist/cron.js ADDED
@@ -0,0 +1,81 @@
1
+ // Cron: scheduled agent runs
2
+ import { runAgent } from "./agent.js";
3
+ import { loadMessages, saveMessage } from "./session.js";
4
+ const activeJobs = new Map();
5
+ function parseInterval(schedule) {
6
+ const match = schedule.match(/^(\d+)(s|m|h|d)$/);
7
+ if (!match)
8
+ return null;
9
+ const value = parseInt(match[1], 10);
10
+ switch (match[2]) {
11
+ case "s": return value * 1000;
12
+ case "m": return value * 60_000;
13
+ case "h": return value * 3_600_000;
14
+ case "d": return value * 86_400_000;
15
+ default: return null;
16
+ }
17
+ }
18
+ function parseCronInterval(schedule) {
19
+ const interval = parseInterval(schedule);
20
+ if (interval)
21
+ return interval;
22
+ const parts = schedule.split(/\s+/);
23
+ if (parts.length !== 5)
24
+ return 60_000;
25
+ const [min] = parts;
26
+ if (min.startsWith("*/")) {
27
+ const every = parseInt(min.slice(2), 10);
28
+ return every * 60_000;
29
+ }
30
+ return 60_000;
31
+ }
32
+ export function startCronJob(job, config, systemPrompt, opts) {
33
+ if (activeJobs.has(job.id)) {
34
+ stopCronJob(job.id);
35
+ }
36
+ const intervalMs = parseCronInterval(job.schedule);
37
+ const sid = job.session ?? `cron-${job.id}`;
38
+ const run = async () => {
39
+ try {
40
+ const history = loadMessages(sid);
41
+ const result = await runAgent(config.apiKey, config.model, systemPrompt, history, job.prompt, {
42
+ maxTurns: opts?.maxTurns ?? 10,
43
+ timeoutMs: opts?.timeoutMs ?? 120_000,
44
+ provider: config.provider,
45
+ baseUrl: config.baseUrl,
46
+ });
47
+ saveMessage(sid, { role: "user", content: job.prompt }, "cron");
48
+ if (result.response) {
49
+ saveMessage(sid, { role: "assistant", content: result.response }, "cron");
50
+ }
51
+ opts?.onRun?.(job.id, result.response);
52
+ }
53
+ catch (err) {
54
+ const error = err instanceof Error ? err : new Error(String(err));
55
+ opts?.onError?.(job.id, error);
56
+ }
57
+ };
58
+ const timer = setInterval(run, intervalMs);
59
+ activeJobs.set(job.id, { job, timer });
60
+ void run();
61
+ }
62
+ export function stopCronJob(id) {
63
+ const active = activeJobs.get(id);
64
+ if (!active)
65
+ return false;
66
+ clearInterval(active.timer);
67
+ activeJobs.delete(id);
68
+ return true;
69
+ }
70
+ export function stopAllCronJobs() {
71
+ for (const [id] of activeJobs) {
72
+ stopCronJob(id);
73
+ }
74
+ }
75
+ export function listCronJobs() {
76
+ return Array.from(activeJobs.values()).map((a) => a.job);
77
+ }
78
+ export function isCronRunning(id) {
79
+ return activeJobs.has(id);
80
+ }
81
+ //# sourceMappingURL=cron.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cron.js","sourceRoot":"","sources":["../src/cron.ts"],"names":[],"mappings":"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"}
package/dist/daemon.js ADDED
@@ -0,0 +1,109 @@
1
+ // Daemon management — launchd (macOS) plist generation + launchctl
2
+ import fs from "node:fs";
3
+ import path from "node:path";
4
+ import os from "node:os";
5
+ import { execSync } from "node:child_process";
6
+ const LABEL = "com.camelagi.server";
7
+ const PLIST_PATH = path.join(os.homedir(), "Library", "LaunchAgents", `${LABEL}.plist`);
8
+ function getNodePath() {
9
+ try {
10
+ return execSync("which node", { encoding: "utf-8" }).trim();
11
+ }
12
+ catch {
13
+ return "/usr/local/bin/node";
14
+ }
15
+ }
16
+ function getEntryPath() {
17
+ return path.resolve(import.meta.dirname, "..", "camelagi.mjs");
18
+ }
19
+ function generatePlist() {
20
+ const nodePath = getNodePath();
21
+ const entryPath = getEntryPath();
22
+ const logDir = path.join(os.homedir(), ".camelagi", "logs");
23
+ return `<?xml version="1.0" encoding="UTF-8"?>
24
+ <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
25
+ <plist version="1.0">
26
+ <dict>
27
+ <key>Label</key>
28
+ <string>${LABEL}</string>
29
+ <key>ProgramArguments</key>
30
+ <array>
31
+ <string>${nodePath}</string>
32
+ <string>${entryPath}</string>
33
+ <string>serve</string>
34
+ </array>
35
+ <key>RunAtLoad</key>
36
+ <true/>
37
+ <key>KeepAlive</key>
38
+ <true/>
39
+ <key>StandardOutPath</key>
40
+ <string>${logDir}/daemon.stdout.log</string>
41
+ <key>StandardErrorPath</key>
42
+ <string>${logDir}/daemon.stderr.log</string>
43
+ <key>EnvironmentVariables</key>
44
+ <dict>
45
+ <key>PATH</key>
46
+ <string>${process.env.PATH}</string>
47
+ </dict>
48
+ </dict>
49
+ </plist>`;
50
+ }
51
+ export function install() {
52
+ fs.mkdirSync(path.dirname(PLIST_PATH), { recursive: true });
53
+ fs.mkdirSync(path.join(os.homedir(), ".camelagi", "logs"), { recursive: true });
54
+ fs.writeFileSync(PLIST_PATH, generatePlist());
55
+ try {
56
+ execSync(`launchctl load -w "${PLIST_PATH}"`, { stdio: "pipe" });
57
+ }
58
+ catch {
59
+ try {
60
+ execSync(`launchctl unload "${PLIST_PATH}"`, { stdio: "pipe" });
61
+ }
62
+ catch { /* ignore */ }
63
+ execSync(`launchctl load -w "${PLIST_PATH}"`, { stdio: "pipe" });
64
+ }
65
+ console.log(`\x1b[32m✓\x1b[0m Daemon installed and started`);
66
+ console.log(` Plist: ${PLIST_PATH}`);
67
+ console.log(` Logs: ~/.camelagi/logs/daemon.{stdout,stderr}.log`);
68
+ console.log(` Run \x1b[36mcamelagi daemon status\x1b[0m to check`);
69
+ }
70
+ export function uninstall() {
71
+ if (!fs.existsSync(PLIST_PATH)) {
72
+ console.log("Daemon is not installed.");
73
+ return;
74
+ }
75
+ try {
76
+ execSync(`launchctl unload "${PLIST_PATH}"`, { stdio: "pipe" });
77
+ }
78
+ catch { /* not loaded */ }
79
+ fs.unlinkSync(PLIST_PATH);
80
+ console.log(`\x1b[32m✓\x1b[0m Daemon uninstalled`);
81
+ }
82
+ export function status() {
83
+ if (!fs.existsSync(PLIST_PATH)) {
84
+ console.log("Daemon: \x1b[90mnot installed\x1b[0m");
85
+ return;
86
+ }
87
+ try {
88
+ const output = execSync(`launchctl list 2>/dev/null | grep ${LABEL}`, { encoding: "utf-8" }).trim();
89
+ if (output) {
90
+ const parts = output.split(/\s+/);
91
+ const pid = parts[0];
92
+ const exitCode = parts[1];
93
+ if (pid !== "-") {
94
+ console.log(`Daemon: \x1b[32mrunning\x1b[0m (PID ${pid})`);
95
+ }
96
+ else {
97
+ console.log(`Daemon: \x1b[31mstopped\x1b[0m (last exit code: ${exitCode})`);
98
+ }
99
+ }
100
+ else {
101
+ console.log("Daemon: \x1b[90mnot loaded\x1b[0m");
102
+ }
103
+ }
104
+ catch {
105
+ console.log("Daemon: \x1b[90mnot loaded\x1b[0m");
106
+ }
107
+ console.log(` Plist: ${PLIST_PATH}`);
108
+ }
109
+ //# sourceMappingURL=daemon.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"daemon.js","sourceRoot":"","sources":["../src/daemon.ts"],"names":[],"mappings":"AAAA,mEAAmE;AAEnE,OAAO,EAAE,MAAM,SAAS,CAAC;AACzB,OAAO,IAAI,MAAM,WAAW,CAAC;AAC7B,OAAO,EAAE,MAAM,SAAS,CAAC;AACzB,OAAO,EAAE,QAAQ,EAAE,MAAM,oBAAoB,CAAC;AAE9C,MAAM,KAAK,GAAG,qBAAqB,CAAC;AACpC,MAAM,UAAU,GAAG,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,EAAE,SAAS,EAAE,cAAc,EAAE,GAAG,KAAK,QAAQ,CAAC,CAAC;AAExF,SAAS,WAAW;IAClB,IAAI,CAAC;QACH,OAAO,QAAQ,CAAC,YAAY,EAAE,EAAE,QAAQ,EAAE,OAAO,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;IAC9D,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,qBAAqB,CAAC;IAC/B,CAAC;AACH,CAAC;AAED,SAAS,YAAY;IACnB,OAAO,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,EAAE,cAAc,CAAC,CAAC;AACjE,CAAC;AAED,SAAS,aAAa;IACpB,MAAM,QAAQ,GAAG,WAAW,EAAE,CAAC;IAC/B,MAAM,SAAS,GAAG,YAAY,EAAE,CAAC;IACjC,MAAM,MAAM,GAAG,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,EAAE,WAAW,EAAE,MAAM,CAAC,CAAC;IAE5D,OAAO;;;;;YAKG,KAAK;;;cAGH,QAAQ;cACR,SAAS;;;;;;;;YAQX,MAAM;;YAEN,MAAM;;;;cAIJ,OAAO,CAAC,GAAG,CAAC,IAAI;;;SAGrB,CAAC;AACV,CAAC;AAED,MAAM,UAAU,OAAO;IACrB,EAAE,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IAC5D,EAAE,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,EAAE,WAAW,EAAE,MAAM,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IAEhF,EAAE,CAAC,aAAa,CAAC,UAAU,EAAE,aAAa,EAAE,CAAC,CAAC;IAE9C,IAAI,CAAC;QACH,QAAQ,CAAC,sBAAsB,UAAU,GAAG,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC;IACnE,CAAC;IAAC,MAAM,CAAC;QACP,IAAI,CAAC;YAAC,QAAQ,CAAC,qBAAqB,UAAU,GAAG,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,YAAY,CAAC,CAAC;QAC/F,QAAQ,CAAC,sBAAsB,UAAU,GAAG,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC;IACnE,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,+CAA+C,CAAC,CAAC;IAC7D,OAAO,CAAC,GAAG,CAAC,YAAY,UAAU,EAAE,CAAC,CAAC;IACtC,OAAO,CAAC,GAAG,CAAC,sDAAsD,CAAC,CAAC;IACpE,OAAO,CAAC,GAAG,CAAC,sDAAsD,CAAC,CAAC;AACtE,CAAC;AAED,MAAM,UAAU,SAAS;IACvB,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE,CAAC;QAC/B,OAAO,CAAC,GAAG,CAAC,0BAA0B,CAAC,CAAC;QACxC,OAAO;IACT,CAAC;IAED,IAAI,CAAC;QAAC,QAAQ,CAAC,qBAAqB,UAAU,GAAG,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC;IAAC,CAAC;IAAC,MAAM,CAAC,CAAC,gBAAgB,CAAC,CAAC;IACnG,EAAE,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC;IAC1B,OAAO,CAAC,GAAG,CAAC,qCAAqC,CAAC,CAAC;AACrD,CAAC;AAED,MAAM,UAAU,MAAM;IACpB,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE,CAAC;QAC/B,OAAO,CAAC,GAAG,CAAC,sCAAsC,CAAC,CAAC;QACpD,OAAO;IACT,CAAC;IAED,IAAI,CAAC;QACH,MAAM,MAAM,GAAG,QAAQ,CAAC,qCAAqC,KAAK,EAAE,EAAE,EAAE,QAAQ,EAAE,OAAO,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;QACpG,IAAI,MAAM,EAAE,CAAC;YACX,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;YAClC,MAAM,GAAG,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC;YACrB,MAAM,QAAQ,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC;YAC1B,IAAI,GAAG,KAAK,GAAG,EAAE,CAAC;gBAChB,OAAO,CAAC,GAAG,CAAC,uCAAuC,GAAG,GAAG,CAAC,CAAC;YAC7D,CAAC;iBAAM,CAAC;gBACN,OAAO,CAAC,GAAG,CAAC,mDAAmD,QAAQ,GAAG,CAAC,CAAC;YAC9E,CAAC;QACH,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,GAAG,CAAC,mCAAmC,CAAC,CAAC;QACnD,CAAC;IACH,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,CAAC,GAAG,CAAC,mCAAmC,CAAC,CAAC;IACnD,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,YAAY,UAAU,EAAE,CAAC,CAAC;AACxC,CAAC"}
package/dist/doctor.js ADDED
@@ -0,0 +1,194 @@
1
+ // Doctor: health checks and diagnostics
2
+ import fs from "node:fs";
3
+ import path from "node:path";
4
+ import { paths, loadConfig } from "./core/config.js";
5
+ import { listSessions } from "./session.js";
6
+ import { workspacePaths } from "./workspace.js";
7
+ export async function runDoctor() {
8
+ const checks = [];
9
+ // 1. Config file
10
+ if (fs.existsSync(paths.configFile)) {
11
+ checks.push({ name: "Config file", status: "ok", message: paths.configFile });
12
+ }
13
+ else {
14
+ checks.push({ name: "Config file", status: "warn", message: "Not found. Run: camelagi setup" });
15
+ }
16
+ // 2. Config valid
17
+ try {
18
+ const config = loadConfig();
19
+ checks.push({ name: "Config valid", status: "ok", message: `provider=${config.provider} model=${config.model}` });
20
+ // 3. API key
21
+ if (config.apiKey) {
22
+ const masked = "***" + config.apiKey.slice(-4);
23
+ checks.push({ name: "API key", status: "ok", message: masked });
24
+ }
25
+ else {
26
+ checks.push({ name: "API key", status: "error", message: "No API key configured" });
27
+ }
28
+ // 4. Base URL
29
+ if (config.baseUrl) {
30
+ checks.push({ name: "Base URL", status: "ok", message: config.baseUrl });
31
+ }
32
+ // 5. Model connectivity
33
+ if (config.apiKey) {
34
+ try {
35
+ const { createClient, chatDirect } = await import("./model.js");
36
+ const client = createClient(config);
37
+ const result = await chatDirect(client, config.model, "You are a test.", "Say OK");
38
+ const text = result.content || "received response";
39
+ checks.push({ name: "Model connectivity", status: "ok", message: `${config.model}: ${text.slice(0, 50)}` });
40
+ }
41
+ catch (err) {
42
+ const msg = err instanceof Error ? err.message : String(err);
43
+ checks.push({ name: "Model connectivity", status: "error", message: msg.slice(0, 100) });
44
+ }
45
+ }
46
+ else {
47
+ checks.push({ name: "Model connectivity", status: "error", message: "Skipped (no API key)" });
48
+ }
49
+ // 6. Telegram bots
50
+ const botTokens = [];
51
+ if (config.telegram.botToken) {
52
+ botTokens.push({ label: "Telegram", token: config.telegram.botToken });
53
+ }
54
+ for (const [id, agent] of Object.entries(config.agents)) {
55
+ if (agent.telegram?.botToken) {
56
+ botTokens.push({ label: `Agent "${id}"`, token: agent.telegram.botToken });
57
+ }
58
+ }
59
+ for (const { label, token } of botTokens) {
60
+ try {
61
+ const resp = await fetch(`https://api.telegram.org/bot${token}/getMe`);
62
+ const data = await resp.json();
63
+ if (data.ok && data.result) {
64
+ const name = data.result.first_name ?? "Bot";
65
+ const username = data.result.username ? `@${data.result.username}` : "";
66
+ checks.push({ name: label, status: "ok", message: `${username} (${name})` });
67
+ }
68
+ else {
69
+ checks.push({ name: label, status: "error", message: `Bot token invalid: ${data.description ?? "Unknown error"}` });
70
+ }
71
+ }
72
+ catch (err) {
73
+ const msg = err instanceof Error ? err.message : String(err);
74
+ checks.push({ name: label, status: "error", message: `Bot token check failed: ${msg}` });
75
+ }
76
+ }
77
+ // 7. Thinking
78
+ if (config.thinking !== "off") {
79
+ checks.push({ name: "Thinking", status: "ok", message: config.thinking });
80
+ }
81
+ }
82
+ catch (err) {
83
+ checks.push({ name: "Config valid", status: "error", message: err instanceof Error ? err.message : String(err) });
84
+ }
85
+ // 8. Workspace
86
+ const { workspaceDir } = workspacePaths;
87
+ if (fs.existsSync(workspaceDir)) {
88
+ const files = fs.readdirSync(workspaceDir);
89
+ checks.push({ name: "Workspace", status: "ok", message: `${workspaceDir} (${files.length} files)` });
90
+ }
91
+ else {
92
+ checks.push({ name: "Workspace", status: "warn", message: "Not found. Run: camelagi setup" });
93
+ }
94
+ // 9. Bootstrap files
95
+ const bootstrapFiles = ["AGENTS.md", "SOUL.md", "USER.md", "TOOLS.md", "IDENTITY.md"];
96
+ const missing = bootstrapFiles.filter((f) => !fs.existsSync(path.join(workspaceDir, f)));
97
+ if (missing.length === 0) {
98
+ checks.push({ name: "Bootstrap files", status: "ok", message: "All present" });
99
+ }
100
+ else {
101
+ checks.push({ name: "Bootstrap files", status: "warn", message: `Missing: ${missing.join(", ")}` });
102
+ }
103
+ // 10. Memory directory
104
+ const memoryDir = path.join(workspaceDir, "memory");
105
+ if (fs.existsSync(memoryDir)) {
106
+ const dailyFiles = fs.readdirSync(memoryDir).filter((f) => f.endsWith(".md"));
107
+ checks.push({ name: "Memory", status: "ok", message: `${dailyFiles.length} daily file(s)` });
108
+ }
109
+ else {
110
+ checks.push({ name: "Memory", status: "warn", message: "memory/ directory not found" });
111
+ }
112
+ // 11. Sessions
113
+ const sessions = listSessions();
114
+ checks.push({ name: "Sessions", status: "ok", message: `${sessions.length} session(s)` });
115
+ // 11b. Usage tracking
116
+ const usageDir = path.join(paths.configDir, "usage");
117
+ if (fs.existsSync(usageDir)) {
118
+ const usageFiles = fs.readdirSync(usageDir).filter((f) => f.endsWith(".json"));
119
+ checks.push({ name: "Token usage", status: "ok", message: `${usageFiles.length} tracked session(s)` });
120
+ }
121
+ // 12. Hooks directory
122
+ const hooksDir = path.join(paths.configDir, "hooks");
123
+ if (fs.existsSync(hooksDir)) {
124
+ const hooks = fs.readdirSync(hooksDir).filter((f) => f.endsWith(".sh") || f.endsWith(".js"));
125
+ checks.push({ name: "Hooks", status: "ok", message: `${hooks.length} hook(s)` });
126
+ }
127
+ // 13. Skills directory
128
+ const skillsDir = path.join(paths.configDir, "skills");
129
+ if (fs.existsSync(skillsDir)) {
130
+ const skills = fs.readdirSync(skillsDir, { withFileTypes: true }).filter((d) => d.isDirectory());
131
+ checks.push({ name: "Skills", status: "ok", message: `${skills.length} skill(s)` });
132
+ }
133
+ // Security checks
134
+ try {
135
+ const config = loadConfig();
136
+ // Config file permissions (should be owner-only)
137
+ if (fs.existsSync(paths.configFile)) {
138
+ const stat = fs.statSync(paths.configFile);
139
+ const mode = stat.mode & 0o777;
140
+ if (mode & 0o077) {
141
+ checks.push({ name: "Config permissions", status: "warn", message: `${paths.configFile} is readable by others (${mode.toString(8)}). Run: chmod 600 ~/.camelagi/config.yaml` });
142
+ }
143
+ else {
144
+ checks.push({ name: "Config permissions", status: "ok", message: `0${mode.toString(8)}` });
145
+ }
146
+ }
147
+ // Auth token
148
+ if (config.serve.token) {
149
+ if (config.serve.token.length < 24) {
150
+ checks.push({ name: "Auth token", status: "warn", message: `Token is short (${config.serve.token.length} chars). 24+ recommended.` });
151
+ }
152
+ else {
153
+ checks.push({ name: "Auth token", status: "ok", message: `${config.serve.token.length} chars` });
154
+ }
155
+ }
156
+ else {
157
+ checks.push({ name: "Auth token", status: "warn", message: "No auth token set. Anyone on localhost can access the API." });
158
+ }
159
+ // Bind address
160
+ if (config.serve.host !== "127.0.0.1" && config.serve.host !== "::1" && config.serve.host !== "localhost") {
161
+ checks.push({ name: "Bind address", status: "warn", message: `Binding to ${config.serve.host} — server is exposed to the network. Use a reverse proxy + TLS.` });
162
+ }
163
+ else {
164
+ checks.push({ name: "Bind address", status: "ok", message: `${config.serve.host} (localhost only)` });
165
+ }
166
+ // State directory permissions
167
+ const statDirStat = fs.statSync(paths.configDir);
168
+ const dirMode = statDirStat.mode & 0o777;
169
+ if (dirMode & 0o077) {
170
+ checks.push({ name: "State directory", status: "warn", message: `~/.camelagi/ is accessible by others (${dirMode.toString(8)}). Run: chmod 700 ~/.camelagi` });
171
+ }
172
+ else {
173
+ checks.push({ name: "State directory", status: "ok", message: `0${dirMode.toString(8)}` });
174
+ }
175
+ }
176
+ catch { /* config already checked above */ }
177
+ // 14. Node.js version
178
+ const nodeVersion = process.version;
179
+ const major = parseInt(nodeVersion.slice(1), 10);
180
+ if (major >= 20) {
181
+ checks.push({ name: "Node.js", status: "ok", message: nodeVersion });
182
+ }
183
+ else {
184
+ checks.push({ name: "Node.js", status: "warn", message: `${nodeVersion} (20+ recommended)` });
185
+ }
186
+ return checks;
187
+ }
188
+ export function formatChecks(checks) {
189
+ const icons = { ok: "\x1b[32m✓\x1b[0m", warn: "\x1b[33m!\x1b[0m", error: "\x1b[31m✗\x1b[0m" };
190
+ return checks
191
+ .map((c) => ` ${icons[c.status]} ${c.name}: ${c.message}`)
192
+ .join("\n");
193
+ }
194
+ //# sourceMappingURL=doctor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"doctor.js","sourceRoot":"","sources":["../src/doctor.ts"],"names":[],"mappings":"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"}
package/dist/errors.js ADDED
@@ -0,0 +1,5 @@
1
+ // Error utilities
2
+ export function errorMessage(err) {
3
+ return err instanceof Error ? err.message : String(err);
4
+ }
5
+ //# sourceMappingURL=errors.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"errors.js","sourceRoot":"","sources":["../src/errors.ts"],"names":[],"mappings":"AAAA,kBAAkB;AAElB,MAAM,UAAU,YAAY,CAAC,GAAY;IACvC,OAAO,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;AAC1D,CAAC"}
@@ -0,0 +1,42 @@
1
+ // Approval forwarding: send approval requests to Telegram when running headless
2
+ //
3
+ // When a tool call needs approval but there's no interactive channel (HTTP API,
4
+ // cron, boot), forward the request to a configured Telegram chat with inline buttons.
5
+ //
6
+ // Config:
7
+ // approvals:
8
+ // forwardTo: 123456789 # your Telegram user/chat ID
9
+ import { InlineKeyboard } from "grammy";
10
+ // The first active bot is used for forwarding
11
+ let forwardBotRef = null;
12
+ /** Register a bot for forwarding approval requests (called by telegram.ts) */
13
+ export function registerForwardBot(bot) {
14
+ forwardBotRef = bot;
15
+ }
16
+ /** Unregister on shutdown */
17
+ export function unregisterForwardBot() {
18
+ forwardBotRef = null;
19
+ }
20
+ /**
21
+ * Send an approval request to a Telegram chat with inline buttons.
22
+ * Returns true if sent successfully, false if no bot available.
23
+ */
24
+ export async function forwardApproval(approvalId, toolName, preview, forwardTo) {
25
+ if (!forwardBotRef || !forwardTo)
26
+ return false;
27
+ try {
28
+ const keyboard = new InlineKeyboard()
29
+ .text("✅ Allow", `approve:${approvalId}:allow-once`)
30
+ .text("♾️ Always", `approve:${approvalId}:allow-always`)
31
+ .text("❌ Deny", `approve:${approvalId}:deny`);
32
+ await forwardBotRef.api.sendMessage(forwardTo, `🔒 ${toolName}\n${preview}`, {
33
+ reply_markup: keyboard,
34
+ });
35
+ return true;
36
+ }
37
+ catch (err) {
38
+ console.error(` [approval-forward] failed: ${err instanceof Error ? err.message : err}`);
39
+ return false;
40
+ }
41
+ }
42
+ //# sourceMappingURL=approval-forward.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"approval-forward.js","sourceRoot":"","sources":["../../src/extensions/approval-forward.ts"],"names":[],"mappings":"AAAA,gFAAgF;AAChF,EAAE;AACF,gFAAgF;AAChF,sFAAsF;AACtF,EAAE;AACF,UAAU;AACV,eAAe;AACf,0DAA0D;AAG1D,OAAO,EAAE,cAAc,EAAE,MAAM,QAAQ,CAAC;AAExC,8CAA8C;AAC9C,IAAI,aAAa,GAAe,IAAI,CAAC;AAErC,8EAA8E;AAC9E,MAAM,UAAU,kBAAkB,CAAC,GAAQ;IACzC,aAAa,GAAG,GAAG,CAAC;AACtB,CAAC;AAED,6BAA6B;AAC7B,MAAM,UAAU,oBAAoB;IAClC,aAAa,GAAG,IAAI,CAAC;AACvB,CAAC;AAED;;;GAGG;AACH,MAAM,CAAC,KAAK,UAAU,eAAe,CACnC,UAAkB,EAClB,QAAgB,EAChB,OAAe,EACf,SAAiB;IAEjB,IAAI,CAAC,aAAa,IAAI,CAAC,SAAS;QAAE,OAAO,KAAK,CAAC;IAE/C,IAAI,CAAC;QACH,MAAM,QAAQ,GAAG,IAAI,cAAc,EAAE;aAClC,IAAI,CAAC,SAAS,EAAE,WAAW,UAAU,aAAa,CAAC;aACnD,IAAI,CAAC,WAAW,EAAE,WAAW,UAAU,eAAe,CAAC;aACvD,IAAI,CAAC,QAAQ,EAAE,WAAW,UAAU,OAAO,CAAC,CAAC;QAEhD,MAAM,aAAa,CAAC,GAAG,CAAC,WAAW,CAAC,SAAS,EAAE,MAAM,QAAQ,KAAK,OAAO,EAAE,EAAE;YAC3E,YAAY,EAAE,QAAQ;SACvB,CAAC,CAAC;QACH,OAAO,IAAI,CAAC;IACd,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,OAAO,CAAC,KAAK,CAAC,gCAAgC,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,EAAE,CAAC,CAAC;QAC1F,OAAO,KAAK,CAAC;IACf,CAAC;AACH,CAAC"}