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,71 @@
1
+ // Channel registry: register, start, stop, reconcile all channels
2
+ const channels = new Map();
3
+ /** Register a channel implementation. Throws if already registered. */
4
+ export function registerChannel(channel) {
5
+ if (channels.has(channel.type)) {
6
+ throw new Error(`Channel "${channel.type}" is already registered`);
7
+ }
8
+ channels.set(channel.type, channel);
9
+ }
10
+ /** Get a registered channel by type. */
11
+ export function getChannel(type) {
12
+ return channels.get(type);
13
+ }
14
+ /** Get all registered channels. */
15
+ export function getAllChannels() {
16
+ return [...channels.values()];
17
+ }
18
+ /** Check which channel types have config for at least one agent. */
19
+ export function getConfiguredChannelTypes(config) {
20
+ const types = new Set();
21
+ // Legacy top-level telegram
22
+ if (config.telegram.botToken)
23
+ types.add("telegram");
24
+ // Per-agent channels
25
+ for (const agent of Object.values(config.agents)) {
26
+ if (agent.telegram?.botToken)
27
+ types.add("telegram");
28
+ if (agent.discord?.botToken)
29
+ types.add("discord");
30
+ }
31
+ return [...types];
32
+ }
33
+ /**
34
+ * Start all channels that have config.
35
+ * Returns a map of channel type -> started agent IDs.
36
+ */
37
+ export async function startAllChannels(getConfig, getSystemPrompt) {
38
+ const results = new Map();
39
+ for (const channel of channels.values()) {
40
+ try {
41
+ const started = await channel.start(getConfig, getSystemPrompt);
42
+ if (started.length > 0)
43
+ results.set(channel.type, started);
44
+ }
45
+ catch (err) {
46
+ console.error(`[channels] Failed to start ${channel.type}:`, err);
47
+ }
48
+ }
49
+ return results;
50
+ }
51
+ /** Stop all channels. */
52
+ export function stopAllChannels() {
53
+ for (const channel of channels.values()) {
54
+ try {
55
+ channel.stop();
56
+ }
57
+ catch { /* best effort */ }
58
+ }
59
+ }
60
+ /** Reconcile all channels after config change. */
61
+ export async function reconcileAllChannels(getConfig, getSystemPrompt) {
62
+ for (const channel of channels.values()) {
63
+ try {
64
+ await channel.reconcile(getConfig, getSystemPrompt);
65
+ }
66
+ catch (err) {
67
+ console.error(`[channels] Failed to reconcile ${channel.type}:`, err);
68
+ }
69
+ }
70
+ }
71
+ //# sourceMappingURL=registry.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"registry.js","sourceRoot":"","sources":["../../src/channels/registry.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,83 @@
1
+ // TelegramChannel: wraps existing src/telegram.ts as a Channel
2
+ import { seedAgentWorkspace } from "../workspace.js";
3
+ export class TelegramChannel {
4
+ type = "telegram";
5
+ mod = null;
6
+ async load() {
7
+ if (!this.mod)
8
+ this.mod = await import("../telegram.js");
9
+ return this.mod;
10
+ }
11
+ async start(getConfig, getSystemPrompt) {
12
+ const config = getConfig();
13
+ const hasBots = config.telegram.botToken || Object.values(config.agents).some((a) => a.telegram?.botToken);
14
+ if (!hasBots)
15
+ return [];
16
+ const mod = await this.load();
17
+ return mod.startTelegram(getConfig, getSystemPrompt);
18
+ }
19
+ stop() {
20
+ if (this.mod)
21
+ this.mod.stopTelegram();
22
+ }
23
+ async reconcile(getConfig, getSystemPrompt) {
24
+ const mod = await this.load();
25
+ const running = new Set(mod.getActiveBotIds());
26
+ const config = getConfig();
27
+ // Start bots for new agents with telegram config
28
+ const usedTokens = new Set();
29
+ for (const [id, agent] of Object.entries(config.agents)) {
30
+ if (!agent.telegram?.botToken)
31
+ continue;
32
+ usedTokens.add(agent.telegram.botToken);
33
+ if (running.has(id))
34
+ continue;
35
+ try {
36
+ seedAgentWorkspace(id, agent.name);
37
+ await mod.startBot(id, agent.telegram.botToken, getConfig, getSystemPrompt);
38
+ console.log(`[telegram] Hot-started bot: ${id}`);
39
+ }
40
+ catch {
41
+ // Already starting, or other error
42
+ }
43
+ }
44
+ // Legacy top-level telegram bot
45
+ if (config.telegram.botToken && !usedTokens.has(config.telegram.botToken) && !running.has("telegram")) {
46
+ try {
47
+ await mod.startBot("telegram", config.telegram.botToken, getConfig, getSystemPrompt);
48
+ console.log(`[telegram] Hot-started legacy telegram bot`);
49
+ }
50
+ catch { }
51
+ }
52
+ // Stop bots whose agents were removed from config
53
+ for (const id of running) {
54
+ if (id === "telegram") {
55
+ if (!config.telegram.botToken || usedTokens.has(config.telegram.botToken)) {
56
+ mod.stopBot(id);
57
+ console.log(`[telegram] Stopped bot: ${id} (token removed or claimed by agent)`);
58
+ }
59
+ continue;
60
+ }
61
+ if (!config.agents[id]?.telegram?.botToken) {
62
+ mod.stopBot(id);
63
+ console.log(`[telegram] Stopped bot: ${id} (agent removed)`);
64
+ }
65
+ }
66
+ }
67
+ getActiveAgentIds() {
68
+ return this.mod ? this.mod.getActiveBotIds() : [];
69
+ }
70
+ async startAgent(agentId, getConfig, getSystemPrompt) {
71
+ const config = getConfig();
72
+ const agent = config.agents[agentId];
73
+ const token = agentId === "telegram" ? config.telegram.botToken : agent?.telegram?.botToken;
74
+ if (!token)
75
+ throw new Error(`No Telegram bot token for agent "${agentId}"`);
76
+ const mod = await this.load();
77
+ await mod.startBot(agentId, token, getConfig, getSystemPrompt);
78
+ }
79
+ stopAgent(agentId) {
80
+ return this.mod ? this.mod.stopBot(agentId) : false;
81
+ }
82
+ }
83
+ //# sourceMappingURL=telegram.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"telegram.js","sourceRoot":"","sources":["../../src/channels/telegram.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ // Channel: pluggable messaging platform interface
2
+ export {};
3
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/channels/types.ts"],"names":[],"mappings":"AAAA,kDAAkD"}
@@ -0,0 +1,102 @@
1
+ // Block chunker: buffers streamed text into sized blocks for channels (Telegram, etc.)
2
+ export class BlockChunker {
3
+ buffer = "";
4
+ minChars;
5
+ maxChars;
6
+ breakPref;
7
+ onChunk;
8
+ inFence = false;
9
+ constructor(opts) {
10
+ this.minChars = opts.minChars ?? 800;
11
+ this.maxChars = opts.maxChars ?? 3500;
12
+ this.breakPref = opts.breakPreference ?? "paragraph";
13
+ this.onChunk = opts.onChunk;
14
+ }
15
+ append(text) {
16
+ this.buffer += text;
17
+ this.trackFences(text);
18
+ this.drain();
19
+ }
20
+ flush() {
21
+ if (this.buffer.length > 0) {
22
+ if (this.inFence) {
23
+ this.buffer += "\n```";
24
+ this.inFence = false;
25
+ }
26
+ this.onChunk(this.buffer);
27
+ this.buffer = "";
28
+ }
29
+ }
30
+ trackFences(text) {
31
+ const fencePattern = /```/g;
32
+ let match;
33
+ while ((match = fencePattern.exec(text)) !== null) {
34
+ this.inFence = !this.inFence;
35
+ }
36
+ }
37
+ drain() {
38
+ while (this.buffer.length >= this.minChars) {
39
+ if (this.inFence && this.buffer.length < this.maxChars) {
40
+ break; // Don't break inside code fences unless forced
41
+ }
42
+ const breakIdx = this.pickBreakIndex();
43
+ if (breakIdx <= 0)
44
+ break;
45
+ let chunk = this.buffer.slice(0, breakIdx);
46
+ this.buffer = this.buffer.slice(breakIdx);
47
+ // If we broke inside a fence, close and reopen
48
+ const fenceCount = (chunk.match(/```/g) || []).length;
49
+ if (fenceCount % 2 !== 0) {
50
+ chunk += "\n```";
51
+ this.buffer = "```\n" + this.buffer;
52
+ }
53
+ this.onChunk(chunk);
54
+ }
55
+ // Hard break at maxChars
56
+ while (this.buffer.length > this.maxChars) {
57
+ let chunk = this.buffer.slice(0, this.maxChars);
58
+ this.buffer = this.buffer.slice(this.maxChars);
59
+ const fenceCount = (chunk.match(/```/g) || []).length;
60
+ if (fenceCount % 2 !== 0) {
61
+ chunk += "\n```";
62
+ this.buffer = "```\n" + this.buffer;
63
+ }
64
+ this.onChunk(chunk);
65
+ }
66
+ }
67
+ pickBreakIndex() {
68
+ const searchRange = Math.min(this.buffer.length, this.maxChars);
69
+ const text = this.buffer.slice(0, searchRange);
70
+ // Try preferred break type
71
+ if (this.breakPref === "paragraph") {
72
+ const idx = this.findLastBreak(text, /\n\n/g);
73
+ if (idx >= this.minChars)
74
+ return idx + 2;
75
+ }
76
+ // Newline break
77
+ const nlIdx = this.findLastBreak(text, /\n/g);
78
+ if (nlIdx >= this.minChars)
79
+ return nlIdx + 1;
80
+ // Sentence break
81
+ const sentIdx = this.findLastBreak(text, /[.!?]\s/g);
82
+ if (sentIdx >= this.minChars)
83
+ return sentIdx + 2;
84
+ // Word break
85
+ const wordIdx = this.findLastBreak(text, /\s/g);
86
+ if (wordIdx >= this.minChars)
87
+ return wordIdx + 1;
88
+ // Force break at maxChars
89
+ return searchRange >= this.maxChars ? this.maxChars : 0;
90
+ }
91
+ findLastBreak(text, pattern) {
92
+ let lastIdx = -1;
93
+ let match;
94
+ while ((match = pattern.exec(text)) !== null) {
95
+ if (match.index >= this.minChars) {
96
+ lastIdx = match.index;
97
+ }
98
+ }
99
+ return lastIdx;
100
+ }
101
+ }
102
+ //# sourceMappingURL=chunker.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"chunker.js","sourceRoot":"","sources":["../src/chunker.ts"],"names":[],"mappings":"AAAA,uFAAuF;AASvF,MAAM,OAAO,YAAY;IACf,MAAM,GAAG,EAAE,CAAC;IACZ,QAAQ,CAAS;IACjB,QAAQ,CAAS;IACjB,SAAS,CAAuC;IAChD,OAAO,CAAyB;IAChC,OAAO,GAAG,KAAK,CAAC;IAExB,YAAY,IAAiB;QAC3B,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,QAAQ,IAAI,GAAG,CAAC;QACrC,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC;QACtC,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,eAAe,IAAI,WAAW,CAAC;QACrD,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC;IAC9B,CAAC;IAED,MAAM,CAAC,IAAY;QACjB,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC;QACpB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;QACvB,IAAI,CAAC,KAAK,EAAE,CAAC;IACf,CAAC;IAED,KAAK;QACH,IAAI,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC3B,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;gBACjB,IAAI,CAAC,MAAM,IAAI,OAAO,CAAC;gBACvB,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;YACvB,CAAC;YACD,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YAC1B,IAAI,CAAC,MAAM,GAAG,EAAE,CAAC;QACnB,CAAC;IACH,CAAC;IAEO,WAAW,CAAC,IAAY;QAC9B,MAAM,YAAY,GAAG,MAAM,CAAC;QAC5B,IAAI,KAAK,CAAC;QACV,OAAO,CAAC,KAAK,GAAG,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,KAAK,IAAI,EAAE,CAAC;YAClD,IAAI,CAAC,OAAO,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC;QAC/B,CAAC;IACH,CAAC;IAEO,KAAK;QACX,OAAO,IAAI,CAAC,MAAM,CAAC,MAAM,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;YAC3C,IAAI,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,IAAI,CAAC,QAAQ,EAAE,CAAC;gBACvD,MAAM,CAAC,+CAA+C;YACxD,CAAC;YAED,MAAM,QAAQ,GAAG,IAAI,CAAC,cAAc,EAAE,CAAC;YACvC,IAAI,QAAQ,IAAI,CAAC;gBAAE,MAAM;YAEzB,IAAI,KAAK,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAC;YAC3C,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;YAE1C,+CAA+C;YAC/C,MAAM,UAAU,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC;YACtD,IAAI,UAAU,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC;gBACzB,KAAK,IAAI,OAAO,CAAC;gBACjB,IAAI,CAAC,MAAM,GAAG,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC;YACtC,CAAC;YAED,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;QACtB,CAAC;QAED,yBAAyB;QACzB,OAAO,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,IAAI,CAAC,QAAQ,EAAE,CAAC;YAC1C,IAAI,KAAK,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;YAChD,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;YAE/C,MAAM,UAAU,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC;YACtD,IAAI,UAAU,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC;gBACzB,KAAK,IAAI,OAAO,CAAC;gBACjB,IAAI,CAAC,MAAM,GAAG,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC;YACtC,CAAC;YAED,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;QACtB,CAAC;IACH,CAAC;IAEO,cAAc;QACpB,MAAM,WAAW,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;QAChE,MAAM,IAAI,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,EAAE,WAAW,CAAC,CAAC;QAE/C,2BAA2B;QAC3B,IAAI,IAAI,CAAC,SAAS,KAAK,WAAW,EAAE,CAAC;YACnC,MAAM,GAAG,GAAG,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;YAC9C,IAAI,GAAG,IAAI,IAAI,CAAC,QAAQ;gBAAE,OAAO,GAAG,GAAG,CAAC,CAAC;QAC3C,CAAC;QAED,gBAAgB;QAChB,MAAM,KAAK,GAAG,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;QAC9C,IAAI,KAAK,IAAI,IAAI,CAAC,QAAQ;YAAE,OAAO,KAAK,GAAG,CAAC,CAAC;QAE7C,iBAAiB;QACjB,MAAM,OAAO,GAAG,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,UAAU,CAAC,CAAC;QACrD,IAAI,OAAO,IAAI,IAAI,CAAC,QAAQ;YAAE,OAAO,OAAO,GAAG,CAAC,CAAC;QAEjD,aAAa;QACb,MAAM,OAAO,GAAG,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;QAChD,IAAI,OAAO,IAAI,IAAI,CAAC,QAAQ;YAAE,OAAO,OAAO,GAAG,CAAC,CAAC;QAEjD,0BAA0B;QAC1B,OAAO,WAAW,IAAI,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC;IAC1D,CAAC;IAEO,aAAa,CAAC,IAAY,EAAE,OAAe;QACjD,IAAI,OAAO,GAAG,CAAC,CAAC,CAAC;QACjB,IAAI,KAAK,CAAC;QACV,OAAO,CAAC,KAAK,GAAG,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,KAAK,IAAI,EAAE,CAAC;YAC7C,IAAI,KAAK,CAAC,KAAK,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;gBACjC,OAAO,GAAG,KAAK,CAAC,KAAK,CAAC;YACxB,CAAC;QACH,CAAC;QACD,OAAO,OAAO,CAAC;IACjB,CAAC;CACF"}
@@ -0,0 +1,65 @@
1
+ import { register } from "./registry.js";
2
+ import { loadConfig, saveConfig, ensureDirs } from "../core/config.js";
3
+ import { hasFlag } from "./parse.js";
4
+ register({
5
+ name: "agents",
6
+ description: "List configured agents",
7
+ usage: `Usage: camelagi agents [subcommand]
8
+
9
+ List or manage configured agents.
10
+
11
+ Subcommands:
12
+ (none) List all agents (default)
13
+ rm <id> Remove an agent (prompts for confirmation)
14
+
15
+ Options:
16
+ --yes, -y Skip confirmation prompt (with rm)
17
+
18
+ Examples:
19
+ camelagi agents
20
+ camelagi agents rm mybot
21
+ camelagi agents rm mybot --yes`,
22
+ run: async (args) => {
23
+ ensureDirs();
24
+ const config = loadConfig();
25
+ if (args[0] === "rm" && args[1]) {
26
+ const agents = { ...(config.agents ?? {}) };
27
+ if (!agents[args[1]]) {
28
+ console.error(`Agent "${args[1]}" not found.`);
29
+ process.exit(1);
30
+ }
31
+ if (!hasFlag(args, "--yes") && !hasFlag(args, "-y")) {
32
+ const { default: readline } = await import("node:readline");
33
+ const rl = readline.createInterface({ input: process.stdin, output: process.stdout });
34
+ const answer = await new Promise((resolve) => rl.question(` Remove agent "${args[1]}"? (yes/no): `, resolve));
35
+ rl.close();
36
+ if (answer.trim().toLowerCase() !== "yes") {
37
+ console.log(" Cancelled.");
38
+ process.exit(0);
39
+ }
40
+ }
41
+ delete agents[args[1]];
42
+ saveConfig({ agents });
43
+ console.log(`Removed agent: ${args[1]}`);
44
+ process.exit(0);
45
+ }
46
+ if (args[0] && args[0] !== "rm") {
47
+ console.error(`Unknown subcommand: ${args[0]}. Use: camelagi agents [rm <id>]`);
48
+ process.exit(1);
49
+ }
50
+ const agentEntries = Object.entries(config.agents ?? {});
51
+ if (agentEntries.length === 0) {
52
+ console.log("No agents configured. Use /agents add in the TUI or edit config.yaml.");
53
+ }
54
+ else {
55
+ for (const [id, a] of agentEntries) {
56
+ const parts = [a.model ?? config.model];
57
+ if (a.telegram?.botToken)
58
+ parts.push("telegram");
59
+ console.log(` ${id} (${a.name}, ${parts.join(", ")})`);
60
+ }
61
+ }
62
+ process.exit(0);
63
+ },
64
+ });
65
+ //# sourceMappingURL=cmd-agents.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cmd-agents.js","sourceRoot":"","sources":["../../src/cli/cmd-agents.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,10 @@
1
+ import { register } from "./registry.js";
2
+ register({
3
+ name: "bootstrap",
4
+ description: "First-time setup via Telegram admin bot",
5
+ run: async (args) => {
6
+ const { runBootstrap } = await import("../bootstrap.js");
7
+ await runBootstrap(args[0]);
8
+ },
9
+ });
10
+ //# sourceMappingURL=cmd-bootstrap.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cmd-bootstrap.js","sourceRoot":"","sources":["../../src/cli/cmd-bootstrap.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AAEzC,QAAQ,CAAC;IACP,IAAI,EAAE,WAAW;IACjB,WAAW,EAAE,yCAAyC;IACtD,GAAG,EAAE,KAAK,EAAE,IAAI,EAAE,EAAE;QAClB,MAAM,EAAE,YAAY,EAAE,GAAG,MAAM,MAAM,CAAC,iBAAiB,CAAC,CAAC;QACzD,MAAM,YAAY,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;IAC9B,CAAC;CACF,CAAC,CAAC"}
@@ -0,0 +1,32 @@
1
+ import { register } from "./registry.js";
2
+ import { getFlag } from "./parse.js";
3
+ register({
4
+ name: "chat",
5
+ description: "Interactive REPL",
6
+ usage: `Usage: camelagi chat [options]
7
+
8
+ Start the interactive TUI chat client.
9
+
10
+ Options:
11
+ --session <id> Resume a specific session
12
+
13
+ Examples:
14
+ camelagi chat
15
+ camelagi chat --session my-session`,
16
+ run: async (args) => {
17
+ const sessionId = getFlag(args, "--session");
18
+ const { startServer } = await import("../serve.js");
19
+ const handle = await startServer({
20
+ port: 0,
21
+ silent: true,
22
+ channels: false,
23
+ boot: false,
24
+ cron: false,
25
+ });
26
+ const wsUrl = `ws://127.0.0.1:${handle.port}`;
27
+ const { runTui } = await import("../tui/tui.js");
28
+ await runTui({ session: sessionId, wsUrl });
29
+ await handle.close();
30
+ },
31
+ });
32
+ //# sourceMappingURL=cmd-chat.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cmd-chat.js","sourceRoot":"","sources":["../../src/cli/cmd-chat.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AACzC,OAAO,EAAE,OAAO,EAAE,MAAM,YAAY,CAAC;AAErC,QAAQ,CAAC;IACP,IAAI,EAAE,MAAM;IACZ,WAAW,EAAE,kBAAkB;IAC/B,KAAK,EAAE;;;;;;;;;qCAS4B;IACnC,GAAG,EAAE,KAAK,EAAE,IAAI,EAAE,EAAE;QAClB,MAAM,SAAS,GAAG,OAAO,CAAC,IAAI,EAAE,WAAW,CAAC,CAAC;QAE7C,MAAM,EAAE,WAAW,EAAE,GAAG,MAAM,MAAM,CAAC,aAAa,CAAC,CAAC;QACpD,MAAM,MAAM,GAAG,MAAM,WAAW,CAAC;YAC/B,IAAI,EAAE,CAAC;YACP,MAAM,EAAE,IAAI;YACZ,QAAQ,EAAE,KAAK;YACf,IAAI,EAAE,KAAK;YACX,IAAI,EAAE,KAAK;SACZ,CAAC,CAAC;QAEH,MAAM,KAAK,GAAG,kBAAkB,MAAM,CAAC,IAAI,EAAE,CAAC;QAE9C,MAAM,EAAE,MAAM,EAAE,GAAG,MAAM,MAAM,CAAC,eAAe,CAAC,CAAC;QACjD,MAAM,MAAM,CAAC,EAAE,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC,CAAC;QAE5C,MAAM,MAAM,CAAC,KAAK,EAAE,CAAC;IACvB,CAAC;CACF,CAAC,CAAC"}
@@ -0,0 +1,88 @@
1
+ import { register } from "./registry.js";
2
+ import { loadConfig, saveConfig, ensureDirs } from "../core/config.js";
3
+ register({
4
+ name: "config",
5
+ description: "View/edit config (get, set, list)",
6
+ usage: `Usage: camelagi config <subcommand> [options]
7
+
8
+ View or edit configuration.
9
+
10
+ Subcommands:
11
+ list List all config values (default)
12
+ get <key> Get a specific config value
13
+ set <key> <value> Set a config value
14
+
15
+ Examples:
16
+ camelagi config
17
+ camelagi config list
18
+ camelagi config get model
19
+ camelagi config set model gpt-4o
20
+ camelagi config set maxTurns 50`,
21
+ run: async (args) => {
22
+ ensureDirs();
23
+ const sub = args[0];
24
+ if (sub === "list" || !sub) {
25
+ const config = loadConfig();
26
+ for (const [key, value] of Object.entries(config)) {
27
+ if (key === "apiKey" && value) {
28
+ console.log(` ${key}: ***${value.slice(-4)}`);
29
+ }
30
+ else if (typeof value === "object") {
31
+ console.log(` ${key}: ${JSON.stringify(value)}`);
32
+ }
33
+ else {
34
+ console.log(` ${key}: ${value}`);
35
+ }
36
+ }
37
+ process.exit(0);
38
+ }
39
+ if (sub === "get") {
40
+ const key = args[1];
41
+ if (!key) {
42
+ console.error("Usage: camelagi config get <key>");
43
+ process.exit(1);
44
+ }
45
+ const config = loadConfig();
46
+ const value = config[key];
47
+ if (value === undefined) {
48
+ console.error(`Unknown key: ${key}`);
49
+ process.exit(1);
50
+ }
51
+ if (key === "apiKey" && value) {
52
+ console.log(`***${value.slice(-4)}`);
53
+ }
54
+ else {
55
+ console.log(typeof value === "object" ? JSON.stringify(value, null, 2) : value);
56
+ }
57
+ process.exit(0);
58
+ }
59
+ if (sub === "set") {
60
+ const key = args[1];
61
+ const value = args[2];
62
+ if (!key || value === undefined) {
63
+ console.error("Usage: camelagi config set <key> <value>");
64
+ process.exit(1);
65
+ }
66
+ // Validate key exists in config schema
67
+ const config = loadConfig();
68
+ if (!(key in config)) {
69
+ const validKeys = Object.keys(config).join(", ");
70
+ console.error(`Unknown config key: "${key}"\nValid keys: ${validKeys}`);
71
+ process.exit(1);
72
+ }
73
+ let parsed = value;
74
+ if (value === "true")
75
+ parsed = true;
76
+ else if (value === "false")
77
+ parsed = false;
78
+ else if (/^\d+$/.test(value))
79
+ parsed = parseInt(value, 10);
80
+ saveConfig({ [key]: parsed });
81
+ console.log(`Set ${key} = ${value}`);
82
+ process.exit(0);
83
+ }
84
+ console.error(`Unknown config subcommand: ${sub}. Use: list, get, set`);
85
+ process.exit(1);
86
+ },
87
+ });
88
+ //# sourceMappingURL=cmd-config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cmd-config.js","sourceRoot":"","sources":["../../src/cli/cmd-config.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,120 @@
1
+ import { register } from "./registry.js";
2
+ import { loadConfig, ensureDirs } from "../core/config.js";
3
+ import { getFlag, validateSchedule } from "./parse.js";
4
+ register({
5
+ name: "cron",
6
+ description: "Manage cron jobs (list, add, rm, run)",
7
+ usage: `Usage: camelagi cron <subcommand> [options]
8
+
9
+ Manage scheduled AI tasks.
10
+
11
+ Subcommands:
12
+ list List all cron jobs (default)
13
+ add [options] Create a new cron job
14
+ rm <id> Remove a runtime job
15
+ run <id> Run a job immediately
16
+
17
+ Add options:
18
+ --name <name> Job name (default: "Untitled")
19
+ --schedule <schedule> Schedule: 5m, 1h, 1d, +20m, */5 * * * *, ISO timestamp
20
+ --prompt <prompt> Prompt to run (required)
21
+ --id <id> Custom job ID
22
+
23
+ Examples:
24
+ camelagi cron list
25
+ camelagi cron add --name "Daily" --schedule "1d" --prompt "summarize today"
26
+ camelagi cron rm job-abc123
27
+ camelagi cron run daily-summary`,
28
+ run: async (args) => {
29
+ ensureDirs();
30
+ const { getAllJobStatuses, addRuntimeJob, removeRuntimeJob, loadRuntimeJobs } = await import("../extensions/cron.js");
31
+ const sub = args[0];
32
+ if (!sub || sub === "list") {
33
+ const config = loadConfig();
34
+ const configJobs = config.cron.filter((j) => j.enabled);
35
+ const runtimeJobs = loadRuntimeJobs();
36
+ const allJobs = [
37
+ ...configJobs.map((j) => ({ ...j, source: "config" })),
38
+ ...runtimeJobs.map((j) => ({ ...j, source: "runtime" })),
39
+ ];
40
+ if (allJobs.length === 0) {
41
+ console.log("No cron jobs. Add one with: camelagi cron add --name 'My Job' --schedule '1h' --prompt 'do something'");
42
+ }
43
+ else {
44
+ for (const j of allJobs) {
45
+ const status = j.enabled ? "\x1b[32m●\x1b[0m" : "\x1b[90m○\x1b[0m";
46
+ const src = j.source === "config" ? "\x1b[90m(config)\x1b[0m" : "\x1b[36m(runtime)\x1b[0m";
47
+ console.log(` ${status} ${j.id} ${j.name || "(unnamed)"} ${j.schedule} ${src}`);
48
+ console.log(` \x1b[90m${j.prompt.slice(0, 100)}${j.prompt.length > 100 ? "..." : ""}\x1b[0m`);
49
+ }
50
+ }
51
+ process.exit(0);
52
+ }
53
+ if (sub === "add") {
54
+ const name = getFlag(args, "--name");
55
+ const schedule = getFlag(args, "--schedule");
56
+ const prompt = getFlag(args, "--prompt");
57
+ const id = getFlag(args, "--id");
58
+ if (!schedule || !prompt) {
59
+ console.error("Usage: camelagi cron add --name 'Job Name' --schedule '5m' --prompt 'do something'");
60
+ console.error("\nSchedule formats: 5m, 1h, 1d, */5 * * * *, +20m (one-shot), ISO timestamp (one-shot)");
61
+ process.exit(1);
62
+ }
63
+ try {
64
+ validateSchedule(schedule);
65
+ }
66
+ catch (err) {
67
+ console.error(`Error: ${err instanceof Error ? err.message : String(err)}`);
68
+ process.exit(1);
69
+ }
70
+ try {
71
+ const job = addRuntimeJob({
72
+ id: id || `job-${Date.now().toString(36)}`,
73
+ name: name || "Untitled",
74
+ schedule,
75
+ prompt,
76
+ enabled: true,
77
+ }, false);
78
+ console.log(`\x1b[32m✓\x1b[0m Created job: ${job.id} (${schedule})`);
79
+ console.log(" Will start on next \x1b[36mcamelagi serve\x1b[0m");
80
+ }
81
+ catch (err) {
82
+ console.error(`Error: ${err instanceof Error ? err.message : String(err)}`);
83
+ process.exit(1);
84
+ }
85
+ process.exit(0);
86
+ }
87
+ if (sub === "rm" && args[1]) {
88
+ const removed = removeRuntimeJob(args[1]);
89
+ if (removed) {
90
+ console.log(`\x1b[32m✓\x1b[0m Removed job: ${args[1]}`);
91
+ }
92
+ else {
93
+ console.error(`Job "${args[1]}" not found. Config-defined jobs must be removed from config.yaml.`);
94
+ process.exit(1);
95
+ }
96
+ process.exit(0);
97
+ }
98
+ if (sub === "run" && args[1]) {
99
+ console.log(`Triggering job "${args[1]}" via embedded server...`);
100
+ const { startServer } = await import("../serve.js");
101
+ const handle = await startServer({ port: 0, silent: true, channels: false, boot: false, cron: false });
102
+ try {
103
+ const { runJobNow } = await import("../extensions/cron.js");
104
+ const response = await runJobNow(args[1]);
105
+ console.log(`\x1b[32m✓\x1b[0m Response:\n${response}`);
106
+ }
107
+ catch (err) {
108
+ console.error(`Error: ${err instanceof Error ? err.message : String(err)}`);
109
+ process.exit(1);
110
+ }
111
+ finally {
112
+ await handle.close();
113
+ }
114
+ process.exit(0);
115
+ }
116
+ console.error(`Unknown cron subcommand: ${sub}. Use: list, add, rm <id>, run <id>`);
117
+ process.exit(1);
118
+ },
119
+ });
120
+ //# sourceMappingURL=cmd-cron.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cmd-cron.js","sourceRoot":"","sources":["../../src/cli/cmd-cron.ts"],"names":[],"mappings":"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"}