homaruscc 0.2.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 (171) hide show
  1. package/.env.example +8 -0
  2. package/LICENSE +21 -0
  3. package/README.md +307 -0
  4. package/bin/event-loop +60 -0
  5. package/config.example.json +55 -0
  6. package/dist/agent-registry.d.ts +38 -0
  7. package/dist/agent-registry.d.ts.map +1 -0
  8. package/dist/agent-registry.js +207 -0
  9. package/dist/agent-registry.js.map +1 -0
  10. package/dist/backend.d.ts +3 -0
  11. package/dist/backend.d.ts.map +1 -0
  12. package/dist/backend.js +59 -0
  13. package/dist/backend.js.map +1 -0
  14. package/dist/browser-service.d.ts +19 -0
  15. package/dist/browser-service.d.ts.map +1 -0
  16. package/dist/browser-service.js +101 -0
  17. package/dist/browser-service.js.map +1 -0
  18. package/dist/channel-adapter.d.ts +22 -0
  19. package/dist/channel-adapter.d.ts.map +1 -0
  20. package/dist/channel-adapter.js +62 -0
  21. package/dist/channel-adapter.js.map +1 -0
  22. package/dist/channel-manager.d.ts +18 -0
  23. package/dist/channel-manager.d.ts.map +1 -0
  24. package/dist/channel-manager.js +73 -0
  25. package/dist/channel-manager.js.map +1 -0
  26. package/dist/compaction-manager.d.ts +23 -0
  27. package/dist/compaction-manager.d.ts.map +1 -0
  28. package/dist/compaction-manager.js +135 -0
  29. package/dist/compaction-manager.js.map +1 -0
  30. package/dist/config.d.ts +21 -0
  31. package/dist/config.d.ts.map +1 -0
  32. package/dist/config.js +169 -0
  33. package/dist/config.js.map +1 -0
  34. package/dist/dashboard-adapter.d.ts +17 -0
  35. package/dist/dashboard-adapter.d.ts.map +1 -0
  36. package/dist/dashboard-adapter.js +41 -0
  37. package/dist/dashboard-adapter.js.map +1 -0
  38. package/dist/dashboard-server.d.ts +29 -0
  39. package/dist/dashboard-server.d.ts.map +1 -0
  40. package/dist/dashboard-server.js +381 -0
  41. package/dist/dashboard-server.js.map +1 -0
  42. package/dist/embedding-provider.d.ts +28 -0
  43. package/dist/embedding-provider.d.ts.map +1 -0
  44. package/dist/embedding-provider.js +91 -0
  45. package/dist/embedding-provider.js.map +1 -0
  46. package/dist/event-bus.d.ts +18 -0
  47. package/dist/event-bus.d.ts.map +1 -0
  48. package/dist/event-bus.js +41 -0
  49. package/dist/event-bus.js.map +1 -0
  50. package/dist/event-queue.d.ts +18 -0
  51. package/dist/event-queue.d.ts.map +1 -0
  52. package/dist/event-queue.js +68 -0
  53. package/dist/event-queue.js.map +1 -0
  54. package/dist/homaruscc.d.ts +69 -0
  55. package/dist/homaruscc.d.ts.map +1 -0
  56. package/dist/homaruscc.js +337 -0
  57. package/dist/homaruscc.js.map +1 -0
  58. package/dist/identity-manager.d.ts +36 -0
  59. package/dist/identity-manager.d.ts.map +1 -0
  60. package/dist/identity-manager.js +142 -0
  61. package/dist/identity-manager.js.map +1 -0
  62. package/dist/mcp-proxy.d.ts +3 -0
  63. package/dist/mcp-proxy.d.ts.map +1 -0
  64. package/dist/mcp-proxy.js +259 -0
  65. package/dist/mcp-proxy.js.map +1 -0
  66. package/dist/mcp-resources.d.ts +10 -0
  67. package/dist/mcp-resources.d.ts.map +1 -0
  68. package/dist/mcp-resources.js +67 -0
  69. package/dist/mcp-resources.js.map +1 -0
  70. package/dist/mcp-server.d.ts +3 -0
  71. package/dist/mcp-server.d.ts.map +1 -0
  72. package/dist/mcp-server.js +169 -0
  73. package/dist/mcp-server.js.map +1 -0
  74. package/dist/mcp-tools.d.ts +14 -0
  75. package/dist/mcp-tools.d.ts.map +1 -0
  76. package/dist/mcp-tools.js +408 -0
  77. package/dist/mcp-tools.js.map +1 -0
  78. package/dist/memory-index.d.ts +79 -0
  79. package/dist/memory-index.d.ts.map +1 -0
  80. package/dist/memory-index.js +437 -0
  81. package/dist/memory-index.js.map +1 -0
  82. package/dist/session-checkpoint.d.ts +22 -0
  83. package/dist/session-checkpoint.d.ts.map +1 -0
  84. package/dist/session-checkpoint.js +100 -0
  85. package/dist/session-checkpoint.js.map +1 -0
  86. package/dist/skill-manager.d.ts +26 -0
  87. package/dist/skill-manager.d.ts.map +1 -0
  88. package/dist/skill-manager.js +156 -0
  89. package/dist/skill-manager.js.map +1 -0
  90. package/dist/skill-transport.d.ts +45 -0
  91. package/dist/skill-transport.d.ts.map +1 -0
  92. package/dist/skill-transport.js +111 -0
  93. package/dist/skill-transport.js.map +1 -0
  94. package/dist/skill.d.ts +22 -0
  95. package/dist/skill.d.ts.map +1 -0
  96. package/dist/skill.js +106 -0
  97. package/dist/skill.js.map +1 -0
  98. package/dist/telegram-adapter.d.ts +43 -0
  99. package/dist/telegram-adapter.d.ts.map +1 -0
  100. package/dist/telegram-adapter.js +188 -0
  101. package/dist/telegram-adapter.js.map +1 -0
  102. package/dist/timer-service.d.ts +30 -0
  103. package/dist/timer-service.d.ts.map +1 -0
  104. package/dist/timer-service.js +176 -0
  105. package/dist/timer-service.js.map +1 -0
  106. package/dist/tool-registry.d.ts +30 -0
  107. package/dist/tool-registry.d.ts.map +1 -0
  108. package/dist/tool-registry.js +108 -0
  109. package/dist/tool-registry.js.map +1 -0
  110. package/dist/tools/bash.d.ts +3 -0
  111. package/dist/tools/bash.d.ts.map +1 -0
  112. package/dist/tools/bash.js +67 -0
  113. package/dist/tools/bash.js.map +1 -0
  114. package/dist/tools/browser.d.ts +4 -0
  115. package/dist/tools/browser.d.ts.map +1 -0
  116. package/dist/tools/browser.js +138 -0
  117. package/dist/tools/browser.js.map +1 -0
  118. package/dist/tools/edit.d.ts +3 -0
  119. package/dist/tools/edit.d.ts.map +1 -0
  120. package/dist/tools/edit.js +47 -0
  121. package/dist/tools/edit.js.map +1 -0
  122. package/dist/tools/git.d.ts +3 -0
  123. package/dist/tools/git.d.ts.map +1 -0
  124. package/dist/tools/git.js +105 -0
  125. package/dist/tools/git.js.map +1 -0
  126. package/dist/tools/glob.d.ts +3 -0
  127. package/dist/tools/glob.d.ts.map +1 -0
  128. package/dist/tools/glob.js +84 -0
  129. package/dist/tools/glob.js.map +1 -0
  130. package/dist/tools/grep.d.ts +3 -0
  131. package/dist/tools/grep.d.ts.map +1 -0
  132. package/dist/tools/grep.js +168 -0
  133. package/dist/tools/grep.js.map +1 -0
  134. package/dist/tools/index.d.ts +6 -0
  135. package/dist/tools/index.d.ts.map +1 -0
  136. package/dist/tools/index.js +46 -0
  137. package/dist/tools/index.js.map +1 -0
  138. package/dist/tools/memory.d.ts +4 -0
  139. package/dist/tools/memory.d.ts.map +1 -0
  140. package/dist/tools/memory.js +64 -0
  141. package/dist/tools/memory.js.map +1 -0
  142. package/dist/tools/read.d.ts +3 -0
  143. package/dist/tools/read.d.ts.map +1 -0
  144. package/dist/tools/read.js +50 -0
  145. package/dist/tools/read.js.map +1 -0
  146. package/dist/tools/web-fetch.d.ts +3 -0
  147. package/dist/tools/web-fetch.d.ts.map +1 -0
  148. package/dist/tools/web-fetch.js +51 -0
  149. package/dist/tools/web-fetch.js.map +1 -0
  150. package/dist/tools/web-search.d.ts +3 -0
  151. package/dist/tools/web-search.d.ts.map +1 -0
  152. package/dist/tools/web-search.js +65 -0
  153. package/dist/tools/web-search.js.map +1 -0
  154. package/dist/tools/write.d.ts +3 -0
  155. package/dist/tools/write.d.ts.map +1 -0
  156. package/dist/tools/write.js +32 -0
  157. package/dist/tools/write.js.map +1 -0
  158. package/dist/transcript-logger.d.ts +23 -0
  159. package/dist/transcript-logger.d.ts.map +1 -0
  160. package/dist/transcript-logger.js +101 -0
  161. package/dist/transcript-logger.js.map +1 -0
  162. package/dist/types.d.ts +190 -0
  163. package/dist/types.d.ts.map +1 -0
  164. package/dist/types.js +14 -0
  165. package/dist/types.js.map +1 -0
  166. package/identity.example/disagreements.md +9 -0
  167. package/identity.example/preferences.md +11 -0
  168. package/identity.example/soul.md +12 -0
  169. package/identity.example/state.md +21 -0
  170. package/identity.example/user.md +14 -0
  171. package/package.json +60 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"telegram-adapter.d.ts","sourceRoot":"","sources":["../src/telegram-adapter.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,eAAe,EAAE,YAAY,EAAE,MAAM,EAAE,aAAa,EAAE,QAAQ,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AAC9G,OAAO,EAAE,cAAc,EAAE,MAAM,sBAAsB,CAAC;AAuCtD,MAAM,WAAW,qBAAqB;IACpC,KAAK,EAAE,MAAM,CAAC;IACd,QAAQ,CAAC,EAAE,QAAQ,CAAC;IACpB,WAAW,CAAC,EAAE,WAAW,CAAC;IAC1B,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,cAAc,CAAC,EAAE,MAAM,EAAE,CAAC;CAC3B;AAOD,qBAAa,sBAAuB,SAAQ,cAAc;IACxD,OAAO,CAAC,KAAK,CAAS;IACtB,OAAO,CAAC,WAAW,CAAM;IACzB,OAAO,CAAC,MAAM,CAAK;IACnB,OAAO,CAAC,SAAS,CAA8C;IAC/D,OAAO,CAAC,QAAQ,CAAS;IACzB,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,YAAY,CAAK;IACzB,OAAO,CAAC,SAAS,CAAmB;IACpC,OAAO,CAAC,eAAe,CAAS;IAChC,OAAO,CAAC,cAAc,CAAc;IACpC,OAAO,CAAC,cAAc,CAAgF;IACtG,OAAO,CAAC,iBAAiB,CAAM;gBAEnB,MAAM,EAAE,qBAAqB,EAAE,MAAM,EAAE,MAAM;IAOzD,MAAM,CAAC,iBAAiB,CAAC,MAAM,EAAE,aAAa,EAAE,MAAM,EAAE,MAAM,GAAG,sBAAsB;IAajF,OAAO,IAAI,OAAO,CAAC,IAAI,CAAC;IAaxB,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC;IAU3B,IAAI,CAAC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,eAAe,GAAG,OAAO,CAAC,IAAI,CAAC;IAsBnE,MAAM,IAAI,YAAY;IAUhB,UAAU,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAK/C,iBAAiB,CAAC,KAAK,SAAK,GAAG,KAAK,CAAC;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC;QAAC,SAAS,EAAE,MAAM,CAAA;KAAE,CAAC;IAIvG,OAAO,CAAC,YAAY;YAKN,IAAI;IAiClB,OAAO,CAAC,aAAa;IAoCrB,OAAO,CAAC,aAAa;IASrB,OAAO,CAAC,YAAY;YAIN,OAAO;CAgBtB"}
