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,197 @@
1
+ /**
2
+ * OpenAI-compatible executor — handles OpenAI, xAI (Grok), Groq, Together, Mistral,
3
+ * and any other provider that implements the OpenAI Chat Completions API.
4
+ *
5
+ * Executor format: "openai:gpt-4o", "grok:grok-3", "groq:llama-3.3-70b-versatile",
6
+ * "together:meta-llama/Llama-3.3-70B-Instruct", "mistral:mistral-large-latest"
7
+ */
8
+ import { log } from "./logger.js";
9
+ export const PROVIDERS = {
10
+ openai: {
11
+ name: "OpenAI",
12
+ baseUrl: "https://api.openai.com/v1",
13
+ keyField: "openai",
14
+ },
15
+ grok: {
16
+ name: "xAI (Grok)",
17
+ baseUrl: "https://api.x.ai/v1",
18
+ keyField: "xai",
19
+ },
20
+ groq: {
21
+ name: "Groq",
22
+ baseUrl: "https://api.groq.com/openai/v1",
23
+ keyField: "groq",
24
+ },
25
+ together: {
26
+ name: "Together",
27
+ baseUrl: "https://api.together.xyz/v1",
28
+ keyField: "together",
29
+ },
30
+ mistral: {
31
+ name: "Mistral",
32
+ baseUrl: "https://api.mistral.ai/v1",
33
+ keyField: "mistral",
34
+ },
35
+ };
36
+ // ─── Execute (non-streaming) ────────────────────────────────────
37
+ export async function executeOpenAICompat(opts) {
38
+ const providerConfig = PROVIDERS[opts.provider];
39
+ const baseUrl = opts.baseUrl || providerConfig?.baseUrl;
40
+ if (!baseUrl)
41
+ throw new Error(`Unknown provider: ${opts.provider}`);
42
+ const timeout = opts.timeout || 300_000;
43
+ const providerName = providerConfig?.name || opts.provider;
44
+ log.info(`[${providerName}] Executing ${opts.model}: ${opts.message.slice(0, 200)}`);
45
+ const controller = new AbortController();
46
+ const timer = setTimeout(() => controller.abort(), timeout);
47
+ try {
48
+ const res = await fetch(`${baseUrl}/chat/completions`, {
49
+ method: "POST",
50
+ headers: {
51
+ "Content-Type": "application/json",
52
+ "Authorization": `Bearer ${opts.apiKey}`,
53
+ },
54
+ body: JSON.stringify({
55
+ model: opts.model,
56
+ messages: [
57
+ { role: "system", content: opts.systemPrompt },
58
+ { role: "user", content: opts.message },
59
+ ],
60
+ temperature: opts.temperature ?? 0.7,
61
+ max_tokens: opts.maxTokens ?? 4096,
62
+ stream: false,
63
+ }),
64
+ signal: controller.signal,
65
+ });
66
+ if (!res.ok) {
67
+ const errText = await res.text();
68
+ log.error(`[${providerName}] Error ${res.status}: ${errText}`);
69
+ // Parse structured error if possible
70
+ try {
71
+ const errJson = JSON.parse(errText);
72
+ const errMsg = errJson.error?.message || errJson.message || errText;
73
+ throw new Error(`${providerName} API error ${res.status}: ${errMsg}`);
74
+ }
75
+ catch (e) {
76
+ if (e instanceof Error && e.message.startsWith(providerName))
77
+ throw e;
78
+ throw new Error(`${providerName} API error ${res.status}: ${errText}`);
79
+ }
80
+ }
81
+ const data = await res.json();
82
+ if (data.error) {
83
+ throw new Error(`${providerName} error: ${data.error.message}`);
84
+ }
85
+ const response = data.choices?.[0]?.message?.content || "";
86
+ if (data.usage) {
87
+ log.debug(`[${providerName}] ${opts.model} usage: ${data.usage.prompt_tokens}+${data.usage.completion_tokens} tokens`);
88
+ }
89
+ log.info(`[${providerName}] Response from ${opts.model}: ${response.slice(0, 200)}`);
90
+ return response;
91
+ }
92
+ finally {
93
+ clearTimeout(timer);
94
+ }
95
+ }
96
+ // ─── Stream ─────────────────────────────────────────────────────
97
+ export async function* streamOpenAICompat(opts) {
98
+ const providerConfig = PROVIDERS[opts.provider];
99
+ const baseUrl = opts.baseUrl || providerConfig?.baseUrl;
100
+ if (!baseUrl)
101
+ throw new Error(`Unknown provider: ${opts.provider}`);
102
+ const timeout = opts.timeout || 300_000;
103
+ const providerName = providerConfig?.name || opts.provider;
104
+ log.info(`[${providerName}] Streaming ${opts.model}: ${opts.message.slice(0, 200)}`);
105
+ const controller = new AbortController();
106
+ const timer = setTimeout(() => controller.abort(), timeout);
107
+ try {
108
+ const res = await fetch(`${baseUrl}/chat/completions`, {
109
+ method: "POST",
110
+ headers: {
111
+ "Content-Type": "application/json",
112
+ "Authorization": `Bearer ${opts.apiKey}`,
113
+ },
114
+ body: JSON.stringify({
115
+ model: opts.model,
116
+ messages: [
117
+ { role: "system", content: opts.systemPrompt },
118
+ { role: "user", content: opts.message },
119
+ ],
120
+ temperature: opts.temperature ?? 0.7,
121
+ max_tokens: opts.maxTokens ?? 4096,
122
+ stream: true,
123
+ }),
124
+ signal: controller.signal,
125
+ });
126
+ if (!res.ok) {
127
+ const errText = await res.text();
128
+ throw new Error(`${providerName} API error ${res.status}: ${errText}`);
129
+ }
130
+ const reader = res.body?.getReader();
131
+ if (!reader)
132
+ throw new Error("No response body");
133
+ const decoder = new TextDecoder();
134
+ let buffer = "";
135
+ while (true) {
136
+ const { done, value } = await reader.read();
137
+ if (done)
138
+ break;
139
+ buffer += decoder.decode(value, { stream: true });
140
+ const lines = buffer.split("\n");
141
+ buffer = lines.pop() || ""; // keep incomplete last line
142
+ for (const line of lines) {
143
+ const trimmed = line.trim();
144
+ if (!trimmed || trimmed === "data: [DONE]")
145
+ continue;
146
+ if (!trimmed.startsWith("data: "))
147
+ continue;
148
+ const jsonStr = trimmed.slice(6);
149
+ try {
150
+ const chunk = JSON.parse(jsonStr);
151
+ const content = chunk.choices?.[0]?.delta?.content;
152
+ if (content) {
153
+ yield content;
154
+ }
155
+ }
156
+ catch { /* skip unparseable lines */ }
157
+ }
158
+ }
159
+ }
160
+ finally {
161
+ clearTimeout(timer);
162
+ }
163
+ }
164
+ // ─── Health check ───────────────────────────────────────────────
165
+ export async function checkOpenAICompatHealth(provider, apiKey, baseUrl) {
166
+ const providerConfig = PROVIDERS[provider];
167
+ const url = baseUrl || providerConfig?.baseUrl;
168
+ if (!url)
169
+ return { ok: false, error: `Unknown provider: ${provider}` };
170
+ const providerName = providerConfig?.name || provider;
171
+ try {
172
+ const res = await fetch(`${url}/models`, {
173
+ headers: { "Authorization": `Bearer ${apiKey}` },
174
+ signal: AbortSignal.timeout(10_000),
175
+ });
176
+ if (res.status === 401 || res.status === 403) {
177
+ return { ok: false, error: `${providerName}: Invalid API key` };
178
+ }
179
+ if (!res.ok) {
180
+ return { ok: false, error: `${providerName}: API returned ${res.status}` };
181
+ }
182
+ const data = await res.json();
183
+ const models = data.data?.map(m => m.id) || [];
184
+ return { ok: true, models: models.slice(0, 20) };
185
+ }
186
+ catch (err) {
187
+ return { ok: false, error: `${providerName}: ${err}` };
188
+ }
189
+ }
190
+ /**
191
+ * Resolve provider prefix to config.
192
+ * Returns null if the prefix is not an OpenAI-compatible provider.
193
+ */
194
+ export function resolveProvider(prefix) {
195
+ return PROVIDERS[prefix] || null;
196
+ }
197
+ //# sourceMappingURL=openai-executor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"openai-executor.js","sourceRoot":"","sources":["../src/openai-executor.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,11 @@
1
+ import type { InboundMessage } from "./channels/types.js";
2
+ import type { AgentConfig, RouteConfig, AppConfig } from "./config.js";
3
+ export interface ResolvedRoute {
4
+ agentId: string;
5
+ agentConfig: AgentConfig;
6
+ route: RouteConfig;
7
+ }
8
+ export declare function resolveRoute(msg: InboundMessage, config: AppConfig, baseDir?: string): ResolvedRoute | null;
9
+ export declare function isPairingAttempt(msg: InboundMessage, config: AppConfig, baseDir: string): boolean;
10
+ export declare function pairSender(msg: InboundMessage, baseDir: string): void;
11
+ //# sourceMappingURL=router.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"router.d.ts","sourceRoot":"","sources":["../src/router.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,qBAAqB,CAAC;AAC1D,OAAO,KAAK,EAAE,WAAW,EAAE,WAAW,EAAE,SAAS,EAAE,MAAM,aAAa,CAAC;AAGvE,MAAM,WAAW,aAAa;IAC5B,OAAO,EAAE,MAAM,CAAC;IAChB,WAAW,EAAE,WAAW,CAAC;IACzB,KAAK,EAAE,WAAW,CAAC;CACpB;AAqHD,wBAAgB,YAAY,CAC1B,GAAG,EAAE,cAAc,EACnB,MAAM,EAAE,SAAS,EACjB,OAAO,CAAC,EAAE,MAAM,GACf,aAAa,GAAG,IAAI,CAkEtB;AAGD,wBAAgB,gBAAgB,CAAC,GAAG,EAAE,cAAc,EAAE,MAAM,EAAE,SAAS,EAAE,OAAO,EAAE,MAAM,GAAG,OAAO,CAMjG;AAED,wBAAgB,UAAU,CAAC,GAAG,EAAE,cAAc,EAAE,OAAO,EAAE,MAAM,GAAG,IAAI,CAMrE"}
package/dist/router.js ADDED
@@ -0,0 +1,185 @@
1
+ import { existsSync, readFileSync, writeFileSync, mkdirSync } from "node:fs";
2
+ import { join } from "node:path";
3
+ import { log } from "./logger.js";
4
+ // ─── Pairing store ───────────────────────────────────────────────────
5
+ const pairedSenders = new Set();
6
+ let pairingStoreLoaded = false;
7
+ let pairingStorePath = "";
8
+ function loadPairingStore(baseDir) {
9
+ if (pairingStoreLoaded)
10
+ return;
11
+ pairingStorePath = join(baseDir, "data", "paired-senders.json");
12
+ try {
13
+ if (existsSync(pairingStorePath)) {
14
+ const data = JSON.parse(readFileSync(pairingStorePath, "utf-8"));
15
+ for (const s of data)
16
+ pairedSenders.add(s);
17
+ }
18
+ }
19
+ catch { /* fresh start */ }
20
+ pairingStoreLoaded = true;
21
+ }
22
+ function savePairingStore() {
23
+ try {
24
+ mkdirSync(join(pairingStorePath, ".."), { recursive: true });
25
+ writeFileSync(pairingStorePath, JSON.stringify([...pairedSenders], null, 2));
26
+ }
27
+ catch (err) {
28
+ log.warn(`Failed to save pairing store: ${err}`);
29
+ }
30
+ }
31
+ // Key: "channel:chatId:sender" → last agent they talked to
32
+ const stickyMap = new Map();
33
+ const DEFAULT_STICKY_TIMEOUT_MS = 300_000; // 5 minutes
34
+ function getStickyMode(config, channel) {
35
+ const channelCfg = config.channels[channel];
36
+ if (!channelCfg)
37
+ return { mode: "none", prefix: "!", timeoutMs: DEFAULT_STICKY_TIMEOUT_MS };
38
+ const raw = channelCfg.config.stickyRouting;
39
+ let mode = "prefix"; // default: prefix mode
40
+ if (raw === "sticky")
41
+ mode = "sticky";
42
+ else if (raw === "none")
43
+ mode = "none";
44
+ else if (raw === "prefix")
45
+ mode = "prefix";
46
+ const prefix = channelCfg.config.stickyPrefix ?? "!";
47
+ const timeoutMs = channelCfg.config.stickyTimeoutMs ?? DEFAULT_STICKY_TIMEOUT_MS;
48
+ return { mode, prefix, timeoutMs };
49
+ }
50
+ function getStickyAgent(msg, config) {
51
+ const { mode, prefix, timeoutMs } = getStickyMode(config, msg.channel);
52
+ if (mode === "none")
53
+ return null;
54
+ const key = `${msg.channel}:${msg.chatId}:${msg.sender}`;
55
+ const entry = stickyMap.get(key);
56
+ if (!entry)
57
+ return null;
58
+ // Check if expired
59
+ if (Date.now() - entry.timestamp > timeoutMs) {
60
+ stickyMap.delete(key);
61
+ return null;
62
+ }
63
+ // Prefix mode: message must start with the trigger character
64
+ if (mode === "prefix") {
65
+ const trimmed = msg.text.trim();
66
+ if (!trimmed.startsWith(prefix)) {
67
+ return null; // No prefix — don't route via sticky
68
+ }
69
+ // Strip the prefix from the message text for the agent
70
+ msg.text = trimmed.slice(prefix.length).trim();
71
+ }
72
+ // Verify the agent still exists
73
+ const agent = config.agents[entry.agentId];
74
+ if (!agent) {
75
+ stickyMap.delete(key);
76
+ return null;
77
+ }
78
+ // Find the matching route for this channel + chatId
79
+ const route = agent.routes.find(r => r.channel === msg.channel && String(r.match.value) === msg.chatId);
80
+ if (!route)
81
+ return null;
82
+ // Permission check
83
+ if (!isAllowed(msg, route))
84
+ return null;
85
+ log.debug(`Sticky routing (${mode}): ${msg.sender} → ${entry.agentId} (${Math.round((Date.now() - entry.timestamp) / 1000)}s ago)`);
86
+ return { agentId: entry.agentId, agentConfig: agent, route };
87
+ }
88
+ function setStickyAgent(msg, agentId) {
89
+ const key = `${msg.channel}:${msg.chatId}:${msg.sender}`;
90
+ stickyMap.set(key, { agentId, timestamp: Date.now() });
91
+ }
92
+ // ─── Route resolver ──────────────────────────────────────────────────
93
+ export function resolveRoute(msg, config, baseDir) {
94
+ // Feature 4: DM pairing gate
95
+ if (config.service.pairingCode && baseDir) {
96
+ loadPairingStore(baseDir);
97
+ const senderKey = `${msg.channel}:${msg.sender}`;
98
+ if (!pairedSenders.has(senderKey)) {
99
+ if (msg.text.trim() === config.service.pairingCode) {
100
+ pairedSenders.add(senderKey);
101
+ savePairingStore();
102
+ log.info(`Paired sender: ${senderKey}`);
103
+ return null;
104
+ }
105
+ log.debug(`Unpaired sender ${senderKey} — ignoring`);
106
+ return null;
107
+ }
108
+ }
109
+ // Try explicit mention routing first
110
+ for (const [agentId, agent] of Object.entries(config.agents)) {
111
+ for (const route of agent.routes) {
112
+ if (route.channel !== msg.channel)
113
+ continue;
114
+ const matchValue = String(route.match.value);
115
+ if (msg.chatId !== matchValue)
116
+ continue;
117
+ if (!isAllowed(msg, route)) {
118
+ log.debug(`Blocked: ${msg.sender} not in allowFrom for ${agentId}`);
119
+ return null;
120
+ }
121
+ if (route.permissions.requireMention) {
122
+ if (!hasMention(msg.text, agent)) {
123
+ continue;
124
+ }
125
+ }
126
+ // Explicit mention found — update sticky and return
127
+ setStickyAgent(msg, agentId);
128
+ return { agentId, agentConfig: agent, route };
129
+ }
130
+ }
131
+ // No explicit mention — try sticky routing
132
+ const sticky = getStickyAgent(msg, config);
133
+ if (sticky) {
134
+ // Refresh the timestamp on each sticky hit
135
+ setStickyAgent(msg, sticky.agentId);
136
+ return sticky;
137
+ }
138
+ // Fall back to default agent — only for the web channel
139
+ if (msg.channel === "web" && config.defaultAgent && config.agents[config.defaultAgent]) {
140
+ const agent = config.agents[config.defaultAgent];
141
+ const matchingRoute = agent.routes.find(r => r.channel === msg.channel);
142
+ if (matchingRoute) {
143
+ return {
144
+ agentId: config.defaultAgent,
145
+ agentConfig: agent,
146
+ route: matchingRoute,
147
+ };
148
+ }
149
+ }
150
+ log.debug(`No route for ${msg.channel}:${msg.chatId} from ${msg.senderName || msg.sender}`);
151
+ return null;
152
+ }
153
+ // Exported for pairing check in index.ts
154
+ export function isPairingAttempt(msg, config, baseDir) {
155
+ if (!config.service.pairingCode)
156
+ return false;
157
+ loadPairingStore(baseDir);
158
+ const senderKey = `${msg.channel}:${msg.sender}`;
159
+ if (pairedSenders.has(senderKey))
160
+ return false;
161
+ return msg.text.trim() === config.service.pairingCode;
162
+ }
163
+ export function pairSender(msg, baseDir) {
164
+ loadPairingStore(baseDir);
165
+ const senderKey = `${msg.channel}:${msg.sender}`;
166
+ pairedSenders.add(senderKey);
167
+ savePairingStore();
168
+ log.info(`Paired sender: ${senderKey}`);
169
+ }
170
+ function hasMention(text, agent) {
171
+ const lower = text.toLowerCase();
172
+ if (agent.mentionAliases?.length) {
173
+ return agent.mentionAliases.some((alias) => lower.includes(alias.toLowerCase()));
174
+ }
175
+ return lower.includes(agent.name.toLowerCase());
176
+ }
177
+ function isAllowed(msg, route) {
178
+ const { allowFrom } = route.permissions;
179
+ if (!allowFrom || allowFrom.length === 0)
180
+ return true;
181
+ if (allowFrom.includes("*"))
182
+ return true;
183
+ return allowFrom.includes(msg.sender);
184
+ }
185
+ //# sourceMappingURL=router.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"router.js","sourceRoot":"","sources":["../src/router.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=test-message.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"test-message.d.ts","sourceRoot":"","sources":["../src/test-message.ts"],"names":[],"mappings":""}
@@ -0,0 +1,60 @@
1
+ import { resolve, dirname } from "node:path";
2
+ import { fileURLToPath } from "node:url";
3
+ import { loadConfig } from "./config.js";
4
+ import { configureLogger, log } from "./logger.js";
5
+ import { executeAgent } from "./executor.js";
6
+ const __dirname = dirname(fileURLToPath(import.meta.url));
7
+ const baseDir = resolve(__dirname, "..");
8
+ async function main() {
9
+ const args = process.argv.slice(2);
10
+ let agentId = null;
11
+ let text = null;
12
+ for (let i = 0; i < args.length; i++) {
13
+ if (args[i] === "--agent" && args[i + 1]) {
14
+ agentId = args[++i];
15
+ }
16
+ else if (args[i] === "--text" && args[i + 1]) {
17
+ text = args[++i];
18
+ }
19
+ }
20
+ if (!agentId || !text) {
21
+ console.log("Usage: npm run test-message -- --agent <agentId> --text <message>");
22
+ console.log('Example: npm run test-message -- --agent fic-show --text "List all episodes"');
23
+ process.exit(1);
24
+ }
25
+ const configPath = resolve(baseDir, "config.json");
26
+ const config = loadConfig(configPath);
27
+ configureLogger("debug");
28
+ const agentConfig = config.agents[agentId];
29
+ if (!agentConfig) {
30
+ console.error(`Agent "${agentId}" not found. Available: ${Object.keys(config.agents).join(", ")}`);
31
+ process.exit(1);
32
+ }
33
+ const msg = {
34
+ id: "test-" + Date.now(),
35
+ channel: "test",
36
+ chatId: "test",
37
+ chatType: "dm",
38
+ sender: "test-user",
39
+ text,
40
+ timestamp: Date.now(),
41
+ isFromMe: false,
42
+ isGroup: false,
43
+ raw: {},
44
+ };
45
+ const route = {
46
+ agentId,
47
+ agentConfig,
48
+ route: agentConfig.routes[0],
49
+ };
50
+ log.info(`Testing agent "${agentId}" with message: "${text}"`);
51
+ const response = await executeAgent(route, msg, baseDir);
52
+ console.log("\n--- Agent Response ---");
53
+ console.log(response);
54
+ console.log("--- End ---\n");
55
+ }
56
+ main().catch((err) => {
57
+ console.error("Fatal error:", err);
58
+ process.exit(1);
59
+ });
60
+ //# sourceMappingURL=test-message.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"test-message.js","sourceRoot":"","sources":["../src/test-message.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,24 @@
1
+ /**
2
+ * Direct iMessage database reader.
3
+ * Reads from ~/Library/Messages/chat.db to get message text from attributedBody
4
+ * when the text column is empty (macOS 15+ behavior).
5
+ */
6
+ interface DBMessage {
7
+ rowid: number;
8
+ text: string;
9
+ chatId: number;
10
+ isFromMe: boolean;
11
+ sender: string;
12
+ createdAt: string;
13
+ guid: string;
14
+ }
15
+ /**
16
+ * Get recent messages from a specific chat, with attributedBody fallback.
17
+ */
18
+ export declare function getRecentMessages(chatId: number, sinceRowId: number, limit?: number): DBMessage[];
19
+ /**
20
+ * Get the latest ROWID for a chat (for polling).
21
+ */
22
+ export declare function getLatestRowId(chatId: number): number;
23
+ export {};
24
+ //# sourceMappingURL=imsg-db-reader.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"imsg-db-reader.d.ts","sourceRoot":"","sources":["../../src/utils/imsg-db-reader.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAQH,UAAU,SAAS;IACjB,KAAK,EAAE,MAAM,CAAC;IACd,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC;IACf,QAAQ,EAAE,OAAO,CAAC;IAClB,MAAM,EAAE,MAAM,CAAC;IACf,SAAS,EAAE,MAAM,CAAC;IAClB,IAAI,EAAE,MAAM,CAAC;CACd;AA6BD;;GAEG;AACH,wBAAgB,iBAAiB,CAAC,MAAM,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,EAAE,KAAK,GAAE,MAAW,GAAG,SAAS,EAAE,CAyCrG;AAED;;GAEG;AACH,wBAAgB,cAAc,CAAC,MAAM,EAAE,MAAM,GAAG,MAAM,CAUrD"}
@@ -0,0 +1,92 @@
1
+ /**
2
+ * Direct iMessage database reader.
3
+ * Reads from ~/Library/Messages/chat.db to get message text from attributedBody
4
+ * when the text column is empty (macOS 15+ behavior).
5
+ */
6
+ import { execSync } from "node:child_process";
7
+ import { homedir } from "node:os";
8
+ import { join } from "node:path";
9
+ const DB_PATH = join(homedir(), "Library", "Messages", "chat.db");
10
+ /**
11
+ * Extract plain text from NSAttributedString binary blob (NSArchiver/typedstream format).
12
+ * The text is stored after a \x01\x2B marker followed by a length byte, then raw UTF-8 bytes.
13
+ */
14
+ function extractTextFromAttributedBody(hexData) {
15
+ try {
16
+ const buf = Buffer.from(hexData, "hex");
17
+ // Search for the \x01\x2B (NSArchiver string marker "+") pattern
18
+ for (let i = 0; i < buf.length - 2; i++) {
19
+ if (buf[i] === 0x01 && buf[i + 1] === 0x2b) {
20
+ const len = buf[i + 2];
21
+ if (len > 0 && i + 3 + len <= buf.length) {
22
+ const text = buf.toString("utf-8", i + 3, i + 3 + len);
23
+ // Sanity check: should contain at least some printable chars
24
+ if (/[\x20-\x7E]/.test(text)) {
25
+ return text.trim();
26
+ }
27
+ }
28
+ }
29
+ }
30
+ return "";
31
+ }
32
+ catch {
33
+ return "";
34
+ }
35
+ }
36
+ /**
37
+ * Get recent messages from a specific chat, with attributedBody fallback.
38
+ */
39
+ export function getRecentMessages(chatId, sinceRowId, limit = 20) {
40
+ try {
41
+ const query = `
42
+ SELECT m.ROWID, m.text, m.is_from_me, m.handle_id,
43
+ datetime(m.date/1000000000 + 978307200, 'unixepoch', 'localtime') as created_at,
44
+ m.guid, hex(m.attributedBody) as attr_hex,
45
+ h.id as sender_id
46
+ FROM message m
47
+ JOIN chat_message_join cmj ON m.ROWID = cmj.message_id
48
+ LEFT JOIN handle h ON m.handle_id = h.ROWID
49
+ WHERE cmj.chat_id = ${chatId}
50
+ AND m.ROWID > ${sinceRowId}
51
+ AND m.associated_message_type = 0
52
+ ORDER BY m.date ASC
53
+ LIMIT ${limit};
54
+ `;
55
+ const result = execSync(`sqlite3 -json "${DB_PATH}" "${query.replace(/"/g, '\\"')}"`, {
56
+ timeout: 5000,
57
+ encoding: "utf-8",
58
+ });
59
+ const rows = JSON.parse(result || "[]");
60
+ return rows.map((row) => {
61
+ let text = row.text || "";
62
+ if (!text && row.attr_hex) {
63
+ text = extractTextFromAttributedBody(row.attr_hex);
64
+ }
65
+ return {
66
+ rowid: row.ROWID,
67
+ text,
68
+ chatId,
69
+ isFromMe: row.is_from_me === 1,
70
+ sender: row.sender_id || (row.is_from_me === 1 ? "me" : "unknown"),
71
+ createdAt: row.created_at,
72
+ guid: row.guid,
73
+ };
74
+ }).filter((m) => m.text.trim().length > 0);
75
+ }
76
+ catch {
77
+ return [];
78
+ }
79
+ }
80
+ /**
81
+ * Get the latest ROWID for a chat (for polling).
82
+ */
83
+ export function getLatestRowId(chatId) {
84
+ try {
85
+ const result = execSync(`sqlite3 "${DB_PATH}" "SELECT MAX(m.ROWID) FROM message m JOIN chat_message_join cmj ON m.ROWID = cmj.message_id WHERE cmj.chat_id = ${chatId};"`, { timeout: 5000, encoding: "utf-8" });
86
+ return parseInt(result.trim()) || 0;
87
+ }
88
+ catch {
89
+ return 0;
90
+ }
91
+ }
92
+ //# sourceMappingURL=imsg-db-reader.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"imsg-db-reader.js","sourceRoot":"","sources":["../../src/utils/imsg-db-reader.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,25 @@
1
+ interface JsonRpcNotification {
2
+ jsonrpc: "2.0";
3
+ method: string;
4
+ params: unknown;
5
+ }
6
+ export declare class ImsgRpcClient {
7
+ private proc;
8
+ private rl;
9
+ private nextId;
10
+ private pending;
11
+ private notificationHandler;
12
+ private cliPath;
13
+ private stopping;
14
+ private restartTimer;
15
+ constructor(cliPath?: string);
16
+ start(): Promise<void>;
17
+ private handleLine;
18
+ request<T>(method: string, params?: object, timeoutMs?: number): Promise<T>;
19
+ onNotification(handler: (notification: JsonRpcNotification) => void): void;
20
+ private restart;
21
+ private startProcess;
22
+ stop(): Promise<void>;
23
+ }
24
+ export {};
25
+ //# sourceMappingURL=imsg-rpc.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"imsg-rpc.d.ts","sourceRoot":"","sources":["../../src/utils/imsg-rpc.ts"],"names":[],"mappings":"AAUA,UAAU,mBAAmB;IAC3B,OAAO,EAAE,KAAK,CAAC;IACf,MAAM,EAAE,MAAM,CAAC;IACf,MAAM,EAAE,OAAO,CAAC;CACjB;AAED,qBAAa,aAAa;IACxB,OAAO,CAAC,IAAI,CAA6B;IACzC,OAAO,CAAC,EAAE,CAA0B;IACpC,OAAO,CAAC,MAAM,CAAK;IACnB,OAAO,CAAC,OAAO,CAAqC;IACpD,OAAO,CAAC,mBAAmB,CAA8D;IACzF,OAAO,CAAC,OAAO,CAAS;IACxB,OAAO,CAAC,QAAQ,CAAS;IACzB,OAAO,CAAC,YAAY,CAA8C;gBAEtD,OAAO,SAAS;IAItB,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAqB5B,OAAO,CAAC,UAAU;IAoCZ,OAAO,CAAC,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,MAAM,EAAE,SAAS,SAAS,GAAG,OAAO,CAAC,CAAC,CAAC;IAwBjF,cAAc,CAAC,OAAO,EAAE,CAAC,YAAY,EAAE,mBAAmB,KAAK,IAAI,GAAG,IAAI;YAI5D,OAAO;YAcP,YAAY;IAmCpB,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;CAa5B"}