myaiforone 1.0.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 (315) hide show
  1. package/README.md +113 -0
  2. package/agents/_template/CLAUDE.md +18 -0
  3. package/agents/_template/agent.json +7 -0
  4. package/agents/platform/agentcreator/CLAUDE.md +300 -0
  5. package/agents/platform/appcreator/CLAUDE.md +158 -0
  6. package/agents/platform/gym/CLAUDE.md +486 -0
  7. package/agents/platform/gym/agent.json +40 -0
  8. package/agents/platform/gym/programs/agent-building/program.json +160 -0
  9. package/agents/platform/gym/programs/automations-mastery/program.json +129 -0
  10. package/agents/platform/gym/programs/getting-started/program.json +124 -0
  11. package/agents/platform/gym/programs/mcp-integrations/program.json +116 -0
  12. package/agents/platform/gym/programs/multi-model-strategy/program.json +115 -0
  13. package/agents/platform/gym/programs/prompt-engineering/program.json +136 -0
  14. package/agents/platform/gym/souls/alex.md +12 -0
  15. package/agents/platform/gym/souls/jordan.md +12 -0
  16. package/agents/platform/gym/souls/morgan.md +12 -0
  17. package/agents/platform/gym/souls/riley.md +12 -0
  18. package/agents/platform/gym/souls/sam.md +12 -0
  19. package/agents/platform/hub/CLAUDE.md +372 -0
  20. package/agents/platform/promptcreator/CLAUDE.md +130 -0
  21. package/agents/platform/skillcreator/CLAUDE.md +163 -0
  22. package/bin/cli.js +566 -0
  23. package/config.example.json +310 -0
  24. package/dist/agent-registry.d.ts +32 -0
  25. package/dist/agent-registry.d.ts.map +1 -0
  26. package/dist/agent-registry.js +144 -0
  27. package/dist/agent-registry.js.map +1 -0
  28. package/dist/channels/discord.d.ts +17 -0
  29. package/dist/channels/discord.d.ts.map +1 -0
  30. package/dist/channels/discord.js +114 -0
  31. package/dist/channels/discord.js.map +1 -0
  32. package/dist/channels/imessage.d.ts +23 -0
  33. package/dist/channels/imessage.d.ts.map +1 -0
  34. package/dist/channels/imessage.js +214 -0
  35. package/dist/channels/imessage.js.map +1 -0
  36. package/dist/channels/slack.d.ts +19 -0
  37. package/dist/channels/slack.d.ts.map +1 -0
  38. package/dist/channels/slack.js +167 -0
  39. package/dist/channels/slack.js.map +1 -0
  40. package/dist/channels/telegram.d.ts +19 -0
  41. package/dist/channels/telegram.d.ts.map +1 -0
  42. package/dist/channels/telegram.js +274 -0
  43. package/dist/channels/telegram.js.map +1 -0
  44. package/dist/channels/types.d.ts +44 -0
  45. package/dist/channels/types.d.ts.map +1 -0
  46. package/dist/channels/types.js +18 -0
  47. package/dist/channels/types.js.map +1 -0
  48. package/dist/channels/whatsapp.d.ts +23 -0
  49. package/dist/channels/whatsapp.d.ts.map +1 -0
  50. package/dist/channels/whatsapp.js +189 -0
  51. package/dist/channels/whatsapp.js.map +1 -0
  52. package/dist/config.d.ts +134 -0
  53. package/dist/config.d.ts.map +1 -0
  54. package/dist/config.js +127 -0
  55. package/dist/config.js.map +1 -0
  56. package/dist/cron.d.ts +8 -0
  57. package/dist/cron.d.ts.map +1 -0
  58. package/dist/cron.js +35 -0
  59. package/dist/cron.js.map +1 -0
  60. package/dist/decrypt-keys.d.ts +7 -0
  61. package/dist/decrypt-keys.d.ts.map +1 -0
  62. package/dist/decrypt-keys.js +53 -0
  63. package/dist/decrypt-keys.js.map +1 -0
  64. package/dist/encrypt-keys.d.ts +8 -0
  65. package/dist/encrypt-keys.d.ts.map +1 -0
  66. package/dist/encrypt-keys.js +62 -0
  67. package/dist/encrypt-keys.js.map +1 -0
  68. package/dist/executor.d.ts +31 -0
  69. package/dist/executor.d.ts.map +1 -0
  70. package/dist/executor.js +2009 -0
  71. package/dist/executor.js.map +1 -0
  72. package/dist/gemini-executor.d.ts +27 -0
  73. package/dist/gemini-executor.d.ts.map +1 -0
  74. package/dist/gemini-executor.js +160 -0
  75. package/dist/gemini-executor.js.map +1 -0
  76. package/dist/goals.d.ts +24 -0
  77. package/dist/goals.d.ts.map +1 -0
  78. package/dist/goals.js +189 -0
  79. package/dist/goals.js.map +1 -0
  80. package/dist/gym/activity-digest.d.ts +30 -0
  81. package/dist/gym/activity-digest.d.ts.map +1 -0
  82. package/dist/gym/activity-digest.js +506 -0
  83. package/dist/gym/activity-digest.js.map +1 -0
  84. package/dist/gym/dimension-scorer.d.ts +76 -0
  85. package/dist/gym/dimension-scorer.d.ts.map +1 -0
  86. package/dist/gym/dimension-scorer.js +236 -0
  87. package/dist/gym/dimension-scorer.js.map +1 -0
  88. package/dist/gym/gym-router.d.ts +7 -0
  89. package/dist/gym/gym-router.d.ts.map +1 -0
  90. package/dist/gym/gym-router.js +718 -0
  91. package/dist/gym/gym-router.js.map +1 -0
  92. package/dist/gym/index.d.ts +11 -0
  93. package/dist/gym/index.d.ts.map +1 -0
  94. package/dist/gym/index.js +11 -0
  95. package/dist/gym/index.js.map +1 -0
  96. package/dist/heartbeat.d.ts +21 -0
  97. package/dist/heartbeat.d.ts.map +1 -0
  98. package/dist/heartbeat.js +163 -0
  99. package/dist/heartbeat.js.map +1 -0
  100. package/dist/index.d.ts +2 -0
  101. package/dist/index.d.ts.map +1 -0
  102. package/dist/index.js +254 -0
  103. package/dist/index.js.map +1 -0
  104. package/dist/keystore.d.ts +22 -0
  105. package/dist/keystore.d.ts.map +1 -0
  106. package/dist/keystore.js +178 -0
  107. package/dist/keystore.js.map +1 -0
  108. package/dist/logger.d.ts +9 -0
  109. package/dist/logger.d.ts.map +1 -0
  110. package/dist/logger.js +45 -0
  111. package/dist/logger.js.map +1 -0
  112. package/dist/memory/daily.d.ts +22 -0
  113. package/dist/memory/daily.d.ts.map +1 -0
  114. package/dist/memory/daily.js +82 -0
  115. package/dist/memory/daily.js.map +1 -0
  116. package/dist/memory/embeddings.d.ts +15 -0
  117. package/dist/memory/embeddings.d.ts.map +1 -0
  118. package/dist/memory/embeddings.js +154 -0
  119. package/dist/memory/embeddings.js.map +1 -0
  120. package/dist/memory/index.d.ts +32 -0
  121. package/dist/memory/index.d.ts.map +1 -0
  122. package/dist/memory/index.js +159 -0
  123. package/dist/memory/index.js.map +1 -0
  124. package/dist/memory/search.d.ts +21 -0
  125. package/dist/memory/search.d.ts.map +1 -0
  126. package/dist/memory/search.js +77 -0
  127. package/dist/memory/search.js.map +1 -0
  128. package/dist/memory/store.d.ts +23 -0
  129. package/dist/memory/store.d.ts.map +1 -0
  130. package/dist/memory/store.js +144 -0
  131. package/dist/memory/store.js.map +1 -0
  132. package/dist/ollama-executor.d.ts +17 -0
  133. package/dist/ollama-executor.d.ts.map +1 -0
  134. package/dist/ollama-executor.js +112 -0
  135. package/dist/ollama-executor.js.map +1 -0
  136. package/dist/openai-executor.d.ts +38 -0
  137. package/dist/openai-executor.d.ts.map +1 -0
  138. package/dist/openai-executor.js +197 -0
  139. package/dist/openai-executor.js.map +1 -0
  140. package/dist/router.d.ts +11 -0
  141. package/dist/router.d.ts.map +1 -0
  142. package/dist/router.js +185 -0
  143. package/dist/router.js.map +1 -0
  144. package/dist/test-message.d.ts +2 -0
  145. package/dist/test-message.d.ts.map +1 -0
  146. package/dist/test-message.js +60 -0
  147. package/dist/test-message.js.map +1 -0
  148. package/dist/utils/imsg-db-reader.d.ts +24 -0
  149. package/dist/utils/imsg-db-reader.d.ts.map +1 -0
  150. package/dist/utils/imsg-db-reader.js +92 -0
  151. package/dist/utils/imsg-db-reader.js.map +1 -0
  152. package/dist/utils/imsg-rpc.d.ts +25 -0
  153. package/dist/utils/imsg-rpc.d.ts.map +1 -0
  154. package/dist/utils/imsg-rpc.js +149 -0
  155. package/dist/utils/imsg-rpc.js.map +1 -0
  156. package/dist/utils/message-formatter.d.ts +3 -0
  157. package/dist/utils/message-formatter.d.ts.map +1 -0
  158. package/dist/utils/message-formatter.js +69 -0
  159. package/dist/utils/message-formatter.js.map +1 -0
  160. package/dist/web-ui.d.ts +12 -0
  161. package/dist/web-ui.d.ts.map +1 -0
  162. package/dist/web-ui.js +5784 -0
  163. package/dist/web-ui.js.map +1 -0
  164. package/dist/whatsapp-chats.d.ts +2 -0
  165. package/dist/whatsapp-chats.d.ts.map +1 -0
  166. package/dist/whatsapp-chats.js +76 -0
  167. package/dist/whatsapp-chats.js.map +1 -0
  168. package/dist/whatsapp-login.d.ts +2 -0
  169. package/dist/whatsapp-login.d.ts.map +1 -0
  170. package/dist/whatsapp-login.js +90 -0
  171. package/dist/whatsapp-login.js.map +1 -0
  172. package/dist/wiki-sync.d.ts +21 -0
  173. package/dist/wiki-sync.d.ts.map +1 -0
  174. package/dist/wiki-sync.js +147 -0
  175. package/dist/wiki-sync.js.map +1 -0
  176. package/docs/AddNewAgentGuide.md +100 -0
  177. package/docs/AddNewMcpGuide.md +72 -0
  178. package/docs/Architecture.md +795 -0
  179. package/docs/CLAUDE-AI-SETUP.md +166 -0
  180. package/docs/Setup.md +297 -0
  181. package/docs/ai-gym-architecture.md +1040 -0
  182. package/docs/ai-gym-build-plan.md +343 -0
  183. package/docs/ai-gym-onboarding.md +122 -0
  184. package/docs/appcreator_plan.md +348 -0
  185. package/docs/platform-mcp-audit.md +320 -0
  186. package/docs/server-deployment-plan.md +503 -0
  187. package/docs/superpowers/plans/2026-03-25-marketplace.md +1281 -0
  188. package/docs/superpowers/specs/2026-03-25-marketplace-design.md +287 -0
  189. package/docs/user-guide.md +2016 -0
  190. package/mcp-catalog.json +628 -0
  191. package/package.json +63 -0
  192. package/public/MyAIforOne-logomark-512.svg +16 -0
  193. package/public/MyAIforOne-logomark-transparent.svg +15 -0
  194. package/public/activity.html +314 -0
  195. package/public/admin.html +1674 -0
  196. package/public/agent-dashboard.html +670 -0
  197. package/public/api-docs.html +1106 -0
  198. package/public/automations.html +722 -0
  199. package/public/canvas.css +223 -0
  200. package/public/canvas.js +588 -0
  201. package/public/changelog.html +231 -0
  202. package/public/gym.html +2766 -0
  203. package/public/home.html +1930 -0
  204. package/public/index.html +2809 -0
  205. package/public/lab.html +1643 -0
  206. package/public/library.html +1442 -0
  207. package/public/marketplace.html +1101 -0
  208. package/public/mcp-docs.html +441 -0
  209. package/public/mini.html +390 -0
  210. package/public/monitor.html +584 -0
  211. package/public/org.html +4304 -0
  212. package/public/projects.html +734 -0
  213. package/public/settings.html +645 -0
  214. package/public/tasks.html +932 -0
  215. package/public/trainers/alex.svg +12 -0
  216. package/public/trainers/jordan.svg +12 -0
  217. package/public/trainers/morgan.svg +12 -0
  218. package/public/trainers/riley.svg +12 -0
  219. package/public/trainers/sam.svg +12 -0
  220. package/public/user-guide.html +218 -0
  221. package/registry/agents.json +3 -0
  222. package/registry/apps.json +20 -0
  223. package/registry/installed-drafts.json +3 -0
  224. package/registry/mcps.json +1084 -0
  225. package/registry/prompts/personal/mcp-test-prompt.md +6 -0
  226. package/registry/prompts/personal/memory-recall.md +6 -0
  227. package/registry/prompts/platform/brainstorm.md +15 -0
  228. package/registry/prompts/platform/code-review.md +16 -0
  229. package/registry/prompts/platform/explain.md +16 -0
  230. package/registry/prompts.json +58 -0
  231. package/registry/skills/external/brainstorming.md +5 -0
  232. package/registry/skills/external/code-review.md +40 -0
  233. package/registry/skills/external/frontend-patterns.md +642 -0
  234. package/registry/skills/external/frontend-slides.md +184 -0
  235. package/registry/skills/external/systematic-debugging.md +5 -0
  236. package/registry/skills/external/tdd.md +328 -0
  237. package/registry/skills/external/verification-before-completion.md +5 -0
  238. package/registry/skills/external/writing-plans.md +5 -0
  239. package/registry/skills/platform/ai41_app_build.md +930 -0
  240. package/registry/skills/platform/ai41_app_deploy.md +168 -0
  241. package/registry/skills/platform/ai41_app_orchestrator.md +239 -0
  242. package/registry/skills/platform/ai41_app_patterns.md +359 -0
  243. package/registry/skills/platform/ai41_app_register.md +85 -0
  244. package/registry/skills/platform/ai41_app_scaffold.md +421 -0
  245. package/registry/skills/platform/ai41_app_verify.md +107 -0
  246. package/registry/skills/platform/opProjectCreate.md +239 -0
  247. package/registry/skills/platform/op_devbrowser.md +136 -0
  248. package/registry/skills/platform/sop_brandguidelines.md +103 -0
  249. package/registry/skills/platform/sop_docx.md +117 -0
  250. package/registry/skills/platform/sop_frontenddesign.md +44 -0
  251. package/registry/skills/platform/sop_frontenddesign_v2.md +659 -0
  252. package/registry/skills/platform/sop_mcpbuilder.md +133 -0
  253. package/registry/skills/platform/sop_pdf.md +172 -0
  254. package/registry/skills/platform/sop_pptx.md +133 -0
  255. package/registry/skills/platform/sop_skillcreator.md +104 -0
  256. package/registry/skills/platform/sop_themefactory.md +128 -0
  257. package/registry/skills/platform/sop_webapptesting.md +75 -0
  258. package/registry/skills/platform/sop_webartifactsbuilder.md +97 -0
  259. package/registry/skills/platform/sop_xlsx.md +134 -0
  260. package/registry/skills.json +1055 -0
  261. package/scripts/discover-chats.sh +11 -0
  262. package/scripts/install-service-windows.ps1 +87 -0
  263. package/scripts/install-service.sh +52 -0
  264. package/scripts/seed-registry.ts +195 -0
  265. package/scripts/test-send.sh +5 -0
  266. package/scripts/tray-indicator.ps1 +35 -0
  267. package/scripts/uninstall-service-windows.ps1 +23 -0
  268. package/scripts/uninstall-service.sh +15 -0
  269. package/scripts/xbar-myagent.5s.sh +32 -0
  270. package/server/mcp-server/dist/index.d.ts +11 -0
  271. package/server/mcp-server/dist/index.js +1332 -0
  272. package/server/mcp-server/dist/lib/api-client.d.ts +165 -0
  273. package/server/mcp-server/dist/lib/api-client.js +241 -0
  274. package/server/mcp-server/index.ts +1545 -0
  275. package/server/mcp-server/lib/api-client.ts +366 -0
  276. package/server/mcp-server/tsconfig.json +14 -0
  277. package/src/agent-registry.ts +180 -0
  278. package/src/channels/discord.ts +129 -0
  279. package/src/channels/imessage.ts +261 -0
  280. package/src/channels/slack.ts +208 -0
  281. package/src/channels/telegram.ts +307 -0
  282. package/src/channels/types.ts +62 -0
  283. package/src/channels/whatsapp.ts +227 -0
  284. package/src/config.ts +281 -0
  285. package/src/cron.ts +43 -0
  286. package/src/decrypt-keys.ts +60 -0
  287. package/src/encrypt-keys.ts +70 -0
  288. package/src/executor.ts +2190 -0
  289. package/src/gemini-executor.ts +212 -0
  290. package/src/goals.ts +240 -0
  291. package/src/gym/activity-digest.ts +546 -0
  292. package/src/gym/dimension-scorer.ts +297 -0
  293. package/src/gym/gym-router.ts +801 -0
  294. package/src/gym/index.ts +19 -0
  295. package/src/heartbeat.ts +220 -0
  296. package/src/index.ts +275 -0
  297. package/src/keystore.ts +190 -0
  298. package/src/logger.ts +51 -0
  299. package/src/memory/daily.ts +101 -0
  300. package/src/memory/embeddings.ts +185 -0
  301. package/src/memory/index.ts +218 -0
  302. package/src/memory/search.ts +124 -0
  303. package/src/memory/store.ts +189 -0
  304. package/src/ollama-executor.ts +126 -0
  305. package/src/openai-executor.ts +259 -0
  306. package/src/router.ts +230 -0
  307. package/src/test-message.ts +72 -0
  308. package/src/utils/imsg-db-reader.ts +109 -0
  309. package/src/utils/imsg-rpc.ts +178 -0
  310. package/src/utils/message-formatter.ts +90 -0
  311. package/src/web-ui.ts +5778 -0
  312. package/src/whatsapp-chats.ts +91 -0
  313. package/src/whatsapp-login.ts +110 -0
  314. package/src/wiki-sync.ts +199 -0
  315. package/tsconfig.json +19 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"imessage.d.ts","sourceRoot":"","sources":["../../src/channels/imessage.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,aAAa,EAAE,cAAc,EAAE,eAAe,EAAE,MAAM,YAAY,CAAC;AA4BjF,qBAAa,cAAe,YAAW,aAAa;IAClD,QAAQ,CAAC,SAAS,cAAc;IAEhC,OAAO,CAAC,MAAM,CAAgB;IAC9B,OAAO,CAAC,cAAc,CAAyD;IAC/E,OAAO,CAAC,WAAW,CAAoC;IACvD,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,cAAc,CAA+C;IACrE,OAAO,CAAC,YAAY,CAA6B;IACjD,OAAO,CAAC,gBAAgB,CAAgB;IACxC,OAAO,CAAC,YAAY,CAA6B;gBAErC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC;IAOrC,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAoBtB,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAc3B,OAAO,CAAC,cAAc;IActB,OAAO,CAAC,MAAM;IAkEd,SAAS,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,cAAc,KAAK,OAAO,CAAC,IAAI,CAAC,GAAG,IAAI;IAI1D,IAAI,CAAC,GAAG,EAAE,eAAe,GAAG,OAAO,CAAC,IAAI,CAAC;IAkB/C,OAAO,CAAC,gBAAgB;IAsBxB,OAAO,CAAC,aAAa;IAwCrB,OAAO,CAAC,uBAAuB;CAehC"}