@@ -0,0 +1,188 @@
1
+ import { ChannelAdapter } from "./channel-adapter.js";
2
+ const BASE_URL = "https://api.telegram.org/bot";
3
+ const POLL_TIMEOUT = 30;
4
+ const MAX_BACKOFF = 30_000;
5
+ const INITIAL_BACKOFF = 1_000;
6
+ export class TelegramChannelAdapter extends ChannelAdapter {
7
+ token;
8
+ botUsername = "";
9
+ offset = 0;
10
+ pollTimer = null;
11
+ stopping = false;
12
+ lastPollOk = false;
13
+ lastPollTime = 0;
14
+ backoffMs = INITIAL_BACKOFF;
15
+ pollingInterval;
16
+ allowedChatIds;
17
+ recentMessages = [];
18
+ maxRecentMessages = 50;
19
+ constructor(config, logger) {
20
+ super("telegram", logger, config.dmPolicy ?? "open", config.groupPolicy ?? "mention_required");
21
+ this.token = config.token;
22
+ this.pollingInterval = config.pollingInterval ?? 1000;
23
+ this.allowedChatIds = new Set(config.allowedChatIds ?? []);
24
+ }
25
+ static fromChannelConfig(config, logger) {
26
+ if (!config.token || typeof config.token !== "string") {
27
+ throw new Error("Telegram adapter requires a 'token' in channel config");
28
+ }
29
+ return new TelegramChannelAdapter({
30
+ token: config.token,
31
+ dmPolicy: config.dmPolicy,
32
+ groupPolicy: config.groupPolicy,
33
+ pollingInterval: config.pollingInterval,
34
+ allowedChatIds: config.allowedChatIds,
35
+ }, logger);
36
+ }
37
+ async connect() {
38
+ this.state = "connecting";
39
+ this.stopping = false;
40
+ const me = await this.apiCall("getMe");
41
+ this.botUsername = me.username ?? "";
42
+ this.logger.info("Telegram connected", { username: this.botUsername });
43
+ this.state = "connected";
44
+ this.backoffMs = INITIAL_BACKOFF;
45
+ this.schedulePoll();
46
+ }
47
+ async disconnect() {
48
+ this.stopping = true;
49
+ if (this.pollTimer) {
50
+ clearTimeout(this.pollTimer);
51
+ this.pollTimer = null;
52
+ }
53
+ this.state = "disconnected";
54
+ this.logger.info("Telegram disconnected");
55
+ }
56
+ async send(target, message) {
57
+ const body = {
58
+ chat_id: target,
59
+ text: message.text,
60
+ parse_mode: "Markdown",
61
+ };
62
+ if (message.replyTo) {
63
+ body.reply_to_message_id = Number(message.replyTo);
64
+ }
65
+ try {
66
+ await this.apiCall("sendMessage", body);
67
+ }
68
+ catch (err) {
69
+ // Markdown parse failures are common — retry without parse_mode
70
+ if (String(err).includes("can't parse entities")) {
71
+ delete body.parse_mode;
72
+ await this.apiCall("sendMessage", body);
73
+ }
74
+ else {
75
+ throw err;
76
+ }
77
+ }
78
+ }
79
+ health() {
80
+ return {
81
+ healthy: this.state === "connected" && this.lastPollOk,
82
+ message: this.state === "connected"
83
+ ? (this.lastPollOk ? "polling" : "poll error, retrying")
84
+ : this.state,
85
+ lastCheck: this.lastPollTime || Date.now(),
86
+ };
87
+ }
88
+ async sendTyping(chatId) {
89
+ await this.apiCall("sendChatAction", { chat_id: chatId, action: "typing" });
90
+ }
91
+ // homaruscc addition: get recent messages for MCP tool
92
+ getRecentMessages(limit = 20) {
93
+ return this.recentMessages.slice(-limit);
94
+ }
95
+ schedulePoll() {
96
+ if (this.stopping)
97
+ return;
98
+ this.pollTimer = setTimeout(() => this.poll(), this.pollingInterval);
99
+ }
100
+ async poll() {
101
+ if (this.stopping)
102
+ return;
103
+ try {
104
+ const updates = await this.apiCall("getUpdates", {
105
+ offset: this.offset,
106
+ timeout: POLL_TIMEOUT,
107
+ });
108
+ this.lastPollOk = true;
109
+ this.lastPollTime = Date.now();
110
+ this.backoffMs = INITIAL_BACKOFF;
111
+ for (const update of updates) {
112
+ this.offset = update.update_id + 1;
113
+ if (update.message) {
114
+ this.handleMessage(update.message);
115
+ }
116
+ }
117
+ this.schedulePoll();
118
+ }
119
+ catch (err) {
120
+ this.lastPollOk = false;
121
+ this.lastPollTime = Date.now();
122
+ this.logger.error("Telegram poll error", { error: String(err), backoffMs: this.backoffMs });
123
+ if (!this.stopping) {
124
+ this.pollTimer = setTimeout(() => this.poll(), this.backoffMs);
125
+ this.backoffMs = Math.min(this.backoffMs * 2, MAX_BACKOFF);
126
+ }
127
+ }
128
+ }
129
+ handleMessage(msg) {
130
+ if (!msg.text)
131
+ return;
132
+ if (this.allowedChatIds.size > 0 && !this.allowedChatIds.has(msg.chat.id)) {
133
+ this.logger.debug("Message from non-whitelisted chat, ignoring", { chatId: msg.chat.id });
134
+ return;
135
+ }
136
+ const isGroup = msg.chat.type === "group" || msg.chat.type === "supergroup";
137
+ const isMention = this.detectMention(msg);
138
+ // Store for MCP read tool
139
+ this.recentMessages.push({
140
+ from: msg.from?.username ?? String(msg.from?.id ?? "unknown"),
141
+ text: isMention ? this.stripMention(msg.text) : msg.text,
142
+ chatId: String(msg.chat.id),
143
+ timestamp: Date.now(),
144
+ });
145
+ if (this.recentMessages.length > this.maxRecentMessages) {
146
+ this.recentMessages.shift();
147
+ }
148
+ // Send typing indicator immediately so user sees we're processing
149
+ this.sendTyping(String(msg.chat.id)).catch(() => { });
150
+ this.deliverWithTarget({
151
+ from: msg.from?.username ?? String(msg.from?.id ?? "unknown"),
152
+ channel: "telegram",
153
+ text: isMention ? this.stripMention(msg.text) : msg.text,
154
+ isGroup,
155
+ isMention,
156
+ replyTo: String(msg.message_id),
157
+ raw: msg,
158
+ }, String(msg.chat.id));
159
+ }
160
+ detectMention(msg) {
161
+ if (!this.botUsername || !msg.entities)
162
+ return false;
163
+ return msg.entities.some((e) => {
164
+ if (e.type !== "mention")
165
+ return false;
166
+ const mentionText = msg.text.substring(e.offset, e.offset + e.length);
167
+ return mentionText.toLowerCase() === `@${this.botUsername.toLowerCase()}`;
168
+ });
169
+ }
170
+ stripMention(text) {
171
+ return text.replace(new RegExp(`@${this.botUsername}\\b`, "gi"), "").trim();
172
+ }
173
+ async apiCall(method, body) {
174
+ const url = `${BASE_URL}${this.token}/${method}`;
175
+ const opts = {
176
+ method: "POST",
177
+ headers: { "Content-Type": "application/json" },
178
+ body: body ? JSON.stringify(body) : undefined,
179
+ };
180
+ const res = await fetch(url, opts);
181
+ const json = (await res.json());
182
+ if (!json.ok) {
183
+ throw new Error(`Telegram API error (${method}): ${json.description ?? "unknown"}`);
184
+ }
185
+ return json.result;
186
+ }
187
+ }
188
+ //# sourceMappingURL=telegram-adapter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"telegram-adapter.js","sourceRoot":"","sources":["../src/telegram-adapter.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,30 @@
1
+ import type { Event, Logger } from "./types.js";
2
+ export interface TimerConfig {
3
+ id?: string;
4
+ name: string;
5
+ type: "cron" | "interval" | "once";
6
+ schedule: string;
7
+ prompt: string;
8
+ timezone?: string;
9
+ }
10
+ export declare class TimerService {
11
+ private timers;
12
+ private storePath;
13
+ private logger;
14
+ private emitFn;
15
+ constructor(logger: Logger, storePath: string);
16
+ setEmitter(fn: (event: Event) => void): void;
17
+ loadTimers(): void;
18
+ saveTimers(): void;
19
+ add(config: TimerConfig): string;
20
+ remove(timerIdOrName: string): void;
21
+ get(timerId: string): TimerConfig | undefined;
22
+ getAll(): TimerConfig[];
23
+ registerDefaults(defaults: TimerConfig[]): void;
24
+ start(): void;
25
+ stop(): void;
26
+ private onFire;
27
+ private startTimer;
28
+ private stopTimer;
29
+ }
30
+ //# sourceMappingURL=timer-service.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"timer-service.d.ts","sourceRoot":"","sources":["../src/timer-service.ts"],"names":[],"mappings":"AAMA,OAAO,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,YAAY,CAAC;AAEhD,MAAM,WAAW,WAAW;IAC1B,EAAE,CAAC,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,GAAG,UAAU,GAAG,MAAM,CAAC;IACnC,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,EAAE,MAAM,CAAC;IACf,QAAQ,CAAC,EAAE,MAAM,CAAC;CACnB;AAQD,qBAAa,YAAY;IACvB,OAAO,CAAC,MAAM,CAAiC;IAC/C,OAAO,CAAC,SAAS,CAAS;IAC1B,OAAO,CAAC,MAAM,CAAS;IACvB,OAAO,CAAC,MAAM,CAAyC;gBAE3C,MAAM,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM;IAK7C,UAAU,CAAC,EAAE,EAAE,CAAC,KAAK,EAAE,KAAK,KAAK,IAAI,GAAG,IAAI;IAI5C,UAAU,IAAI,IAAI;IAwBlB,UAAU,IAAI,IAAI;IAOlB,GAAG,CAAC,MAAM,EAAE,WAAW,GAAG,MAAM;IAkBhC,MAAM,CAAC,aAAa,EAAE,MAAM,GAAG,IAAI;IAmBnC,GAAG,CAAC,OAAO,EAAE,MAAM,GAAG,WAAW,GAAG,SAAS;IAI7C,MAAM,IAAI,WAAW,EAAE;IAIvB,gBAAgB,CAAC,QAAQ,EAAE,WAAW,EAAE,GAAG,IAAI;IAc/C,KAAK,IAAI,IAAI;IAOb,IAAI,IAAI,IAAI;IAQZ,OAAO,CAAC,MAAM;IAyBd,OAAO,CAAC,UAAU;IAsBlB,OAAO,CAAC,SAAS;CAWlB"}
@@ -0,0 +1,176 @@
1
+ // CRC: crc-TimerService.md | Seq: seq-timer-fire.md
2
+ // Timer service — from HomarUS
3
+ import { readFileSync, writeFileSync, existsSync, mkdirSync } from "node:fs";
4
+ import { dirname } from "node:path";
5
+ import { v4 as uuid } from "uuid";
6
+ import { Cron } from "croner";
7
+ export class TimerService {
8
+ timers = new Map();
9
+ storePath;
10
+ logger;
11
+ emitFn = null;
12
+ constructor(logger, storePath) {
13
+ this.logger = logger;
14
+ this.storePath = storePath;
15
+ }
16
+ setEmitter(fn) {
17
+ this.emitFn = fn;
18
+ }
19
+ loadTimers() {
20
+ if (!existsSync(this.storePath))
21
+ return;
22
+ try {
23
+ const data = JSON.parse(readFileSync(this.storePath, "utf-8"));
24
+ // Dedup by name on load — last entry wins
25
+ const byName = new Map();
26
+ for (const config of data) {
27
+ config.id = config.id ?? uuid();
28
+ byName.set(config.name, config);
29
+ }
30
+ for (const config of byName.values()) {
31
+ this.timers.set(config.id, { config });
32
+ }
33
+ // Save deduped version back
34
+ if (byName.size < data.length) {
35
+ this.saveTimers();
36
+ this.logger.info("Deduped timers on load", { before: data.length, after: byName.size });
37
+ }
38
+ this.logger.info("Loaded timers", { count: this.timers.size });
39
+ }
40
+ catch (err) {
41
+ this.logger.error("Failed to load timers", { error: String(err) });
42
+ }
43
+ }
44
+ saveTimers() {
45
+ const data = [...this.timers.values()].map((e) => e.config);
46
+ const dir = dirname(this.storePath);
47
+ if (!existsSync(dir))
48
+ mkdirSync(dir, { recursive: true });
49
+ writeFileSync(this.storePath, JSON.stringify(data, null, 2));
50
+ }
51
+ add(config) {
52
+ // Dedup: if a timer with the same name exists, remove it first
53
+ for (const [existingId, entry] of this.timers) {
54
+ if (entry.config.name === config.name) {
55
+ this.stopTimer(entry);
56
+ this.timers.delete(existingId);
57
+ this.logger.info("Replaced existing timer", { id: existingId, name: config.name });
58
+ }
59
+ }
60
+ const id = config.id ?? uuid();
61
+ config.id = id;
62
+ this.timers.set(id, { config });
63
+ this.startTimer(id);
64
+ this.saveTimers();
65
+ this.logger.info("Timer added", { id, name: config.name, type: config.type });
66
+ return id;
67
+ }
68
+ remove(timerIdOrName) {
69
+ // Try by ID first, then by name
70
+ let id = timerIdOrName;
71
+ if (!this.timers.has(id)) {
72
+ for (const [existingId, entry] of this.timers) {
73
+ if (entry.config.name === timerIdOrName) {
74
+ id = existingId;
75
+ break;
76
+ }
77
+ }
78
+ }
79
+ const entry = this.timers.get(id);
80
+ if (!entry)
81
+ return;
82
+ this.stopTimer(entry);
83
+ this.timers.delete(id);
84
+ this.saveTimers();
85
+ this.logger.info("Timer removed", { id, name: entry.config.name });
86
+ }
87
+ get(timerId) {
88
+ return this.timers.get(timerId)?.config;
89
+ }
90
+ getAll() {
91
+ return [...this.timers.values()].map((e) => e.config);
92
+ }
93
+ registerDefaults(defaults) {
94
+ const existingNames = new Set([...this.timers.values()].map(e => e.config.name));
95
+ let added = 0;
96
+ for (const config of defaults) {
97
+ if (existingNames.has(config.name))
98
+ continue;
99
+ const id = config.id ?? uuid();
100
+ config.id = id;
101
+ this.timers.set(id, { config });
102
+ added++;
103
+ this.logger.info("Registered default timer", { name: config.name });
104
+ }
105
+ if (added > 0)
106
+ this.saveTimers();
107
+ }
108
+ start() {
109
+ for (const [id] of this.timers) {
110
+ this.startTimer(id);
111
+ }
112
+ this.logger.info("Timer service started", { count: this.timers.size });
113
+ }
114
+ stop() {
115
+ for (const entry of this.timers.values()) {
116
+ this.stopTimer(entry);
117
+ }
118
+ this.saveTimers();
119
+ this.logger.info("Timer service stopped");
120
+ }
121
+ onFire(timerId) {
122
+ const entry = this.timers.get(timerId);
123
+ if (!entry || !this.emitFn)
124
+ return;
125
+ const event = {
126
+ id: uuid(),
127
+ type: "timer_fired",
128
+ source: `timer:${timerId}`,
129
+ timestamp: Date.now(),
130
+ payload: {
131
+ timerId,
132
+ name: entry.config.name,
133
+ prompt: entry.config.prompt,
134
+ },
135
+ };
136
+ this.logger.info("Timer fired", { id: timerId, name: entry.config.name });
137
+ this.emitFn(event);
138
+ if (entry.config.type === "once") {
139
+ this.timers.delete(timerId);
140
+ this.saveTimers();
141
+ }
142
+ }
143
+ startTimer(id) {
144
+ const entry = this.timers.get(id);
145
+ if (!entry)
146
+ return;
147
+ this.stopTimer(entry);
148
+ const { config } = entry;
149
+ switch (config.type) {
150
+ case "cron":
151
+ entry.job = new Cron(config.schedule, { timezone: config.timezone }, () => this.onFire(id));
152
+ break;
153
+ case "interval":
154
+ entry.timeout = setInterval(() => this.onFire(id), parseInt(config.schedule, 10));
155
+ break;
156
+ case "once": {
157
+ const fireAt = new Date(config.schedule).getTime();
158
+ const delay = Math.max(0, fireAt - Date.now());
159
+ entry.timeout = setTimeout(() => this.onFire(id), delay);
160
+ break;
161
+ }
162
+ }
163
+ }
164
+ stopTimer(entry) {
165
+ if (entry.job) {
166
+ entry.job.stop();
167
+ entry.job = undefined;
168
+ }
169
+ if (entry.timeout) {
170
+ clearTimeout(entry.timeout);
171
+ clearInterval(entry.timeout);
172
+ entry.timeout = undefined;
173
+ }
174
+ }
175
+ }
176
+ //# sourceMappingURL=timer-service.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"timer-service.js","sourceRoot":"","sources":["../src/timer-service.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,30 @@
1
+ import type { ToolDefinition, ToolResult, ToolContext, Logger } from "./types.js";
2
+ export interface ToolPolicy {
3
+ name: string;
4
+ allow?: string[];
5
+ deny?: string[];
6
+ }
7
+ export declare class ToolRegistry {
8
+ private tools;
9
+ private policies;
10
+ private groups;
11
+ private logger;
12
+ constructor(logger: Logger);
13
+ register(tool: ToolDefinition): void;
14
+ unregister(name: string): void;
15
+ get(name: string): ToolDefinition | undefined;
16
+ getAll(): ToolDefinition[];
17
+ getForAgent(allowedTools?: string[]): ToolDefinition[];
18
+ execute(name: string, params: unknown, context: ToolContext): Promise<ToolResult>;
19
+ registerGroup(name: string, toolNames: string[]): void;
20
+ resolveGroup(groupName: string): string[];
21
+ addPolicy(policy: ToolPolicy): void;
22
+ checkPolicy(toolName: string, _context: ToolContext): boolean;
23
+ private resolveNames;
24
+ toSchemas(): Array<{
25
+ name: string;
26
+ description: string;
27
+ parameters: Record<string, unknown>;
28
+ }>;
29
+ }
30
+ //# sourceMappingURL=tool-registry.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tool-registry.d.ts","sourceRoot":"","sources":["../src/tool-registry.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,cAAc,EAAE,UAAU,EAAE,WAAW,EAAE,MAAM,EAAE,MAAM,YAAY,CAAC;AAElF,MAAM,WAAW,UAAU;IACzB,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,CAAC,EAAE,MAAM,EAAE,CAAC;IACjB,IAAI,CAAC,EAAE,MAAM,EAAE,CAAC;CACjB;AAUD,qBAAa,YAAY;IACvB,OAAO,CAAC,KAAK,CAAqC;IAClD,OAAO,CAAC,QAAQ,CAAoB;IACpC,OAAO,CAAC,MAAM,CAA6D;IAC3E,OAAO,CAAC,MAAM,CAAS;gBAEX,MAAM,EAAE,MAAM;IAI1B,QAAQ,CAAC,IAAI,EAAE,cAAc,GAAG,IAAI;IAKpC,UAAU,CAAC,IAAI,EAAE,MAAM,GAAG,IAAI;IAI9B,GAAG,CAAC,IAAI,EAAE,MAAM,GAAG,cAAc,GAAG,SAAS;IAI7C,MAAM,IAAI,cAAc,EAAE;IAI1B,WAAW,CAAC,YAAY,CAAC,EAAE,MAAM,EAAE,GAAG,cAAc,EAAE;IAiBhD,OAAO,CAAC,IAAI,EAAE,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,WAAW,GAAG,OAAO,CAAC,UAAU,CAAC;IAyBvF,aAAa,CAAC,IAAI,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,EAAE,GAAG,IAAI;IAItD,YAAY,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM,EAAE;IAIzC,SAAS,CAAC,MAAM,EAAE,UAAU,GAAG,IAAI;IAInC,WAAW,CAAC,QAAQ,EAAE,MAAM,EAAE,QAAQ,EAAE,WAAW,GAAG,OAAO;IAY7D,OAAO,CAAC,YAAY;IAUpB,SAAS,IAAI,KAAK,CAAC;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,WAAW,EAAE,MAAM,CAAC;QAAC,UAAU,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;KAAE,CAAC;CAO/F"}
@@ -0,0 +1,108 @@
1
+ const BUILTIN_GROUPS = {
2
+ "group:fs": ["read", "write", "edit", "glob", "grep"],
3
+ "group:runtime": ["bash", "git"],
4
+ "group:web": ["web_fetch", "web_search", "browser"],
5
+ "group:code": ["lsp"],
6
+ "group:memory": ["memory_search", "memory_get", "memory_store"],
7
+ };
8
+ export class ToolRegistry {
9
+ tools = new Map();
10
+ policies = [];
11
+ groups = new Map(Object.entries(BUILTIN_GROUPS));
12
+ logger;
13
+ constructor(logger) {
14
+ this.logger = logger;
15
+ }
16
+ register(tool) {
17
+ this.tools.set(tool.name, tool);
18
+ this.logger.debug("Registered tool", { name: tool.name, source: tool.source });
19
+ }
20
+ unregister(name) {
21
+ this.tools.delete(name);
22
+ }
23
+ get(name) {
24
+ return this.tools.get(name);
25
+ }
26
+ getAll() {
27
+ return [...this.tools.values()];
28
+ }
29
+ getForAgent(allowedTools) {
30
+ const all = this.getAll();
31
+ if (!allowedTools)
32
+ return all;
33
+ const resolved = new Set();
34
+ for (const name of allowedTools) {
35
+ const group = this.groups.get(name);
36
+ if (group) {
37
+ group.forEach((t) => resolved.add(t));
38
+ }
39
+ else {
40
+ resolved.add(name);
41
+ }
42
+ }
43
+ return all.filter((t) => resolved.has(t.name));
44
+ }
45
+ async execute(name, params, context) {
46
+ const tool = this.tools.get(name);
47
+ if (!tool) {
48
+ return { output: "", error: `Unknown tool: ${name}` };
49
+ }
50
+ if (!this.checkPolicy(name, context)) {
51
+ return { output: "", error: `Tool ${name} denied by policy` };
52
+ }
53
+ if (params == null || typeof params !== "object" || Array.isArray(params)) {
54
+ return { output: "", error: `Tool ${name} requires an object parameter, got ${typeof params}` };
55
+ }
56
+ const start = Date.now();
57
+ try {
58
+ const result = await tool.execute(params, context);
59
+ this.logger.debug("Tool executed", { name, durationMs: Date.now() - start });
60
+ return result;
61
+ }
62
+ catch (err) {
63
+ this.logger.error("Tool execution failed", { name, error: String(err) });
64
+ return { output: "", error: String(err) };
65
+ }
66
+ }
67
+ registerGroup(name, toolNames) {
68
+ this.groups.set(name, toolNames);
69
+ }
70
+ resolveGroup(groupName) {
71
+ return this.groups.get(groupName) ?? [];
72
+ }
73
+ addPolicy(policy) {
74
+ this.policies.push(policy);
75
+ }
76
+ checkPolicy(toolName, _context) {
77
+ for (const policy of this.policies) {
78
+ const denied = this.resolveNames(policy.deny ?? []);
79
+ if (denied.has(toolName))
80
+ return false;
81
+ if (policy.allow) {
82
+ const allowed = this.resolveNames(policy.allow);
83
+ if (!allowed.has(toolName))
84
+ return false;
85
+ }
86
+ }
87
+ return true;
88
+ }
89
+ resolveNames(names) {
90
+ const resolved = new Set();
91
+ for (const name of names) {
92
+ const group = this.groups.get(name);
93
+ if (group)
94
+ group.forEach((t) => resolved.add(t));
95
+ else
96
+ resolved.add(name);
97
+ }
98
+ return resolved;
99
+ }
100
+ toSchemas() {
101
+ return this.getAll().map((t) => ({
102
+ name: t.name,
103
+ description: t.description,
104
+ parameters: t.parameters,
105
+ }));
106
+ }
107
+ }
108
+ //# sourceMappingURL=tool-registry.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tool-registry.js","sourceRoot":"","sources":["../src/tool-registry.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import type { ToolDefinition } from "../types.js";
2
+ export declare const bashTool: ToolDefinition;
3
+ //# sourceMappingURL=bash.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"bash.d.ts","sourceRoot":"","sources":["../../src/tools/bash.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,cAAc,EAA2B,MAAM,aAAa,CAAC;AAwB3E,eAAO,MAAM,QAAQ,EAAE,cAmDtB,CAAC"}
@@ -0,0 +1,67 @@
1
+ // CRC: crc-ToolRegistry.md
2
+ // Built-in tool: bash — from HomarUS
3
+ import { exec } from "node:child_process";
4
+ const MAX_OUTPUT = 50_000;
5
+ const BLOCKED_PATTERNS = [
6
+ { pattern: /\brm\s+(-[a-zA-Z]*f|-[a-zA-Z]*r|--force|--recursive).*\//, label: "rm -rf /" },
7
+ { pattern: /\bsudo\b/, label: "sudo" },
8
+ { pattern: /\bmkfs\b/, label: "mkfs" },
9
+ { pattern: /\bdd\s+.*of=\/dev\//, label: "dd to device" },
10
+ { pattern: />\s*\/dev\/sd/, label: "redirect to disk device" },
11
+ { pattern: /\bchmod\s+777\b/, label: "chmod 777" },
12
+ { pattern: /\bcurl\b.*\|\s*(ba)?sh/, label: "curl pipe to shell" },
13
+ { pattern: /\bwget\b.*\|\s*(ba)?sh/, label: "wget pipe to shell" },
14
+ { pattern: /\b(shutdown|reboot|halt|poweroff)\b/, label: "system control" },
15
+ { pattern: /\bkillall\b/, label: "killall" },
16
+ { pattern: /:\(\)\s*\{\s*:\|:&\s*\}\s*;:/, label: "fork bomb" },
17
+ ];
18
+ export const bashTool = {
19
+ name: "bash",
20
+ description: "Execute a bash command and return stdout/stderr.",
21
+ parameters: {
22
+ type: "object",
23
+ properties: {
24
+ command: { type: "string", description: "The bash command to execute" },
25
+ timeout: { type: "number", description: "Timeout in milliseconds (default 120000)" },
26
+ workingDir: { type: "string", description: "Working directory for the command" },
27
+ },
28
+ required: ["command"],
29
+ },
30
+ source: "builtin",
31
+ async execute(params, context) {
32
+ const { command, timeout = 120_000, workingDir } = params;
33
+ for (const { pattern, label } of BLOCKED_PATTERNS) {
34
+ if (pattern.test(command)) {
35
+ return { output: "", error: `Blocked: command matches dangerous pattern (${label}).` };
36
+ }
37
+ }
38
+ if (context.sandbox) {
39
+ return { output: "", error: "bash tool is not available in sandbox mode" };
40
+ }
41
+ return new Promise((resolve) => {
42
+ exec(command, {
43
+ cwd: workingDir ?? context.workingDir,
44
+ timeout,
45
+ maxBuffer: 10 * 1024 * 1024,
46
+ env: { ...process.env, TERM: "dumb" },
47
+ }, (error, stdout, stderr) => {
48
+ let output = stdout;
49
+ if (stderr)
50
+ output += (output ? "\n" : "") + stderr;
51
+ if (output.length > MAX_OUTPUT) {
52
+ output = output.slice(0, MAX_OUTPUT) + `\n... (truncated, ${output.length} total chars)`;
53
+ }
54
+ if (error && error.killed) {
55
+ resolve({ output, error: `Command timed out after ${timeout}ms` });
56
+ }
57
+ else if (error) {
58
+ resolve({ output, error: `Exit code ${error.code}: ${error.message}` });
59
+ }
60
+ else {
61
+ resolve({ output });
62
+ }
63
+ });
64
+ });
65
+ },
66
+ };
67
+ //# sourceMappingURL=bash.js.map