@@ -0,0 +1,214 @@
1
+ import { splitText } from "./types.js";
2
+ import { ImsgRpcClient } from "../utils/imsg-rpc.js";
3
+ import { getRecentMessages, getLatestRowId } from "../utils/imsg-db-reader.js";
4
+ import { log } from "../logger.js";
5
+ export class IMessageDriver {
6
+ channelId = "imessage";
7
+ client;
8
+ messageHandler = null;
9
+ debounceMap = new Map();
10
+ debounceMs;
11
+ dbPollInterval = null;
12
+ dbPollRowIds = new Map(); // chatId -> lastRowId
13
+ monitoredChatIds = [];
14
+ recentlySent = new Map(); // text -> timestamp (for echo filtering)
15
+ constructor(config) {
16
+ const cliPath = config.cliPath ?? "imsg";
17
+ this.debounceMs = config.debounceMs ?? 2000;
18
+ this.monitoredChatIds = (config.monitoredChatIds ?? []);
19
+ this.client = new ImsgRpcClient(cliPath);
20
+ }
21
+ async start() {
22
+ await this.client.start();
23
+ this.client.onNotification((notification) => {
24
+ if (notification.method === "message") {
25
+ const raw = notification.params.message;
26
+ this.handleRawMessage(raw);
27
+ }
28
+ });
29
+ // Subscribe to message watch (attachments enabled for image support)
30
+ await this.client.request("watch.subscribe", { attachments: true });
31
+ // DB polling fallback — macOS 15+ stores text in attributedBody, not text column
32
+ // imsg watch may miss these messages, so we poll the DB directly
33
+ this.startDbPolling();
34
+ log.info("iMessage driver started — watching for messages + DB polling");
35
+ }
36
+ async stop() {
37
+ if (this.dbPollInterval) {
38
+ clearInterval(this.dbPollInterval);
39
+ this.dbPollInterval = null;
40
+ }
41
+ for (const [, entry] of this.debounceMap) {
42
+ clearTimeout(entry.timer);
43
+ }
44
+ this.debounceMap.clear();
45
+ await this.client.stop();
46
+ log.info("iMessage driver stopped");
47
+ }
48
+ startDbPolling() {
49
+ // Discover chat IDs to monitor from config routes
50
+ // If monitoredChatIds not set, we'll pick them up dynamically
51
+ // Initialize last-seen rowid for each chat
52
+ for (const chatId of this.monitoredChatIds) {
53
+ this.dbPollRowIds.set(chatId, getLatestRowId(chatId));
54
+ }
55
+ // Poll every 3 seconds
56
+ this.dbPollInterval = setInterval(() => {
57
+ this.pollDb();
58
+ }, 3000);
59
+ }
60
+ pollDb() {
61
+ // Also discover chat IDs dynamically from recent messages if not configured
62
+ if (this.monitoredChatIds.length === 0)
63
+ return;
64
+ for (const chatId of this.monitoredChatIds) {
65
+ const lastRowId = this.dbPollRowIds.get(chatId) || 0;
66
+ const messages = getRecentMessages(chatId, lastRowId);
67
+ for (const msg of messages) {
68
+ // Update last seen
69
+ if (msg.rowid > (this.dbPollRowIds.get(chatId) || 0)) {
70
+ this.dbPollRowIds.set(chatId, msg.rowid);
71
+ }
72
+ // Skip empty
73
+ if (!msg.text.trim())
74
+ continue;
75
+ // Skip bot's own messages — check against recently sent texts
76
+ // Use fuzzy match: attributedBody extraction can prepend/append garbage bytes
77
+ const msgTrimmed = msg.text.trim();
78
+ let isEcho = false;
79
+ for (const [sentText, ts] of this.recentlySent) {
80
+ // Match if either contains the other, or first 30 chars match
81
+ if (sentText === msgTrimmed ||
82
+ sentText.includes(msgTrimmed) ||
83
+ msgTrimmed.includes(sentText) ||
84
+ (msgTrimmed.length > 30 && sentText.startsWith(msgTrimmed.slice(0, 30))) ||
85
+ (msgTrimmed.length > 30 && sentText.includes(msgTrimmed.slice(1, 31)))) {
86
+ this.recentlySent.delete(sentText);
87
+ isEcho = true;
88
+ break;
89
+ }
90
+ }
91
+ if (isEcho) {
92
+ log.debug(`iMessage DB poll: skipping echo "${msg.text.slice(0, 40)}..."`);
93
+ continue;
94
+ }
95
+ // Build an InboundMessage and pass to handler
96
+ const inbound = {
97
+ id: String(msg.rowid),
98
+ channel: this.channelId,
99
+ chatId: String(chatId),
100
+ chatType: "group",
101
+ sender: msg.sender || "unknown",
102
+ text: msg.text,
103
+ timestamp: new Date(msg.createdAt).getTime(),
104
+ // Always false — user's phone shares Apple ID with this Mac,
105
+ // so all messages appear as is_from_me in the DB.
106
+ // Echo prevention is handled in index.ts via recentBotMessages.
107
+ isFromMe: false,
108
+ isGroup: true,
109
+ raw: msg,
110
+ };
111
+ log.debug(`iMessage DB poll: ${msg.sender} in chat ${chatId}: ${msg.text.slice(0, 80)}`);
112
+ if (this.messageHandler) {
113
+ this.messageHandler(inbound).catch((err) => {
114
+ log.error(`iMessage DB poll handler error: ${err}`);
115
+ });
116
+ }
117
+ }
118
+ }
119
+ }
120
+ onMessage(handler) {
121
+ this.messageHandler = handler;
122
+ }
123
+ async send(msg) {
124
+ // iMessage doesn't have a strict char limit but long messages are unwieldy on phones
125
+ const chunks = splitText(msg.text, 3000);
126
+ for (const chunk of chunks) {
127
+ // Track what we send so DB poller can filter echoes
128
+ this.recentlySent.set(chunk.trim(), Date.now());
129
+ await this.client.request("send", {
130
+ text: chunk,
131
+ chat_id: Number(msg.chatId),
132
+ }, 30_000);
133
+ }
134
+ // Prune old entries (>60s) to prevent memory leak
135
+ const cutoff = Date.now() - 60_000;
136
+ for (const [text, ts] of this.recentlySent) {
137
+ if (ts < cutoff)
138
+ this.recentlySent.delete(text);
139
+ }
140
+ }
141
+ handleRawMessage(raw) {
142
+ // Note: can't filter is_from_me here because the user's phone shares
143
+ // the same Apple ID as this Mac, so all messages appear as is_from_me.
144
+ // Echo prevention is handled in index.ts via recentBotMessages tracking.
145
+ if (!raw.text?.trim() && !raw.attachments?.length)
146
+ return;
147
+ // Debounce: coalesce rapid messages from same sender in same chat
148
+ const key = `${raw.chat_id}:${raw.sender}`;
149
+ const existing = this.debounceMap.get(key);
150
+ if (existing) {
151
+ clearTimeout(existing.timer);
152
+ existing.messages.push(raw);
153
+ existing.timer = setTimeout(() => this.flushDebounce(key), this.debounceMs);
154
+ }
155
+ else {
156
+ this.debounceMap.set(key, {
157
+ messages: [raw],
158
+ timer: setTimeout(() => this.flushDebounce(key), this.debounceMs),
159
+ });
160
+ }
161
+ }
162
+ flushDebounce(key) {
163
+ const entry = this.debounceMap.get(key);
164
+ if (!entry)
165
+ return;
166
+ this.debounceMap.delete(key);
167
+ // Coalesce multiple messages into one
168
+ const messages = entry.messages;
169
+ const first = messages[0];
170
+ const text = messages.map((m) => m.text).join("\n");
171
+ const inbound = {
172
+ id: String(first.id),
173
+ channel: this.channelId,
174
+ chatId: String(first.chat_id),
175
+ chatType: first.is_group ? "group" : "dm",
176
+ sender: first.sender,
177
+ text,
178
+ timestamp: new Date(first.created_at).getTime(),
179
+ isFromMe: first.is_from_me,
180
+ isGroup: first.is_group,
181
+ groupName: first.chat_name ?? undefined,
182
+ participants: first.participants,
183
+ replyTo: first.reply_to_id
184
+ ? {
185
+ id: String(first.reply_to_id),
186
+ text: first.reply_to_text ?? "",
187
+ sender: first.reply_to_sender ?? undefined,
188
+ }
189
+ : undefined,
190
+ attachments: this.collectImageAttachments(messages),
191
+ raw: messages.length === 1 ? first : messages,
192
+ };
193
+ if (this.messageHandler) {
194
+ this.messageHandler(inbound).catch((err) => {
195
+ log.error(`Message handler error: ${err}`);
196
+ });
197
+ }
198
+ }
199
+ collectImageAttachments(messages) {
200
+ const imageTypes = ["image/png", "image/jpeg", "image/gif", "image/webp", "image/heic"];
201
+ const attachments = [];
202
+ for (const msg of messages) {
203
+ if (!msg.attachments?.length)
204
+ continue;
205
+ for (const a of msg.attachments) {
206
+ if (a.mime_type && imageTypes.includes(a.mime_type)) {
207
+ attachments.push({ path: a.path, mimeType: a.mime_type });
208
+ }
209
+ }
210
+ }
211
+ return attachments.length > 0 ? attachments : undefined;
212
+ }
213
+ }
214
+ //# sourceMappingURL=imessage.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"imessage.js","sourceRoot":"","sources":["../../src/channels/imessage.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,19 @@
1
+ import type { ChannelDriver, InboundMessage, OutboundMessage } from "./types.js";
2
+ export declare class SlackDriver implements ChannelDriver {
3
+ readonly channelId = "slack";
4
+ private web;
5
+ private socket;
6
+ private botToken;
7
+ private messageHandler;
8
+ private botUserId;
9
+ constructor(config: Record<string, unknown>);
10
+ start(): Promise<void>;
11
+ stop(): Promise<void>;
12
+ onMessage(handler: (msg: InboundMessage) => Promise<void>): void;
13
+ send(msg: OutboundMessage): Promise<void>;
14
+ sendTyping(chatId: string): Promise<void>;
15
+ sendFile(chatId: string, filePath: string, caption?: string): Promise<void>;
16
+ private handleEvent;
17
+ private downloadFiles;
18
+ }
19
+ //# sourceMappingURL=slack.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"slack.d.ts","sourceRoot":"","sources":["../../src/channels/slack.ts"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAE,aAAa,EAAE,cAAc,EAAE,eAAe,EAAE,MAAM,YAAY,CAAC;AA0BjF,qBAAa,WAAY,YAAW,aAAa;IAC/C,QAAQ,CAAC,SAAS,WAAW;IAE7B,OAAO,CAAC,GAAG,CAAY;IACvB,OAAO,CAAC,MAAM,CAAmB;IACjC,OAAO,CAAC,QAAQ,CAAS;IACzB,OAAO,CAAC,cAAc,CAAyD;IAC/E,OAAO,CAAC,SAAS,CAAuB;gBAE5B,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC;IAkBrC,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAsBtB,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAK3B,SAAS,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,cAAc,KAAK,OAAO,CAAC,IAAI,CAAC,GAAG,IAAI;IAI1D,IAAI,CAAC,GAAG,EAAE,eAAe,GAAG,OAAO,CAAC,IAAI,CAAC;IAYzC,UAAU,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAQzC,QAAQ,CAAC,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;YAcnE,WAAW;YAkCX,aAAa;CAkD5B"}
@@ -0,0 +1,167 @@
1
+ import { writeFileSync, mkdirSync, readFileSync } from "node:fs";
2
+ import { join, basename } from "node:path";
3
+ import { tmpdir } from "node:os";
4
+ import { WebClient } from "@slack/web-api";
5
+ import { SocketModeClient } from "@slack/socket-mode";
6
+ import { splitText } from "./types.js";
7
+ import { log } from "../logger.js";
8
+ export class SlackDriver {
9
+ channelId = "slack";
10
+ web;
11
+ socket;
12
+ botToken;
13
+ messageHandler = null;
14
+ botUserId = null;
15
+ constructor(config) {
16
+ const botToken = config.botToken;
17
+ const appToken = config.appToken;
18
+ if (!botToken || !appToken) {
19
+ throw new Error("Slack driver requires botToken and appToken in config");
20
+ }
21
+ this.botToken = botToken;
22
+ this.web = new WebClient(botToken);
23
+ this.socket = new SocketModeClient({
24
+ appToken,
25
+ clientPingTimeout: 30_000,
26
+ serverPingTimeout: 30_000,
27
+ pingPongLoggingEnabled: false,
28
+ });
29
+ }
30
+ async start() {
31
+ const auth = await this.web.auth.test();
32
+ this.botUserId = auth.user_id;
33
+ log.info(`Slack bot authenticated as ${auth.user} (${this.botUserId})`);
34
+ this.socket.on("message", async ({ event, ack }) => {
35
+ await ack();
36
+ this.handleEvent(event);
37
+ });
38
+ this.socket.on("connected", () => {
39
+ log.info("Slack socket mode connected");
40
+ });
41
+ this.socket.on("disconnected", () => {
42
+ log.warn("Slack socket mode disconnected");
43
+ });
44
+ await this.socket.start();
45
+ log.info("Slack driver started — listening for messages");
46
+ }
47
+ async stop() {
48
+ await this.socket.disconnect();
49
+ log.info("Slack driver stopped");
50
+ }
51
+ onMessage(handler) {
52
+ this.messageHandler = handler;
53
+ }
54
+ async send(msg) {
55
+ // Slack limit is ~4000 chars (with some overhead for formatting)
56
+ const chunks = splitText(msg.text, 3900);
57
+ for (const chunk of chunks) {
58
+ await this.web.chat.postMessage({
59
+ channel: msg.chatId,
60
+ text: chunk,
61
+ ...(msg.replyToId ? { thread_ts: msg.replyToId } : {}),
62
+ });
63
+ }
64
+ }
65
+ async sendTyping(chatId) {
66
+ try {
67
+ // Slack doesn't have a direct typing indicator API for bots,
68
+ // but we can use a subtle reaction or just skip
69
+ // The "On it..." message in index.ts serves this purpose
70
+ }
71
+ catch { /* ignore */ }
72
+ }
73
+ async sendFile(chatId, filePath, caption) {
74
+ try {
75
+ const content = readFileSync(filePath);
76
+ await this.web.filesUploadV2({
77
+ channel_id: chatId,
78
+ file: content,
79
+ filename: basename(filePath),
80
+ initial_comment: caption || undefined,
81
+ });
82
+ }
83
+ catch (err) {
84
+ log.warn(`Failed to send file to Slack: ${err}`);
85
+ }
86
+ }
87
+ async handleEvent(event) {
88
+ if (event.user === this.botUserId)
89
+ return;
90
+ if (event.subtype && event.subtype !== "file_share")
91
+ return;
92
+ if (!event.text?.trim() && !event.files?.length)
93
+ return;
94
+ const isGroup = event.channel_type === "channel" || event.channel_type === "group";
95
+ const attachments = await this.downloadFiles(event.files);
96
+ const inbound = {
97
+ id: event.ts,
98
+ channel: this.channelId,
99
+ chatId: event.channel,
100
+ chatType: isGroup ? "group" : "dm",
101
+ sender: event.user,
102
+ text: event.text || "",
103
+ timestamp: Math.floor(parseFloat(event.ts) * 1000),
104
+ isFromMe: false,
105
+ isGroup,
106
+ replyTo: event.thread_ts
107
+ ? { id: event.thread_ts, text: "" }
108
+ : undefined,
109
+ attachments: attachments.length > 0 ? attachments : undefined,
110
+ raw: event,
111
+ };
112
+ log.debug(`Slack received from ${event.user} in ${event.channel}: ${(event.text || "").slice(0, 100)}${attachments.length ? ` [${attachments.length} image(s)]` : ""}`);
113
+ if (this.messageHandler) {
114
+ this.messageHandler(inbound).catch((err) => {
115
+ log.error(`Slack message handler error: ${err}`);
116
+ });
117
+ }
118
+ }
119
+ async downloadFiles(files) {
120
+ if (!files?.length)
121
+ return [];
122
+ const imageTypes = ["png", "jpg", "jpeg", "gif", "webp", "heic", "bmp"];
123
+ const downloadDir = join(tmpdir(), "channelToAgent-slack-images");
124
+ mkdirSync(downloadDir, { recursive: true });
125
+ const results = [];
126
+ for (const file of files) {
127
+ try {
128
+ const info = await this.web.files.info({ file: file.id });
129
+ const fullFile = info.file;
130
+ if (!fullFile) {
131
+ log.warn(`Slack files.info returned no file for ${file.id}`);
132
+ continue;
133
+ }
134
+ const mimetype = fullFile.mimetype;
135
+ const filetype = fullFile.filetype;
136
+ const name = fullFile.name || `${file.id}.${filetype || "png"}`;
137
+ const size = fullFile.size;
138
+ const downloadUrl = fullFile.url_private_download;
139
+ if (!mimetype?.startsWith("image/") && !imageTypes.includes(filetype || ""))
140
+ continue;
141
+ if ((size || 0) > 10_000_000)
142
+ continue;
143
+ if (!downloadUrl) {
144
+ log.warn(`No download URL for Slack file ${name}`);
145
+ continue;
146
+ }
147
+ const resp = await fetch(downloadUrl, {
148
+ headers: { Authorization: `Bearer ${this.botToken}` },
149
+ });
150
+ if (!resp.ok) {
151
+ log.warn(`Failed to download Slack file ${name}: ${resp.status}`);
152
+ continue;
153
+ }
154
+ const buffer = Buffer.from(await resp.arrayBuffer());
155
+ const localPath = join(downloadDir, `${file.id}-${name}`);
156
+ writeFileSync(localPath, buffer);
157
+ results.push({ path: localPath, mimeType: mimetype });
158
+ log.debug(`Downloaded Slack file: ${name} (${buffer.length} bytes)`);
159
+ }
160
+ catch (err) {
161
+ log.warn(`Error downloading Slack file ${file.id}: ${err}`);
162
+ }
163
+ }
164
+ return results;
165
+ }
166
+ }
167
+ //# sourceMappingURL=slack.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"slack.js","sourceRoot":"","sources":["../../src/channels/slack.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,19 @@
1
+ import type { ChannelDriver, InboundMessage, OutboundMessage } from "./types.js";
2
+ export declare class TelegramDriver implements ChannelDriver {
3
+ readonly channelId = "telegram";
4
+ private bot;
5
+ private botToken;
6
+ private messageHandler;
7
+ private botId;
8
+ constructor(config: Record<string, unknown>);
9
+ start(): Promise<void>;
10
+ stop(): Promise<void>;
11
+ onMessage(handler: (msg: InboundMessage) => Promise<void>): void;
12
+ send(msg: OutboundMessage): Promise<void>;
13
+ sendTyping(chatId: string): Promise<void>;
14
+ sendFile(chatId: string, filePath: string, caption?: string): Promise<void>;
15
+ private handleMessage;
16
+ private handleVoiceMessage;
17
+ private downloadPhotos;
18
+ }
19
+ //# sourceMappingURL=telegram.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"telegram.d.ts","sourceRoot":"","sources":["../../src/channels/telegram.ts"],"names":[],"mappings":"AAMA,OAAO,KAAK,EAAE,aAAa,EAAE,cAAc,EAAE,eAAe,EAAE,MAAM,YAAY,CAAC;AAKjF,qBAAa,cAAe,YAAW,aAAa;IAClD,QAAQ,CAAC,SAAS,cAAc;IAEhC,OAAO,CAAC,GAAG,CAAM;IACjB,OAAO,CAAC,QAAQ,CAAS;IACzB,OAAO,CAAC,cAAc,CAAyD;IAC/E,OAAO,CAAC,KAAK,CAAuB;gBAExB,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC;IASrC,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IA8BtB,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAK3B,SAAS,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,cAAc,KAAK,OAAO,CAAC,IAAI,CAAC,GAAG,IAAI;IAI1D,IAAI,CAAC,GAAG,EAAE,eAAe,GAAG,OAAO,CAAC,IAAI,CAAC;IAUzC,UAAU,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAMzC,QAAQ,CAAC,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;YAoBnE,aAAa;YAsDb,kBAAkB;YA+DlB,cAAc;CA2C7B"}