agent-relay 2.0.28 → 2.0.32

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 (894) hide show
  1. package/README.md +19 -0
  2. package/dist/index.cjs +85691 -0
  3. package/dist/src/bridge/index.d.ts.map +1 -0
  4. package/dist/src/bridge/index.js.map +1 -0
  5. package/dist/src/cli/commands/doctor.d.ts +2 -0
  6. package/dist/src/cli/commands/doctor.d.ts.map +1 -0
  7. package/dist/src/cli/commands/doctor.js +451 -0
  8. package/dist/src/cli/commands/doctor.js.map +1 -0
  9. package/dist/src/cli/index.d.ts.map +1 -0
  10. package/dist/src/cli/index.js +29 -1
  11. package/dist/src/cli/index.js.map +1 -0
  12. package/dist/src/config/relay-config.d.ts.map +1 -0
  13. package/dist/src/config/relay-config.js.map +1 -0
  14. package/dist/src/continuity/index.d.ts.map +1 -0
  15. package/dist/src/continuity/index.js.map +1 -0
  16. package/dist/src/daemon/index.d.ts.map +1 -0
  17. package/dist/src/daemon/index.js.map +1 -0
  18. package/dist/src/health-worker-manager.d.ts.map +1 -0
  19. package/dist/src/health-worker-manager.js.map +1 -0
  20. package/dist/src/health-worker.d.ts.map +1 -0
  21. package/dist/src/health-worker.js.map +1 -0
  22. package/dist/src/hooks/index.d.ts.map +1 -0
  23. package/dist/src/hooks/index.js.map +1 -0
  24. package/dist/src/index.d.ts.map +1 -0
  25. package/dist/src/index.js.map +1 -0
  26. package/dist/src/memory/index.d.ts.map +1 -0
  27. package/dist/src/memory/index.js.map +1 -0
  28. package/dist/src/policy/index.d.ts.map +1 -0
  29. package/dist/src/policy/index.js.map +1 -0
  30. package/dist/src/protocol/index.d.ts.map +1 -0
  31. package/dist/src/protocol/index.js.map +1 -0
  32. package/dist/src/resiliency/index.d.ts.map +1 -0
  33. package/dist/src/resiliency/index.js.map +1 -0
  34. package/dist/src/shared/cli-auth-config.d.ts.map +1 -0
  35. package/dist/src/shared/cli-auth-config.js.map +1 -0
  36. package/dist/src/state/index.d.ts.map +1 -0
  37. package/dist/src/state/index.js.map +1 -0
  38. package/dist/src/storage/index.d.ts.map +1 -0
  39. package/dist/src/storage/index.js.map +1 -0
  40. package/dist/src/trajectory/index.d.ts.map +1 -0
  41. package/dist/src/trajectory/index.js.map +1 -0
  42. package/dist/src/utils/index.d.ts.map +1 -0
  43. package/dist/src/utils/index.js.map +1 -0
  44. package/dist/src/wrapper/index.d.ts.map +1 -0
  45. package/dist/src/wrapper/index.js.map +1 -0
  46. package/package.json +83 -20
  47. package/packages/api-types/dist/index.d.ts.map +1 -0
  48. package/packages/api-types/dist/index.js.map +1 -0
  49. package/packages/api-types/dist/schemas/agent.d.ts.map +1 -0
  50. package/packages/api-types/dist/schemas/agent.js.map +1 -0
  51. package/packages/api-types/dist/schemas/api.d.ts.map +1 -0
  52. package/packages/api-types/dist/schemas/api.js.map +1 -0
  53. package/packages/api-types/dist/schemas/decision.d.ts.map +1 -0
  54. package/packages/api-types/dist/schemas/decision.js.map +1 -0
  55. package/packages/api-types/dist/schemas/fleet.d.ts.map +1 -0
  56. package/packages/api-types/dist/schemas/fleet.js.map +1 -0
  57. package/packages/api-types/dist/schemas/history.d.ts.map +1 -0
  58. package/packages/api-types/dist/schemas/history.js.map +1 -0
  59. package/packages/api-types/dist/schemas/index.d.ts.map +1 -0
  60. package/packages/api-types/dist/schemas/index.js.map +1 -0
  61. package/packages/api-types/dist/schemas/message.d.ts.map +1 -0
  62. package/packages/api-types/dist/schemas/message.js.map +1 -0
  63. package/packages/api-types/dist/schemas/session.d.ts.map +1 -0
  64. package/packages/api-types/dist/schemas/session.js.map +1 -0
  65. package/packages/api-types/dist/schemas/task.d.ts.map +1 -0
  66. package/packages/api-types/dist/schemas/task.js.map +1 -0
  67. package/packages/api-types/package.json +1 -1
  68. package/packages/api-types/src/index.ts +22 -0
  69. package/packages/api-types/src/schemas/agent.test.ts +164 -0
  70. package/packages/api-types/src/schemas/agent.ts +110 -0
  71. package/packages/api-types/src/schemas/api.test.ts +372 -0
  72. package/packages/api-types/src/schemas/api.ts +194 -0
  73. package/packages/api-types/src/schemas/decision.test.ts +324 -0
  74. package/packages/api-types/src/schemas/decision.ts +136 -0
  75. package/packages/api-types/src/schemas/fleet.test.ts +212 -0
  76. package/packages/api-types/src/schemas/fleet.ts +83 -0
  77. package/packages/api-types/src/schemas/history.test.ts +242 -0
  78. package/packages/api-types/src/schemas/history.ts +84 -0
  79. package/packages/api-types/src/schemas/index.ts +148 -0
  80. package/packages/api-types/src/schemas/message.test.ts +192 -0
  81. package/packages/api-types/src/schemas/message.ts +98 -0
  82. package/packages/api-types/src/schemas/session.test.ts +104 -0
  83. package/packages/api-types/src/schemas/session.ts +40 -0
  84. package/packages/api-types/src/schemas/task.test.ts +192 -0
  85. package/packages/api-types/src/schemas/task.ts +78 -0
  86. package/packages/api-types/tsconfig.json +19 -0
  87. package/packages/api-types/vitest.config.ts +9 -0
  88. package/packages/benchmark/README.md +200 -0
  89. package/packages/benchmark/datasets/coding-tasks.yaml +127 -0
  90. package/packages/benchmark/datasets/coordination-tasks.yaml +122 -0
  91. package/packages/benchmark/dist/benchmark.d.ts +47 -0
  92. package/packages/benchmark/dist/benchmark.d.ts.map +1 -0
  93. package/packages/benchmark/dist/benchmark.js +224 -0
  94. package/packages/benchmark/dist/benchmark.js.map +1 -0
  95. package/packages/benchmark/dist/cli.d.ts +8 -0
  96. package/packages/benchmark/dist/cli.d.ts.map +1 -0
  97. package/packages/benchmark/dist/cli.js +185 -0
  98. package/packages/benchmark/dist/cli.js.map +1 -0
  99. package/packages/benchmark/dist/harbor.d.ts +53 -0
  100. package/packages/benchmark/dist/harbor.d.ts.map +1 -0
  101. package/packages/benchmark/dist/harbor.js +127 -0
  102. package/packages/benchmark/dist/harbor.js.map +1 -0
  103. package/packages/benchmark/dist/index.d.ts +48 -0
  104. package/packages/benchmark/dist/index.d.ts.map +1 -0
  105. package/packages/benchmark/dist/index.js +50 -0
  106. package/packages/benchmark/dist/index.js.map +1 -0
  107. package/packages/benchmark/dist/runners/base.d.ts +63 -0
  108. package/packages/benchmark/dist/runners/base.d.ts.map +1 -0
  109. package/packages/benchmark/dist/runners/base.js +155 -0
  110. package/packages/benchmark/dist/runners/base.js.map +1 -0
  111. package/packages/benchmark/dist/runners/index.d.ts +10 -0
  112. package/packages/benchmark/dist/runners/index.d.ts.map +1 -0
  113. package/packages/benchmark/dist/runners/index.js +10 -0
  114. package/packages/benchmark/dist/runners/index.js.map +1 -0
  115. package/packages/benchmark/dist/runners/single.d.ts +19 -0
  116. package/packages/benchmark/dist/runners/single.d.ts.map +1 -0
  117. package/packages/benchmark/dist/runners/single.js +111 -0
  118. package/packages/benchmark/dist/runners/single.js.map +1 -0
  119. package/packages/benchmark/dist/runners/subagent.d.ts +32 -0
  120. package/packages/benchmark/dist/runners/subagent.d.ts.map +1 -0
  121. package/packages/benchmark/dist/runners/subagent.js +212 -0
  122. package/packages/benchmark/dist/runners/subagent.js.map +1 -0
  123. package/packages/benchmark/dist/runners/swarm.d.ts +36 -0
  124. package/packages/benchmark/dist/runners/swarm.d.ts.map +1 -0
  125. package/packages/benchmark/dist/runners/swarm.js +273 -0
  126. package/packages/benchmark/dist/runners/swarm.js.map +1 -0
  127. package/packages/benchmark/dist/types.d.ts +178 -0
  128. package/packages/benchmark/dist/types.d.ts.map +1 -0
  129. package/packages/benchmark/dist/types.js +16 -0
  130. package/packages/benchmark/dist/types.js.map +1 -0
  131. package/packages/benchmark/package.json +80 -0
  132. package/packages/benchmark/src/benchmark.ts +298 -0
  133. package/packages/benchmark/src/cli.ts +240 -0
  134. package/packages/benchmark/src/harbor.ts +170 -0
  135. package/packages/benchmark/src/index.ts +73 -0
  136. package/packages/benchmark/src/runners/base.ts +204 -0
  137. package/packages/benchmark/src/runners/index.ts +10 -0
  138. package/packages/benchmark/src/runners/single.ts +121 -0
  139. package/packages/benchmark/src/runners/subagent.ts +240 -0
  140. package/packages/benchmark/src/runners/swarm.ts +326 -0
  141. package/packages/benchmark/src/types.ts +205 -0
  142. package/packages/benchmark/tsconfig.json +20 -0
  143. package/packages/bridge/dist/index.d.ts.map +1 -0
  144. package/packages/bridge/dist/index.js.map +1 -0
  145. package/packages/bridge/dist/multi-project-client.d.ts.map +1 -0
  146. package/packages/bridge/dist/multi-project-client.js.map +1 -0
  147. package/packages/bridge/dist/shadow-cli.d.ts.map +1 -0
  148. package/packages/bridge/dist/shadow-cli.js.map +1 -0
  149. package/packages/bridge/dist/spawner.d.ts.map +1 -0
  150. package/packages/bridge/dist/spawner.js +15 -2
  151. package/packages/bridge/dist/spawner.js.map +1 -0
  152. package/packages/bridge/dist/types.d.ts.map +1 -0
  153. package/packages/bridge/dist/types.js.map +1 -0
  154. package/packages/bridge/dist/utils.d.ts.map +1 -0
  155. package/packages/bridge/dist/utils.js.map +1 -0
  156. package/packages/bridge/package.json +8 -8
  157. package/packages/bridge/src/index.ts +25 -0
  158. package/packages/bridge/src/multi-project-client.test.ts +340 -0
  159. package/packages/bridge/src/multi-project-client.ts +469 -0
  160. package/packages/bridge/src/shadow-cli.ts +95 -0
  161. package/packages/bridge/src/spawner-mcp.test.ts +505 -0
  162. package/packages/bridge/src/spawner.ts +1724 -0
  163. package/packages/bridge/src/types.ts +145 -0
  164. package/packages/bridge/src/utils.test.ts +98 -0
  165. package/packages/bridge/src/utils.ts +67 -0
  166. package/packages/bridge/tsconfig.json +29 -0
  167. package/packages/bridge/vitest.config.ts +9 -0
  168. package/packages/cli-tester/dist/index.d.ts.map +1 -0
  169. package/packages/cli-tester/dist/index.js.map +1 -0
  170. package/packages/cli-tester/dist/utils/credential-check.d.ts.map +1 -0
  171. package/packages/cli-tester/dist/utils/credential-check.js.map +1 -0
  172. package/packages/cli-tester/dist/utils/socket-client.d.ts.map +1 -0
  173. package/packages/cli-tester/dist/utils/socket-client.js.map +1 -0
  174. package/packages/cli-tester/docker/Dockerfile +61 -0
  175. package/packages/cli-tester/docker/docker-compose.yml +71 -0
  176. package/packages/cli-tester/package.json +1 -1
  177. package/packages/cli-tester/src/index.ts +40 -0
  178. package/packages/cli-tester/src/utils/credential-check.ts +284 -0
  179. package/packages/cli-tester/src/utils/socket-client.ts +211 -0
  180. package/packages/cli-tester/tests/credential-check.test.ts +56 -0
  181. package/packages/cli-tester/tsconfig.json +11 -0
  182. package/packages/config/dist/agent-config.d.ts.map +1 -0
  183. package/packages/config/dist/agent-config.js.map +1 -0
  184. package/packages/config/dist/bridge-config.d.ts.map +1 -0
  185. package/packages/config/dist/bridge-config.js.map +1 -0
  186. package/packages/config/dist/bridge-utils.d.ts.map +1 -0
  187. package/packages/config/dist/bridge-utils.js.map +1 -0
  188. package/packages/config/dist/cli-auth-config.d.ts.map +1 -0
  189. package/packages/config/dist/cli-auth-config.js.map +1 -0
  190. package/packages/config/dist/cloud-config.d.ts.map +1 -0
  191. package/packages/config/dist/cloud-config.js.map +1 -0
  192. package/packages/config/dist/index.d.ts.map +1 -0
  193. package/packages/config/dist/index.js.map +1 -0
  194. package/packages/config/dist/project-namespace.d.ts.map +1 -0
  195. package/packages/config/dist/project-namespace.js.map +1 -0
  196. package/packages/config/dist/relay-config.d.ts.map +1 -0
  197. package/packages/config/dist/relay-config.js.map +1 -0
  198. package/packages/config/dist/relay-file-writer.d.ts.map +1 -0
  199. package/packages/config/dist/relay-file-writer.js.map +1 -0
  200. package/packages/config/dist/schemas.d.ts.map +1 -0
  201. package/packages/config/dist/schemas.js.map +1 -0
  202. package/packages/config/dist/shadow-config.d.ts.map +1 -0
  203. package/packages/config/dist/shadow-config.js.map +1 -0
  204. package/packages/config/dist/teams-config.d.ts.map +1 -0
  205. package/packages/config/dist/teams-config.js.map +1 -0
  206. package/packages/config/dist/trajectory-config.d.ts.map +1 -0
  207. package/packages/config/dist/trajectory-config.js.map +1 -0
  208. package/packages/config/package.json +2 -2
  209. package/packages/config/src/agent-config.test.ts +245 -0
  210. package/packages/config/src/agent-config.ts +160 -0
  211. package/packages/config/src/bridge-config.test.ts +132 -0
  212. package/packages/config/src/bridge-config.ts +189 -0
  213. package/packages/config/src/bridge-utils.ts +59 -0
  214. package/packages/config/src/cli-auth-config.ts +548 -0
  215. package/packages/config/src/cloud-config.ts +208 -0
  216. package/packages/config/src/index.ts +12 -0
  217. package/packages/config/src/project-namespace.ts +344 -0
  218. package/packages/config/src/relay-config.test.ts +51 -0
  219. package/packages/config/src/relay-config.ts +36 -0
  220. package/packages/config/src/relay-file-writer.test.ts +351 -0
  221. package/packages/config/src/relay-file-writer.ts +508 -0
  222. package/packages/config/src/schemas.test.ts +59 -0
  223. package/packages/config/src/schemas.ts +201 -0
  224. package/packages/config/src/shadow-config.ts +205 -0
  225. package/packages/config/src/teams-config.ts +135 -0
  226. package/packages/config/src/trajectory-config.ts +222 -0
  227. package/packages/config/tsconfig.json +21 -0
  228. package/packages/config/vitest.config.ts +9 -0
  229. package/packages/continuity/dist/formatter.d.ts.map +1 -0
  230. package/packages/continuity/dist/formatter.js.map +1 -0
  231. package/packages/continuity/dist/handoff-store.d.ts.map +1 -0
  232. package/packages/continuity/dist/handoff-store.js.map +1 -0
  233. package/packages/continuity/dist/index.d.ts.map +1 -0
  234. package/packages/continuity/dist/index.js.map +1 -0
  235. package/packages/continuity/dist/ledger-store.d.ts.map +1 -0
  236. package/packages/continuity/dist/ledger-store.js.map +1 -0
  237. package/packages/continuity/dist/manager.d.ts.map +1 -0
  238. package/packages/continuity/dist/manager.js.map +1 -0
  239. package/packages/continuity/dist/parser.d.ts.map +1 -0
  240. package/packages/continuity/dist/parser.js.map +1 -0
  241. package/packages/continuity/dist/types.d.ts.map +1 -0
  242. package/packages/continuity/dist/types.js.map +1 -0
  243. package/packages/continuity/package.json +1 -1
  244. package/packages/continuity/src/formatter.ts +371 -0
  245. package/packages/continuity/src/handoff-store.ts +523 -0
  246. package/packages/continuity/src/index.ts +9 -0
  247. package/packages/continuity/src/ledger-store.ts +594 -0
  248. package/packages/continuity/src/manager.test.ts +291 -0
  249. package/packages/continuity/src/manager.ts +774 -0
  250. package/packages/continuity/src/parser.test.ts +292 -0
  251. package/packages/continuity/src/parser.ts +680 -0
  252. package/packages/continuity/src/types.ts +211 -0
  253. package/packages/continuity/tsconfig.json +21 -0
  254. package/packages/continuity/vitest.config.ts +9 -0
  255. package/packages/daemon/dist/agent-manager.d.ts.map +1 -0
  256. package/packages/daemon/dist/agent-manager.js.map +1 -0
  257. package/packages/daemon/dist/agent-registry.d.ts.map +1 -0
  258. package/packages/daemon/dist/agent-registry.js.map +1 -0
  259. package/packages/daemon/dist/agent-signing.d.ts.map +1 -0
  260. package/packages/daemon/dist/agent-signing.js.map +1 -0
  261. package/packages/daemon/dist/api.d.ts.map +1 -0
  262. package/packages/daemon/dist/api.js.map +1 -0
  263. package/packages/daemon/dist/auth.d.ts.map +1 -0
  264. package/packages/daemon/dist/auth.js.map +1 -0
  265. package/packages/daemon/dist/channel-membership-store.d.ts.map +1 -0
  266. package/packages/daemon/dist/channel-membership-store.js.map +1 -0
  267. package/packages/daemon/dist/cli-auth.d.ts.map +1 -0
  268. package/packages/daemon/dist/cli-auth.js.map +1 -0
  269. package/packages/daemon/dist/cloud-sync.d.ts.map +1 -0
  270. package/packages/daemon/dist/cloud-sync.js.map +1 -0
  271. package/packages/daemon/dist/connection.d.ts.map +1 -0
  272. package/packages/daemon/dist/connection.js.map +1 -0
  273. package/packages/daemon/dist/consensus-integration.d.ts.map +1 -0
  274. package/packages/daemon/dist/consensus-integration.js.map +1 -0
  275. package/packages/daemon/dist/consensus.d.ts.map +1 -0
  276. package/packages/daemon/dist/consensus.js.map +1 -0
  277. package/packages/daemon/dist/delivery-tracker.d.ts.map +1 -0
  278. package/packages/daemon/dist/delivery-tracker.js.map +1 -0
  279. package/packages/daemon/dist/enhanced-features.d.ts.map +1 -0
  280. package/packages/daemon/dist/enhanced-features.js.map +1 -0
  281. package/packages/daemon/dist/index.d.ts.map +1 -0
  282. package/packages/daemon/dist/index.js.map +1 -0
  283. package/packages/daemon/dist/migrations/index.d.ts.map +1 -0
  284. package/packages/daemon/dist/migrations/index.js.map +1 -0
  285. package/packages/daemon/dist/orchestrator.d.ts.map +1 -0
  286. package/packages/daemon/dist/orchestrator.js.map +1 -0
  287. package/packages/daemon/dist/rate-limiter.d.ts.map +1 -0
  288. package/packages/daemon/dist/rate-limiter.js.map +1 -0
  289. package/packages/daemon/dist/registry.d.ts.map +1 -0
  290. package/packages/daemon/dist/registry.js.map +1 -0
  291. package/packages/daemon/dist/relay-ledger.d.ts.map +1 -0
  292. package/packages/daemon/dist/relay-ledger.js.map +1 -0
  293. package/packages/daemon/dist/relay-watchdog.d.ts.map +1 -0
  294. package/packages/daemon/dist/relay-watchdog.js.map +1 -0
  295. package/packages/daemon/dist/repo-manager.d.ts.map +1 -0
  296. package/packages/daemon/dist/repo-manager.js.map +1 -0
  297. package/packages/daemon/dist/router.d.ts.map +1 -0
  298. package/packages/daemon/dist/router.js.map +1 -0
  299. package/packages/daemon/dist/server.d.ts +1 -0
  300. package/packages/daemon/dist/server.d.ts.map +1 -0
  301. package/packages/daemon/dist/server.js +46 -16
  302. package/packages/daemon/dist/server.js.map +1 -0
  303. package/packages/daemon/dist/spawn-manager.d.ts.map +1 -0
  304. package/packages/daemon/dist/spawn-manager.js.map +1 -0
  305. package/packages/daemon/dist/sync-queue.d.ts.map +1 -0
  306. package/packages/daemon/dist/sync-queue.js.map +1 -0
  307. package/packages/daemon/dist/types.d.ts.map +1 -0
  308. package/packages/daemon/dist/types.js.map +1 -0
  309. package/packages/daemon/dist/workspace-manager.d.ts.map +1 -0
  310. package/packages/daemon/dist/workspace-manager.js.map +1 -0
  311. package/packages/daemon/package.json +12 -12
  312. package/packages/daemon/src/agent-manager.ts +679 -0
  313. package/packages/daemon/src/agent-registry.ts +284 -0
  314. package/packages/daemon/src/agent-signing.ts +707 -0
  315. package/packages/daemon/src/api.ts +1012 -0
  316. package/packages/daemon/src/auth.ts +276 -0
  317. package/packages/daemon/src/channel-membership-store.ts +217 -0
  318. package/packages/daemon/src/cli-auth.ts +906 -0
  319. package/packages/daemon/src/cloud-sync.ts +902 -0
  320. package/packages/daemon/src/connection.ts +534 -0
  321. package/packages/daemon/src/consensus-integration.ts +510 -0
  322. package/packages/daemon/src/consensus.ts +848 -0
  323. package/packages/daemon/src/delivery-tracker.ts +145 -0
  324. package/packages/daemon/src/enhanced-features.ts +390 -0
  325. package/packages/daemon/src/index.ts +52 -0
  326. package/packages/daemon/src/migrations/0001_initial.sql +72 -0
  327. package/packages/daemon/src/migrations/index.test.ts +195 -0
  328. package/packages/daemon/src/migrations/index.ts +286 -0
  329. package/packages/daemon/src/orchestrator.test.ts +231 -0
  330. package/packages/daemon/src/orchestrator.ts +1376 -0
  331. package/packages/daemon/src/rate-limiter.ts +172 -0
  332. package/packages/daemon/src/registry.ts +8 -0
  333. package/packages/daemon/src/relay-ledger.test.ts +358 -0
  334. package/packages/daemon/src/relay-ledger.ts +713 -0
  335. package/packages/daemon/src/relay-watchdog.test.ts +881 -0
  336. package/packages/daemon/src/relay-watchdog.ts +785 -0
  337. package/packages/daemon/src/repo-manager.ts +468 -0
  338. package/packages/daemon/src/router.test.ts +149 -0
  339. package/packages/daemon/src/router.ts +1885 -0
  340. package/packages/daemon/src/server.ts +1871 -0
  341. package/packages/daemon/src/spawn-manager.ts +275 -0
  342. package/packages/daemon/src/sync-queue.ts +477 -0
  343. package/packages/daemon/src/types.ts +158 -0
  344. package/packages/daemon/src/workspace-manager.ts +371 -0
  345. package/packages/daemon/tsconfig.json +21 -0
  346. package/packages/hooks/dist/browser.d.ts.map +1 -0
  347. package/packages/hooks/dist/browser.js.map +1 -0
  348. package/packages/hooks/dist/emitter.d.ts.map +1 -0
  349. package/packages/hooks/dist/emitter.js.map +1 -0
  350. package/packages/hooks/dist/inbox-check/hook.d.ts.map +1 -0
  351. package/packages/hooks/dist/inbox-check/hook.js.map +1 -0
  352. package/packages/hooks/dist/inbox-check/index.d.ts.map +1 -0
  353. package/packages/hooks/dist/inbox-check/index.js.map +1 -0
  354. package/packages/hooks/dist/inbox-check/types.d.ts.map +1 -0
  355. package/packages/hooks/dist/inbox-check/types.js.map +1 -0
  356. package/packages/hooks/dist/inbox-check/utils.d.ts.map +1 -0
  357. package/packages/hooks/dist/inbox-check/utils.js.map +1 -0
  358. package/packages/hooks/dist/index.d.ts.map +1 -0
  359. package/packages/hooks/dist/index.js.map +1 -0
  360. package/packages/hooks/dist/registry.d.ts.map +1 -0
  361. package/packages/hooks/dist/registry.js.map +1 -0
  362. package/packages/hooks/dist/trajectory-hooks.d.ts.map +1 -0
  363. package/packages/hooks/dist/trajectory-hooks.js.map +1 -0
  364. package/packages/hooks/dist/types.d.ts.map +1 -0
  365. package/packages/hooks/dist/types.js.map +1 -0
  366. package/packages/hooks/package.json +4 -4
  367. package/packages/hooks/src/browser.ts +2 -0
  368. package/packages/hooks/src/emitter.ts +84 -0
  369. package/packages/hooks/src/inbox-check/hook.ts +114 -0
  370. package/packages/hooks/src/inbox-check/index.ts +8 -0
  371. package/packages/hooks/src/inbox-check/types.ts +39 -0
  372. package/packages/hooks/src/inbox-check/utils.test.ts +287 -0
  373. package/packages/hooks/src/inbox-check/utils.ts +125 -0
  374. package/packages/hooks/src/index.ts +11 -0
  375. package/packages/hooks/src/registry.ts +614 -0
  376. package/packages/hooks/src/shims.d.ts +3 -0
  377. package/packages/hooks/src/trajectory-hooks.ts +251 -0
  378. package/packages/hooks/src/types.ts +342 -0
  379. package/packages/hooks/tsconfig.json +21 -0
  380. package/packages/hooks/vitest.config.ts +9 -0
  381. package/packages/mcp/dist/bin.d.ts.map +1 -0
  382. package/packages/mcp/dist/bin.js.map +1 -0
  383. package/packages/mcp/dist/client.d.ts +9 -15
  384. package/packages/mcp/dist/client.d.ts.map +1 -0
  385. package/packages/mcp/dist/client.js +42 -74
  386. package/packages/mcp/dist/client.js.map +1 -0
  387. package/packages/mcp/dist/cloud.d.ts.map +1 -0
  388. package/packages/mcp/dist/cloud.js.map +1 -0
  389. package/packages/mcp/dist/errors.d.ts.map +1 -0
  390. package/packages/mcp/dist/errors.js.map +1 -0
  391. package/packages/mcp/dist/file-transport.d.ts.map +1 -0
  392. package/packages/mcp/dist/file-transport.js.map +1 -0
  393. package/packages/mcp/dist/hybrid-client.d.ts.map +1 -0
  394. package/packages/mcp/dist/hybrid-client.js.map +1 -0
  395. package/packages/mcp/dist/index.d.ts.map +1 -0
  396. package/packages/mcp/dist/index.js.map +1 -0
  397. package/packages/mcp/dist/install-cli.d.ts.map +1 -0
  398. package/packages/mcp/dist/install-cli.js.map +1 -0
  399. package/packages/mcp/dist/install.d.ts.map +1 -0
  400. package/packages/mcp/dist/install.js.map +1 -0
  401. package/packages/mcp/dist/prompts/index.d.ts.map +1 -0
  402. package/packages/mcp/dist/prompts/index.js.map +1 -0
  403. package/packages/mcp/dist/prompts/protocol.d.ts.map +1 -0
  404. package/packages/mcp/dist/prompts/protocol.js.map +1 -0
  405. package/packages/mcp/dist/resources/agents.d.ts.map +1 -0
  406. package/packages/mcp/dist/resources/agents.js.map +1 -0
  407. package/packages/mcp/dist/resources/inbox.d.ts.map +1 -0
  408. package/packages/mcp/dist/resources/inbox.js.map +1 -0
  409. package/packages/mcp/dist/resources/index.d.ts.map +1 -0
  410. package/packages/mcp/dist/resources/index.js.map +1 -0
  411. package/packages/mcp/dist/resources/project.d.ts.map +1 -0
  412. package/packages/mcp/dist/resources/project.js.map +1 -0
  413. package/packages/mcp/dist/server.d.ts.map +1 -0
  414. package/packages/mcp/dist/server.js.map +1 -0
  415. package/packages/mcp/dist/simple.d.ts +2 -5
  416. package/packages/mcp/dist/simple.d.ts.map +1 -0
  417. package/packages/mcp/dist/simple.js.map +1 -0
  418. package/packages/mcp/dist/tools/index.d.ts.map +1 -0
  419. package/packages/mcp/dist/tools/index.js.map +1 -0
  420. package/packages/mcp/dist/tools/relay-broadcast.d.ts.map +1 -0
  421. package/packages/mcp/dist/tools/relay-broadcast.js.map +1 -0
  422. package/packages/mcp/dist/tools/relay-channel.d.ts.map +1 -0
  423. package/packages/mcp/dist/tools/relay-channel.js.map +1 -0
  424. package/packages/mcp/dist/tools/relay-connected.d.ts.map +1 -0
  425. package/packages/mcp/dist/tools/relay-connected.js.map +1 -0
  426. package/packages/mcp/dist/tools/relay-consensus.d.ts.map +1 -0
  427. package/packages/mcp/dist/tools/relay-consensus.js.map +1 -0
  428. package/packages/mcp/dist/tools/relay-continuity.d.ts.map +1 -0
  429. package/packages/mcp/dist/tools/relay-continuity.js.map +1 -0
  430. package/packages/mcp/dist/tools/relay-health.d.ts.map +1 -0
  431. package/packages/mcp/dist/tools/relay-health.js.map +1 -0
  432. package/packages/mcp/dist/tools/relay-inbox.d.ts.map +1 -0
  433. package/packages/mcp/dist/tools/relay-inbox.js.map +1 -0
  434. package/packages/mcp/dist/tools/relay-logs.d.ts.map +1 -0
  435. package/packages/mcp/dist/tools/relay-logs.js.map +1 -0
  436. package/packages/mcp/dist/tools/relay-metrics.d.ts.map +1 -0
  437. package/packages/mcp/dist/tools/relay-metrics.js.map +1 -0
  438. package/packages/mcp/dist/tools/relay-release.d.ts.map +1 -0
  439. package/packages/mcp/dist/tools/relay-release.js.map +1 -0
  440. package/packages/mcp/dist/tools/relay-remove-agent.d.ts.map +1 -0
  441. package/packages/mcp/dist/tools/relay-remove-agent.js.map +1 -0
  442. package/packages/mcp/dist/tools/relay-send.d.ts.map +1 -0
  443. package/packages/mcp/dist/tools/relay-send.js +4 -2
  444. package/packages/mcp/dist/tools/relay-send.js.map +1 -0
  445. package/packages/mcp/dist/tools/relay-shadow.d.ts.map +1 -0
  446. package/packages/mcp/dist/tools/relay-shadow.js.map +1 -0
  447. package/packages/mcp/dist/tools/relay-spawn.d.ts.map +1 -0
  448. package/packages/mcp/dist/tools/relay-spawn.js.map +1 -0
  449. package/packages/mcp/dist/tools/relay-status.d.ts.map +1 -0
  450. package/packages/mcp/dist/tools/relay-status.js.map +1 -0
  451. package/packages/mcp/dist/tools/relay-subscribe.d.ts.map +1 -0
  452. package/packages/mcp/dist/tools/relay-subscribe.js.map +1 -0
  453. package/packages/mcp/dist/tools/relay-who.d.ts.map +1 -0
  454. package/packages/mcp/dist/tools/relay-who.js.map +1 -0
  455. package/packages/mcp/package.json +3 -3
  456. package/packages/mcp/src/bin.ts +149 -0
  457. package/packages/mcp/src/client.ts +400 -0
  458. package/packages/mcp/src/cloud.ts +523 -0
  459. package/packages/mcp/src/errors.ts +54 -0
  460. package/packages/mcp/src/file-transport.ts +268 -0
  461. package/packages/mcp/src/hybrid-client.ts +209 -0
  462. package/packages/mcp/src/index.ts +122 -0
  463. package/packages/mcp/src/install-cli.ts +210 -0
  464. package/packages/mcp/src/install.ts +745 -0
  465. package/packages/mcp/src/prompts/index.ts +1 -0
  466. package/packages/mcp/src/prompts/protocol.ts +164 -0
  467. package/packages/mcp/src/resources/agents.ts +21 -0
  468. package/packages/mcp/src/resources/inbox.ts +21 -0
  469. package/packages/mcp/src/resources/index.ts +3 -0
  470. package/packages/mcp/src/resources/project.ts +29 -0
  471. package/packages/mcp/src/server.ts +431 -0
  472. package/packages/mcp/src/simple.ts +214 -0
  473. package/packages/mcp/src/tools/index.ts +133 -0
  474. package/packages/mcp/src/tools/relay-broadcast.ts +32 -0
  475. package/packages/mcp/src/tools/relay-channel.ts +93 -0
  476. package/packages/mcp/src/tools/relay-connected.ts +52 -0
  477. package/packages/mcp/src/tools/relay-consensus.ts +92 -0
  478. package/packages/mcp/src/tools/relay-continuity.ts +127 -0
  479. package/packages/mcp/src/tools/relay-health.ts +148 -0
  480. package/packages/mcp/src/tools/relay-inbox.ts +70 -0
  481. package/packages/mcp/src/tools/relay-logs.ts +106 -0
  482. package/packages/mcp/src/tools/relay-metrics.ts +140 -0
  483. package/packages/mcp/src/tools/relay-release.ts +54 -0
  484. package/packages/mcp/src/tools/relay-remove-agent.ts +58 -0
  485. package/packages/mcp/src/tools/relay-send.ts +84 -0
  486. package/packages/mcp/src/tools/relay-shadow.ts +67 -0
  487. package/packages/mcp/src/tools/relay-spawn.ts +87 -0
  488. package/packages/mcp/src/tools/relay-status.ts +57 -0
  489. package/packages/mcp/src/tools/relay-subscribe.ts +61 -0
  490. package/packages/mcp/src/tools/relay-who.ts +59 -0
  491. package/packages/mcp/tests/client.test.ts +476 -0
  492. package/packages/mcp/tests/discover.test.ts +195 -0
  493. package/packages/mcp/tests/install.test.ts +123 -0
  494. package/packages/mcp/tests/prompts.test.ts +12 -0
  495. package/packages/mcp/tests/resources.test.ts +53 -0
  496. package/packages/mcp/tests/tools.test.ts +1242 -0
  497. package/packages/mcp/tsconfig.json +22 -0
  498. package/packages/mcp/vitest.config.ts +9 -0
  499. package/packages/memory/dist/adapters/index.d.ts.map +1 -0
  500. package/packages/memory/dist/adapters/index.js.map +1 -0
  501. package/packages/memory/dist/adapters/inmemory.d.ts.map +1 -0
  502. package/packages/memory/dist/adapters/inmemory.js.map +1 -0
  503. package/packages/memory/dist/adapters/supermemory.d.ts.map +1 -0
  504. package/packages/memory/dist/adapters/supermemory.js.map +1 -0
  505. package/packages/memory/dist/context-compaction.d.ts.map +1 -0
  506. package/packages/memory/dist/context-compaction.js.map +1 -0
  507. package/packages/memory/dist/factory.d.ts.map +1 -0
  508. package/packages/memory/dist/factory.js.map +1 -0
  509. package/packages/memory/dist/index.d.ts.map +1 -0
  510. package/packages/memory/dist/index.js.map +1 -0
  511. package/packages/memory/dist/memory-hooks.d.ts.map +1 -0
  512. package/packages/memory/dist/memory-hooks.js.map +1 -0
  513. package/packages/memory/dist/service.d.ts.map +1 -0
  514. package/packages/memory/dist/service.js.map +1 -0
  515. package/packages/memory/dist/types.d.ts.map +1 -0
  516. package/packages/memory/dist/types.js.map +1 -0
  517. package/packages/memory/package.json +2 -2
  518. package/packages/memory/src/adapters/index.ts +8 -0
  519. package/packages/memory/src/adapters/inmemory.ts +265 -0
  520. package/packages/memory/src/adapters/supermemory.ts +449 -0
  521. package/packages/memory/src/context-compaction.test.ts +660 -0
  522. package/packages/memory/src/context-compaction.ts +612 -0
  523. package/packages/memory/src/factory.ts +170 -0
  524. package/packages/memory/src/index.ts +33 -0
  525. package/packages/memory/src/memory-hooks.ts +410 -0
  526. package/packages/memory/src/service.ts +194 -0
  527. package/packages/memory/src/types.ts +211 -0
  528. package/packages/memory/tsconfig.json +21 -0
  529. package/packages/memory/vitest.config.ts +9 -0
  530. package/packages/policy/dist/agent-policy.d.ts.map +1 -0
  531. package/packages/policy/dist/agent-policy.js.map +1 -0
  532. package/packages/policy/dist/cloud-policy-fetcher.d.ts.map +1 -0
  533. package/packages/policy/dist/cloud-policy-fetcher.js.map +1 -0
  534. package/packages/policy/dist/index.d.ts.map +1 -0
  535. package/packages/policy/dist/index.js.map +1 -0
  536. package/packages/policy/package.json +2 -2
  537. package/packages/policy/src/agent-policy.ts +866 -0
  538. package/packages/policy/src/cloud-policy-fetcher.ts +78 -0
  539. package/packages/policy/src/index.ts +21 -0
  540. package/packages/policy/tsconfig.json +21 -0
  541. package/packages/policy/vitest.config.ts +9 -0
  542. package/packages/protocol/dist/channels.d.ts.map +1 -0
  543. package/packages/protocol/dist/channels.js.map +1 -0
  544. package/packages/protocol/dist/framing.d.ts.map +1 -0
  545. package/packages/protocol/dist/framing.js.map +1 -0
  546. package/packages/protocol/dist/id-generator.d.ts.map +1 -0
  547. package/packages/protocol/dist/id-generator.js.map +1 -0
  548. package/packages/protocol/dist/index.d.ts.map +1 -0
  549. package/packages/protocol/dist/index.js.map +1 -0
  550. package/packages/protocol/dist/relay-pty-schemas.d.ts +70 -2
  551. package/packages/protocol/dist/relay-pty-schemas.d.ts.map +1 -0
  552. package/packages/protocol/dist/relay-pty-schemas.js.map +1 -0
  553. package/packages/protocol/dist/types.d.ts +8 -0
  554. package/packages/protocol/dist/types.d.ts.map +1 -0
  555. package/packages/protocol/dist/types.js.map +1 -0
  556. package/packages/protocol/package.json +1 -1
  557. package/packages/protocol/src/channels.test.ts +330 -0
  558. package/packages/protocol/src/channels.ts +270 -0
  559. package/packages/protocol/src/framing.test.ts +164 -0
  560. package/packages/protocol/src/framing.ts +242 -0
  561. package/packages/protocol/src/id-generator.ts +69 -0
  562. package/packages/protocol/src/index.ts +4 -0
  563. package/packages/protocol/src/relay-pty-schemas.ts +400 -0
  564. package/packages/protocol/src/types.test.ts +271 -0
  565. package/packages/protocol/src/types.ts +846 -0
  566. package/packages/protocol/tsconfig.json +21 -0
  567. package/packages/protocol/vitest.config.ts +9 -0
  568. package/packages/resiliency/dist/cgroup-manager.d.ts.map +1 -0
  569. package/packages/resiliency/dist/cgroup-manager.js.map +1 -0
  570. package/packages/resiliency/dist/context-persistence.d.ts.map +1 -0
  571. package/packages/resiliency/dist/context-persistence.js.map +1 -0
  572. package/packages/resiliency/dist/crash-insights.d.ts.map +1 -0
  573. package/packages/resiliency/dist/crash-insights.js.map +1 -0
  574. package/packages/resiliency/dist/gossip-health.d.ts.map +1 -0
  575. package/packages/resiliency/dist/gossip-health.js.map +1 -0
  576. package/packages/resiliency/dist/health-monitor.d.ts.map +1 -0
  577. package/packages/resiliency/dist/health-monitor.js.map +1 -0
  578. package/packages/resiliency/dist/index.d.ts.map +1 -0
  579. package/packages/resiliency/dist/index.js.map +1 -0
  580. package/packages/resiliency/dist/leader-watchdog.d.ts.map +1 -0
  581. package/packages/resiliency/dist/leader-watchdog.js.map +1 -0
  582. package/packages/resiliency/dist/logger.d.ts.map +1 -0
  583. package/packages/resiliency/dist/logger.js.map +1 -0
  584. package/packages/resiliency/dist/memory-monitor.d.ts.map +1 -0
  585. package/packages/resiliency/dist/memory-monitor.js.map +1 -0
  586. package/packages/resiliency/dist/metrics.d.ts.map +1 -0
  587. package/packages/resiliency/dist/metrics.js.map +1 -0
  588. package/packages/resiliency/dist/provider-context.d.ts.map +1 -0
  589. package/packages/resiliency/dist/provider-context.js.map +1 -0
  590. package/packages/resiliency/dist/stateless-lead.d.ts.map +1 -0
  591. package/packages/resiliency/dist/stateless-lead.js.map +1 -0
  592. package/packages/resiliency/dist/supervisor.d.ts.map +1 -0
  593. package/packages/resiliency/dist/supervisor.js.map +1 -0
  594. package/packages/resiliency/package.json +1 -1
  595. package/packages/resiliency/src/cgroup-manager.ts +468 -0
  596. package/packages/resiliency/src/context-persistence.ts +538 -0
  597. package/packages/resiliency/src/crash-insights.test.ts +620 -0
  598. package/packages/resiliency/src/crash-insights.ts +660 -0
  599. package/packages/resiliency/src/gossip-health.ts +333 -0
  600. package/packages/resiliency/src/health-monitor.ts +371 -0
  601. package/packages/resiliency/src/index.ts +157 -0
  602. package/packages/resiliency/src/leader-watchdog.ts +260 -0
  603. package/packages/resiliency/src/logger.ts +320 -0
  604. package/packages/resiliency/src/memory-monitor.test.ts +637 -0
  605. package/packages/resiliency/src/memory-monitor.ts +740 -0
  606. package/packages/resiliency/src/metrics.ts +311 -0
  607. package/packages/resiliency/src/provider-context.ts +452 -0
  608. package/packages/resiliency/src/stateless-lead.ts +408 -0
  609. package/packages/resiliency/src/supervisor.ts +578 -0
  610. package/packages/resiliency/tsconfig.json +21 -0
  611. package/packages/resiliency/vitest.config.ts +9 -0
  612. package/packages/sdk/dist/client.d.ts.map +1 -0
  613. package/packages/sdk/dist/client.js.map +1 -0
  614. package/packages/sdk/dist/index.d.ts.map +1 -0
  615. package/packages/sdk/dist/index.js.map +1 -0
  616. package/packages/sdk/dist/logs.d.ts.map +1 -0
  617. package/packages/sdk/dist/logs.js.map +1 -0
  618. package/packages/sdk/dist/protocol/index.d.ts.map +1 -0
  619. package/packages/sdk/dist/protocol/index.js.map +1 -0
  620. package/packages/sdk/dist/standalone.d.ts.map +1 -0
  621. package/packages/sdk/dist/standalone.js.map +1 -0
  622. package/packages/sdk/examples/SWARM_CAPABILITIES.md +498 -0
  623. package/packages/sdk/examples/SWARM_PATTERNS.md +541 -0
  624. package/packages/sdk/package.json +2 -2
  625. package/packages/sdk/src/client.test.ts +568 -0
  626. package/packages/sdk/src/client.ts +1418 -0
  627. package/packages/sdk/src/index.ts +103 -0
  628. package/packages/sdk/src/logs.test.ts +98 -0
  629. package/packages/sdk/src/logs.ts +126 -0
  630. package/packages/sdk/src/protocol/framing.test.ts +164 -0
  631. package/packages/sdk/src/protocol/index.ts +8 -0
  632. package/packages/sdk/src/standalone.ts +176 -0
  633. package/packages/sdk/tsconfig.json +22 -0
  634. package/packages/sdk/vitest.config.ts +9 -0
  635. package/packages/spawner/.trajectories/index.json +5 -0
  636. package/packages/spawner/dist/index.d.ts.map +1 -0
  637. package/packages/spawner/dist/index.js.map +1 -0
  638. package/packages/spawner/dist/types.d.ts.map +1 -0
  639. package/packages/spawner/dist/types.js.map +1 -0
  640. package/packages/spawner/package.json +1 -1
  641. package/packages/spawner/src/index.ts +8 -0
  642. package/packages/spawner/src/types.test.ts +385 -0
  643. package/packages/spawner/src/types.ts +228 -0
  644. package/packages/spawner/tsconfig.json +19 -0
  645. package/packages/spawner/vitest.config.ts +9 -0
  646. package/packages/state/dist/agent-state.d.ts.map +1 -0
  647. package/packages/state/dist/agent-state.js.map +1 -0
  648. package/packages/state/dist/index.d.ts.map +1 -0
  649. package/packages/state/dist/index.js.map +1 -0
  650. package/packages/state/package.json +1 -1
  651. package/packages/state/src/agent-state.test.ts +335 -0
  652. package/packages/state/src/agent-state.ts +153 -0
  653. package/packages/state/src/index.ts +12 -0
  654. package/packages/state/tsconfig.json +21 -0
  655. package/packages/state/vitest.config.ts +9 -0
  656. package/packages/storage/dist/adapter.d.ts +28 -1
  657. package/packages/storage/dist/adapter.d.ts.map +1 -0
  658. package/packages/storage/dist/adapter.js +104 -10
  659. package/packages/storage/dist/adapter.js.map +1 -0
  660. package/packages/storage/dist/batched-sqlite-adapter.d.ts.map +1 -0
  661. package/packages/storage/dist/batched-sqlite-adapter.js.map +1 -0
  662. package/packages/storage/dist/dead-letter-queue.d.ts.map +1 -0
  663. package/packages/storage/dist/dead-letter-queue.js.map +1 -0
  664. package/packages/storage/dist/dlq-adapter.d.ts.map +1 -0
  665. package/packages/storage/dist/dlq-adapter.js.map +1 -0
  666. package/packages/storage/dist/index.d.ts +1 -0
  667. package/packages/storage/dist/index.d.ts.map +1 -0
  668. package/packages/storage/dist/index.js +1 -0
  669. package/packages/storage/dist/index.js.map +1 -0
  670. package/packages/storage/dist/jsonl-adapter.d.ts +77 -0
  671. package/packages/storage/dist/jsonl-adapter.d.ts.map +1 -0
  672. package/packages/storage/dist/jsonl-adapter.js +505 -0
  673. package/packages/storage/dist/jsonl-adapter.js.map +1 -0
  674. package/packages/storage/dist/sqlite-adapter.d.ts +6 -1
  675. package/packages/storage/dist/sqlite-adapter.d.ts.map +1 -0
  676. package/packages/storage/dist/sqlite-adapter.js +47 -0
  677. package/packages/storage/dist/sqlite-adapter.js.map +1 -0
  678. package/packages/storage/package.json +2 -2
  679. package/packages/storage/src/adapter.ts +438 -0
  680. package/packages/storage/src/batched-sqlite-adapter.test.ts +240 -0
  681. package/packages/storage/src/batched-sqlite-adapter.ts +239 -0
  682. package/packages/storage/src/dead-letter-queue.ts +643 -0
  683. package/packages/storage/src/dlq-adapter.test.ts +492 -0
  684. package/packages/storage/src/dlq-adapter.ts +954 -0
  685. package/packages/storage/src/index.ts +6 -0
  686. package/packages/storage/src/jsonl-adapter.test.ts +200 -0
  687. package/packages/storage/src/jsonl-adapter.ts +618 -0
  688. package/packages/storage/src/memory-adapter.test.ts +36 -0
  689. package/packages/storage/src/sqlite-adapter.test.ts +562 -0
  690. package/packages/storage/src/sqlite-adapter.ts +1058 -0
  691. package/packages/storage/tsconfig.json +21 -0
  692. package/packages/storage/vitest.config.ts +9 -0
  693. package/packages/telemetry/dist/client.d.ts.map +1 -0
  694. package/packages/telemetry/dist/client.js.map +1 -0
  695. package/packages/telemetry/dist/config.d.ts.map +1 -0
  696. package/packages/telemetry/dist/config.js.map +1 -0
  697. package/packages/telemetry/dist/events.d.ts.map +1 -0
  698. package/packages/telemetry/dist/events.js.map +1 -0
  699. package/packages/telemetry/dist/index.d.ts.map +1 -0
  700. package/packages/telemetry/dist/index.js.map +1 -0
  701. package/packages/telemetry/dist/machine-id.d.ts.map +1 -0
  702. package/packages/telemetry/dist/machine-id.js.map +1 -0
  703. package/packages/telemetry/dist/posthog-config.d.ts.map +1 -0
  704. package/packages/telemetry/dist/posthog-config.js.map +1 -0
  705. package/packages/telemetry/package.json +1 -1
  706. package/packages/telemetry/src/client.ts +158 -0
  707. package/packages/telemetry/src/config.ts +110 -0
  708. package/packages/telemetry/src/events.ts +137 -0
  709. package/packages/telemetry/src/index.ts +46 -0
  710. package/packages/telemetry/src/machine-id.ts +63 -0
  711. package/packages/telemetry/src/posthog-config.ts +39 -0
  712. package/packages/telemetry/tsconfig.json +21 -0
  713. package/packages/trajectory/dist/index.d.ts.map +1 -0
  714. package/packages/trajectory/dist/index.js.map +1 -0
  715. package/packages/trajectory/dist/integration.d.ts.map +1 -0
  716. package/packages/trajectory/dist/integration.js.map +1 -0
  717. package/packages/trajectory/package.json +2 -2
  718. package/packages/trajectory/src/index.ts +1 -0
  719. package/packages/trajectory/src/integration.ts +1268 -0
  720. package/packages/trajectory/tsconfig.json +21 -0
  721. package/packages/trajectory/vitest.config.ts +9 -0
  722. package/packages/user-directory/dist/index.d.ts.map +1 -0
  723. package/packages/user-directory/dist/index.js.map +1 -0
  724. package/packages/user-directory/dist/user-directory.d.ts.map +1 -0
  725. package/packages/user-directory/dist/user-directory.js.map +1 -0
  726. package/packages/user-directory/package.json +2 -2
  727. package/packages/user-directory/src/index.ts +12 -0
  728. package/packages/user-directory/src/user-directory.ts +393 -0
  729. package/packages/user-directory/tsconfig.json +21 -0
  730. package/packages/user-directory/vitest.config.ts +9 -0
  731. package/packages/utils/dist/cjs/client-helpers.js +127 -0
  732. package/packages/utils/dist/cjs/command-resolver.js +89 -0
  733. package/packages/utils/dist/cjs/error-tracking.js +106 -0
  734. package/packages/utils/dist/cjs/git-remote.js +120 -0
  735. package/packages/utils/dist/cjs/index.js +40 -0
  736. package/packages/utils/dist/cjs/logger.js +105 -0
  737. package/packages/utils/dist/cjs/model-mapping.js +54 -0
  738. package/packages/utils/dist/cjs/name-generator.js +179 -0
  739. package/packages/utils/dist/cjs/package.json +3 -0
  740. package/packages/utils/dist/cjs/precompiled-patterns.js +271 -0
  741. package/packages/utils/dist/cjs/relay-pty-path.js +143 -0
  742. package/packages/utils/dist/cjs/update-checker.js +185 -0
  743. package/packages/utils/dist/client-helpers.d.ts +73 -0
  744. package/packages/utils/dist/client-helpers.d.ts.map +1 -0
  745. package/packages/utils/dist/client-helpers.js +130 -0
  746. package/packages/utils/dist/client-helpers.js.map +1 -0
  747. package/packages/utils/dist/command-resolver.d.ts.map +1 -0
  748. package/packages/utils/dist/command-resolver.js.map +1 -0
  749. package/packages/utils/dist/error-tracking.d.ts.map +1 -0
  750. package/packages/utils/dist/error-tracking.js.map +1 -0
  751. package/packages/utils/dist/git-remote.d.ts.map +1 -0
  752. package/packages/utils/dist/git-remote.js.map +1 -0
  753. package/packages/utils/dist/index.d.ts +1 -0
  754. package/packages/utils/dist/index.d.ts.map +1 -0
  755. package/packages/utils/dist/index.js +1 -0
  756. package/packages/utils/dist/index.js.map +1 -0
  757. package/packages/utils/dist/logger.d.ts.map +1 -0
  758. package/packages/utils/dist/logger.js.map +1 -0
  759. package/packages/utils/dist/model-mapping.d.ts.map +1 -0
  760. package/packages/utils/dist/model-mapping.js.map +1 -0
  761. package/packages/utils/dist/name-generator.d.ts.map +1 -0
  762. package/packages/utils/dist/name-generator.js.map +1 -0
  763. package/packages/utils/dist/precompiled-patterns.d.ts.map +1 -0
  764. package/packages/utils/dist/precompiled-patterns.js.map +1 -0
  765. package/packages/utils/dist/relay-pty-path.d.ts +11 -5
  766. package/packages/utils/dist/relay-pty-path.d.ts.map +1 -0
  767. package/packages/utils/dist/relay-pty-path.js +60 -5
  768. package/packages/utils/dist/relay-pty-path.js.map +1 -0
  769. package/packages/utils/dist/update-checker.d.ts.map +1 -0
  770. package/packages/utils/dist/update-checker.js.map +1 -0
  771. package/packages/utils/package.json +37 -14
  772. package/packages/utils/scripts/build-cjs.mjs +24 -0
  773. package/packages/utils/src/client-helpers.ts +221 -0
  774. package/packages/utils/src/command-resolver.ts +82 -0
  775. package/packages/utils/src/error-tracking.ts +189 -0
  776. package/packages/utils/src/git-remote.ts +143 -0
  777. package/packages/utils/src/index.ts +10 -0
  778. package/packages/utils/src/logger.ts +107 -0
  779. package/packages/utils/src/model-mapping.test.ts +122 -0
  780. package/packages/utils/src/model-mapping.ts +58 -0
  781. package/packages/utils/src/name-generator.test.ts +259 -0
  782. package/packages/utils/src/name-generator.ts +56 -0
  783. package/packages/utils/src/precompiled-patterns.test.ts +452 -0
  784. package/packages/utils/src/precompiled-patterns.ts +395 -0
  785. package/packages/utils/src/relay-pty-path.ts +196 -0
  786. package/packages/utils/src/update-checker.test.ts +260 -0
  787. package/packages/utils/src/update-checker.ts +211 -0
  788. package/packages/utils/tsconfig.json +21 -0
  789. package/packages/utils/vitest.config.ts +9 -0
  790. package/packages/wrapper/dist/__fixtures__/claude-outputs.d.ts.map +1 -0
  791. package/packages/wrapper/dist/__fixtures__/claude-outputs.js.map +1 -0
  792. package/packages/wrapper/dist/__fixtures__/codex-outputs.d.ts.map +1 -0
  793. package/packages/wrapper/dist/__fixtures__/codex-outputs.js.map +1 -0
  794. package/packages/wrapper/dist/__fixtures__/gemini-outputs.d.ts.map +1 -0
  795. package/packages/wrapper/dist/__fixtures__/gemini-outputs.js.map +1 -0
  796. package/packages/wrapper/dist/__fixtures__/index.d.ts.map +1 -0
  797. package/packages/wrapper/dist/__fixtures__/index.js.map +1 -0
  798. package/packages/wrapper/dist/auth-detection.d.ts.map +1 -0
  799. package/packages/wrapper/dist/auth-detection.js.map +1 -0
  800. package/packages/wrapper/dist/base-wrapper.d.ts.map +1 -0
  801. package/packages/wrapper/dist/base-wrapper.js.map +1 -0
  802. package/packages/wrapper/dist/client.d.ts.map +1 -0
  803. package/packages/wrapper/dist/client.js.map +1 -0
  804. package/packages/wrapper/dist/id-generator.d.ts.map +1 -0
  805. package/packages/wrapper/dist/id-generator.js.map +1 -0
  806. package/packages/wrapper/dist/idle-detector.d.ts.map +1 -0
  807. package/packages/wrapper/dist/idle-detector.js.map +1 -0
  808. package/packages/wrapper/dist/inbox.d.ts.map +1 -0
  809. package/packages/wrapper/dist/inbox.js.map +1 -0
  810. package/packages/wrapper/dist/index.d.ts.map +1 -0
  811. package/packages/wrapper/dist/index.js.map +1 -0
  812. package/packages/wrapper/dist/parser.d.ts.map +1 -0
  813. package/packages/wrapper/dist/parser.js.map +1 -0
  814. package/packages/wrapper/dist/prompt-composer.d.ts.map +1 -0
  815. package/packages/wrapper/dist/prompt-composer.js.map +1 -0
  816. package/packages/wrapper/dist/relay-pty-orchestrator.d.ts +10 -0
  817. package/packages/wrapper/dist/relay-pty-orchestrator.d.ts.map +1 -0
  818. package/packages/wrapper/dist/relay-pty-orchestrator.js +69 -0
  819. package/packages/wrapper/dist/relay-pty-orchestrator.js.map +1 -0
  820. package/packages/wrapper/dist/shared.d.ts.map +1 -0
  821. package/packages/wrapper/dist/shared.js.map +1 -0
  822. package/packages/wrapper/dist/stuck-detector.d.ts.map +1 -0
  823. package/packages/wrapper/dist/stuck-detector.js.map +1 -0
  824. package/packages/wrapper/dist/tmux-resolver.d.ts.map +1 -0
  825. package/packages/wrapper/dist/tmux-resolver.js.map +1 -0
  826. package/packages/wrapper/dist/tmux-wrapper.d.ts.map +1 -0
  827. package/packages/wrapper/dist/tmux-wrapper.js.map +1 -0
  828. package/packages/wrapper/dist/trajectory-integration.d.ts.map +1 -0
  829. package/packages/wrapper/dist/trajectory-integration.js.map +1 -0
  830. package/packages/wrapper/dist/wrapper-types.d.ts.map +1 -0
  831. package/packages/wrapper/dist/wrapper-types.js.map +1 -0
  832. package/packages/wrapper/package.json +6 -9
  833. package/packages/wrapper/src/__fixtures__/claude-outputs.ts +471 -0
  834. package/packages/wrapper/src/__fixtures__/codex-outputs.ts +99 -0
  835. package/packages/wrapper/src/__fixtures__/gemini-outputs.ts +151 -0
  836. package/packages/wrapper/src/__fixtures__/index.ts +47 -0
  837. package/packages/wrapper/src/auth-detection.ts +244 -0
  838. package/packages/wrapper/src/base-wrapper.test.ts +589 -0
  839. package/packages/wrapper/src/base-wrapper.ts +810 -0
  840. package/packages/wrapper/src/client.test.ts +262 -0
  841. package/packages/wrapper/src/client.ts +984 -0
  842. package/packages/wrapper/src/id-generator.test.ts +71 -0
  843. package/packages/wrapper/src/id-generator.ts +69 -0
  844. package/packages/wrapper/src/idle-detector.test.ts +418 -0
  845. package/packages/wrapper/src/idle-detector.ts +384 -0
  846. package/packages/wrapper/src/inbox.test.ts +233 -0
  847. package/packages/wrapper/src/inbox.ts +89 -0
  848. package/packages/wrapper/src/index.ts +170 -0
  849. package/packages/wrapper/src/parser.regression.test.ts +251 -0
  850. package/packages/wrapper/src/parser.test.ts +1359 -0
  851. package/packages/wrapper/src/parser.ts +1477 -0
  852. package/packages/wrapper/src/prompt-composer.test.ts +219 -0
  853. package/packages/wrapper/src/prompt-composer.ts +231 -0
  854. package/packages/wrapper/src/relay-pty-orchestrator.test.ts +1204 -0
  855. package/packages/wrapper/src/relay-pty-orchestrator.ts +2626 -0
  856. package/packages/wrapper/src/shared.test.ts +322 -0
  857. package/packages/wrapper/src/shared.ts +495 -0
  858. package/packages/wrapper/src/stuck-detector.test.ts +303 -0
  859. package/packages/wrapper/src/stuck-detector.ts +511 -0
  860. package/packages/wrapper/src/tmux-resolver.test.ts +104 -0
  861. package/packages/wrapper/src/tmux-resolver.ts +207 -0
  862. package/packages/wrapper/src/tmux-wrapper.test.ts +316 -0
  863. package/packages/wrapper/src/tmux-wrapper.ts +2095 -0
  864. package/packages/wrapper/src/trajectory-detection.test.ts +151 -0
  865. package/packages/wrapper/src/trajectory-integration.ts +1261 -0
  866. package/packages/wrapper/src/wrapper-types.ts +45 -0
  867. package/packages/wrapper/tsconfig.json +19 -0
  868. package/packages/wrapper/vitest.config.ts +9 -0
  869. package/scripts/build-cjs.mjs +23 -0
  870. package/scripts/postinstall.js +132 -0
  871. package/.cursor/mcp.json +0 -11
  872. package/.gitattributes +0 -3
  873. package/.gitleaks.toml +0 -26
  874. package/.mcp.json +0 -11
  875. package/.nvmrc +0 -1
  876. package/ARCHITECTURE.md +0 -1245
  877. package/CHANGELOG.md +0 -231
  878. package/TESTING.md +0 -278
  879. package/TRAIL_GIT_AUTH_FIX.md +0 -113
  880. package/scripts/demos/README.md +0 -79
  881. package/scripts/demos/server-capacity.sh +0 -69
  882. package/scripts/demos/sprint-planning.sh +0 -73
  883. package/scripts/hooks/install.sh +0 -16
  884. package/scripts/hooks/pre-commit +0 -60
  885. package/scripts/post-publish-verify/README.md +0 -80
  886. package/scripts/post-publish-verify/run-verify.sh +0 -127
  887. package/scripts/post-publish-verify/verify-install.sh +0 -249
  888. package/scripts/stress-test-orchestrator-integration.mts +0 -1366
  889. package/scripts/stress-test-orchestrator.mjs +0 -584
  890. package/scripts/stress-test-relay-pty.sh +0 -452
  891. package/scripts/test-interactive-terminal.sh +0 -248
  892. package/specs/PRIMITIVES_ROADMAP.md +0 -2154
  893. package/tests/benchmarks/protocol.bench.ts +0 -310
  894. package/turbo.json +0 -37
@@ -0,0 +1,1885 @@
1
+ /**
2
+ * Message router for the agent relay daemon.
3
+ * Handles routing messages between agents, topic subscriptions, and broadcast.
4
+ */
5
+
6
+ import { generateId } from '@agent-relay/wrapper';
7
+ import {
8
+ type Envelope,
9
+ type SendEnvelope,
10
+ type DeliverEnvelope,
11
+ type AckPayload,
12
+ type ErrorPayload,
13
+ type ShadowConfig,
14
+ type SpeakOnTrigger,
15
+ type EntityType,
16
+ PROTOCOL_VERSION,
17
+ } from '@agent-relay/protocol/types';
18
+ import type {
19
+ ChannelJoinPayload,
20
+ ChannelLeavePayload,
21
+ ChannelMessagePayload,
22
+ } from '@agent-relay/protocol/channels';
23
+ import type { StorageAdapter } from '@agent-relay/storage/adapter';
24
+ import type { AgentRegistry } from './agent-registry.js';
25
+ import { routerLog } from '@agent-relay/utils/logger';
26
+ import { RateLimiter, NoOpRateLimiter, type RateLimitConfig } from './rate-limiter.js';
27
+ import * as crypto from 'node:crypto';
28
+ import {
29
+ DeliveryTracker,
30
+ type DeliveryReliabilityOptions,
31
+ } from './delivery-tracker.js';
32
+ import type { ChannelMembershipStore } from './channel-membership-store.js';
33
+
34
+ export interface RoutableConnection {
35
+ id: string;
36
+ agentName?: string;
37
+ /** Entity type: 'agent' (default) or 'user' for human users */
38
+ entityType?: EntityType;
39
+ cli?: string;
40
+ program?: string;
41
+ model?: string;
42
+ task?: string;
43
+ workingDirectory?: string;
44
+ sessionId: string;
45
+ close(): void;
46
+ send(envelope: Envelope): boolean;
47
+ getNextSeq(topic: string, peer: string): number;
48
+ }
49
+
50
+ export interface RemoteAgentInfo {
51
+ name: string;
52
+ status: string;
53
+ daemonId: string;
54
+ daemonName: string;
55
+ machineId: string;
56
+ }
57
+
58
+ export interface CrossMachineHandler {
59
+ sendCrossMachineMessage(
60
+ targetDaemonId: string,
61
+ targetAgent: string,
62
+ fromAgent: string,
63
+ content: string,
64
+ metadata?: Record<string, unknown>
65
+ ): Promise<boolean>;
66
+ isRemoteAgent(agentName: string): RemoteAgentInfo | undefined;
67
+ /** Check if a user is on a remote machine (connected via cloud dashboard) */
68
+ isRemoteUser?(userName: string): RemoteAgentInfo | undefined;
69
+ }
70
+
71
+ interface ProcessingState {
72
+ startedAt: number;
73
+ messageId: string;
74
+ timer?: NodeJS.Timeout;
75
+ }
76
+
77
+ /** Internal shadow relationship with resolved defaults */
78
+ interface ShadowRelationship extends ShadowConfig {
79
+ shadowAgent: string;
80
+ }
81
+
82
+ export class Router {
83
+ private storage?: StorageAdapter;
84
+ private channelMembershipStore?: ChannelMembershipStore;
85
+ private connections: Map<string, RoutableConnection> = new Map(); // connectionId -> Connection
86
+ private agents: Map<string, RoutableConnection> = new Map(); // agentName -> Connection
87
+ private subscriptions: Map<string, Set<string>> = new Map(); // topic -> Set<agentName>
88
+ private processingAgents: Map<string, ProcessingState> = new Map(); // agentName -> processing state
89
+ private registry?: AgentRegistry;
90
+ private crossMachineHandler?: CrossMachineHandler;
91
+ private deliveryTracker: DeliveryTracker;
92
+
93
+ /** Shadow relationships: primaryAgent -> list of shadow configs */
94
+ private shadowsByPrimary: Map<string, ShadowRelationship[]> = new Map();
95
+ /** Reverse lookup: shadowAgent -> primaryAgent (for cleanup) */
96
+ private primaryByShadow: Map<string, string> = new Map();
97
+
98
+ /** Channel membership: channel -> Set of member names */
99
+ private channels: Map<string, Set<string>> = new Map();
100
+ /** User entities (human users, not agents) - supports multiple connections per user (multi-tab) */
101
+ private users: Map<string, Set<RoutableConnection>> = new Map();
102
+ /** Reverse lookup: member name -> Set of channels they're in */
103
+ private memberChannels: Map<string, Set<string>> = new Map();
104
+
105
+ /**
106
+ * Agents that are currently being spawned but haven't completed HELLO yet.
107
+ * Maps agent name to timestamp when spawn started.
108
+ * Messages sent to these agents will be queued for delivery after HELLO completes.
109
+ */
110
+ private spawningAgents: Map<string, number> = new Map();
111
+
112
+ /** Default timeout for processing indicator (30 seconds) */
113
+ private static readonly PROCESSING_TIMEOUT_MS = 30_000;
114
+
115
+ /** Timeout for spawning agent entries (60 seconds) */
116
+ private static readonly SPAWNING_TIMEOUT_MS = 60_000;
117
+
118
+ /** Callback when processing state changes (for real-time dashboard updates) */
119
+ private onProcessingStateChange?: () => void;
120
+
121
+ /** Rate limiter for per-agent throttling */
122
+ private rateLimiter: RateLimiter;
123
+
124
+ constructor(options: {
125
+ storage?: StorageAdapter;
126
+ delivery?: Partial<DeliveryReliabilityOptions>;
127
+ registry?: AgentRegistry;
128
+ onProcessingStateChange?: () => void;
129
+ crossMachineHandler?: CrossMachineHandler;
130
+ /** Rate limit configuration. Set to null to disable rate limiting. */
131
+ rateLimit?: Partial<RateLimitConfig> | null;
132
+ channelMembershipStore?: ChannelMembershipStore;
133
+ } = {}) {
134
+ this.storage = options.storage;
135
+ this.channelMembershipStore = options.channelMembershipStore;
136
+ this.registry = options.registry;
137
+ this.onProcessingStateChange = options.onProcessingStateChange;
138
+ this.crossMachineHandler = options.crossMachineHandler;
139
+ this.deliveryTracker = new DeliveryTracker({
140
+ storage: this.storage,
141
+ delivery: options.delivery,
142
+ getConnection: (id) => this.connections.get(id),
143
+ });
144
+ // Initialize rate limiter (null = disabled)
145
+ this.rateLimiter = options.rateLimit === null
146
+ ? new NoOpRateLimiter()
147
+ : new RateLimiter(options.rateLimit);
148
+ }
149
+
150
+ /**
151
+ * Restore channel memberships from persisted storage.
152
+ */
153
+ async restoreChannelMemberships(): Promise<void> {
154
+ if (!this.storage && !this.channelMembershipStore) return;
155
+
156
+ try {
157
+ if (this.channelMembershipStore) {
158
+ const memberships = await this.channelMembershipStore.loadMemberships();
159
+ for (const membership of memberships) {
160
+ this.handleMembershipUpdate({
161
+ channel: membership.channel,
162
+ member: membership.member,
163
+ action: 'join',
164
+ });
165
+ }
166
+ }
167
+
168
+ if (this.storage) {
169
+ const messages = await this.storage.getMessages({ order: 'asc' });
170
+ for (const msg of messages) {
171
+ const channel = msg.to;
172
+ const data = msg.data as Record<string, unknown> | undefined;
173
+ const membership = data?._channelMembership as { member?: string; action?: 'join' | 'leave' | 'invite' } | undefined;
174
+ if (!channel || !membership?.member) {
175
+ continue;
176
+ }
177
+ const action = membership.action ?? 'join';
178
+ this.handleMembershipUpdate({
179
+ channel,
180
+ member: membership.member,
181
+ action,
182
+ });
183
+ }
184
+ }
185
+ } catch (err) {
186
+ routerLog.error('Failed to restore channel memberships', { error: String(err) });
187
+ }
188
+ }
189
+
190
+ /**
191
+ * Set or update the cross-machine handler.
192
+ */
193
+ setCrossMachineHandler(handler: CrossMachineHandler): void {
194
+ this.crossMachineHandler = handler;
195
+ }
196
+
197
+ /**
198
+ * Mark an agent as spawning (before HELLO completes).
199
+ * Messages sent to this agent will be queued for delivery after registration.
200
+ */
201
+ markSpawning(agentName: string): void {
202
+ this.spawningAgents.set(agentName, Date.now());
203
+ routerLog.info(`Agent marked as spawning: ${agentName}`, {
204
+ currentSpawning: Array.from(this.spawningAgents.keys()),
205
+ });
206
+ // Clean up stale spawning entries
207
+ this.cleanupStaleSpawning();
208
+ }
209
+
210
+ /**
211
+ * Clear the spawning flag for an agent.
212
+ * Called when agent completes registration or spawn fails.
213
+ */
214
+ clearSpawning(agentName: string): void {
215
+ if (this.spawningAgents.delete(agentName)) {
216
+ routerLog.debug(`Agent spawning flag cleared: ${agentName}`);
217
+ }
218
+ }
219
+
220
+ /**
221
+ * Check if an agent is currently spawning.
222
+ */
223
+ isSpawning(agentName: string): boolean {
224
+ const timestamp = this.spawningAgents.get(agentName);
225
+ if (!timestamp) return false;
226
+ // Check if spawn has timed out
227
+ if (Date.now() - timestamp > Router.SPAWNING_TIMEOUT_MS) {
228
+ this.spawningAgents.delete(agentName);
229
+ return false;
230
+ }
231
+ return true;
232
+ }
233
+
234
+ /**
235
+ * Clean up spawning entries older than SPAWNING_TIMEOUT_MS.
236
+ */
237
+ private cleanupStaleSpawning(): void {
238
+ const now = Date.now();
239
+ for (const [name, timestamp] of this.spawningAgents) {
240
+ if (now - timestamp > Router.SPAWNING_TIMEOUT_MS) {
241
+ this.spawningAgents.delete(name);
242
+ routerLog.debug(`Cleaned up stale spawning entry: ${name}`);
243
+ }
244
+ }
245
+ }
246
+
247
+ /**
248
+ * Register a connection after successful handshake.
249
+ */
250
+ register(connection: RoutableConnection): void {
251
+ this.connections.set(connection.id, connection);
252
+
253
+ if (connection.agentName) {
254
+ const isUser = connection.entityType === 'user';
255
+
256
+ if (isUser) {
257
+ // Users can have multiple connections (e.g., multiple browser tabs)
258
+ // Track ALL connections per user so messages reach all tabs
259
+ if (!this.users.has(connection.agentName)) {
260
+ this.users.set(connection.agentName, new Set());
261
+ }
262
+ this.users.get(connection.agentName)!.add(connection);
263
+ routerLog.info(`User registered: ${connection.agentName} (${this.users.get(connection.agentName)!.size} connections)`);
264
+ } else {
265
+ // Handle existing agent connection with same name (disconnect old)
266
+ const existing = this.agents.get(connection.agentName);
267
+ if (existing && existing.id !== connection.id) {
268
+ routerLog.warn('Duplicate agent connection detected, closing old connection', {
269
+ agent: connection.agentName,
270
+ oldConnectionId: existing.id,
271
+ newConnectionId: connection.id,
272
+ });
273
+ // Send fatal error before closing to prevent reconnection loop
274
+ const errorEnvelope: Envelope<ErrorPayload> = {
275
+ v: PROTOCOL_VERSION,
276
+ type: 'ERROR',
277
+ id: generateId(),
278
+ ts: Date.now(),
279
+ payload: {
280
+ code: 'DUPLICATE_CONNECTION',
281
+ message: `Another agent with name "${connection.agentName}" connected. This connection will be closed.`,
282
+ fatal: true,
283
+ },
284
+ };
285
+ existing.send(errorEnvelope);
286
+ existing.close();
287
+ this.connections.delete(existing.id);
288
+ }
289
+ this.agents.set(connection.agentName, connection);
290
+ // Clear spawning flag now that agent has completed registration
291
+ this.clearSpawning(connection.agentName);
292
+ this.registry?.registerOrUpdate({
293
+ name: connection.agentName,
294
+ cli: connection.cli,
295
+ program: connection.program,
296
+ model: connection.model,
297
+ task: connection.task,
298
+ workingDirectory: connection.workingDirectory,
299
+ });
300
+ }
301
+ }
302
+ }
303
+
304
+ /**
305
+ * Unregister a connection.
306
+ */
307
+ unregister(connection: RoutableConnection): void {
308
+ this.connections.delete(connection.id);
309
+ if (connection.agentName) {
310
+ const isUser = connection.entityType === 'user';
311
+ let wasCurrentConnection = false;
312
+
313
+ if (isUser) {
314
+ const userConnections = this.users.get(connection.agentName);
315
+ if (userConnections) {
316
+ userConnections.delete(connection);
317
+ if (userConnections.size === 0) {
318
+ this.users.delete(connection.agentName);
319
+ wasCurrentConnection = true;
320
+ routerLog.info(`User fully unregistered: ${connection.agentName} (all connections closed)`);
321
+ } else {
322
+ routerLog.debug(`User connection closed: ${connection.agentName} (${userConnections.size} connections remaining)`);
323
+ }
324
+ }
325
+ } else {
326
+ const current = this.agents.get(connection.agentName);
327
+ if (current?.id === connection.id) {
328
+ this.agents.delete(connection.agentName);
329
+ wasCurrentConnection = true;
330
+ }
331
+ }
332
+
333
+ // Only clean up channel/subscription state if this was the current connection.
334
+ // If a new connection replaced this one, we don't want to remove channel memberships
335
+ // that the new connection should inherit.
336
+ if (wasCurrentConnection) {
337
+ // Remove from all subscriptions
338
+ for (const subscribers of this.subscriptions.values()) {
339
+ subscribers.delete(connection.agentName);
340
+ }
341
+
342
+ // Remove from all channels and notify remaining members
343
+ this.removeFromAllChannels(connection.agentName);
344
+
345
+ // Clean up shadow relationships
346
+ this.unbindShadow(connection.agentName);
347
+
348
+ // Clear processing state
349
+ this.clearProcessing(connection.agentName);
350
+ }
351
+ }
352
+
353
+ this.clearPendingForConnection(connection.id);
354
+ }
355
+
356
+ /**
357
+ * Remove a member from all channels they're in.
358
+ */
359
+ private removeFromAllChannels(memberName: string): void {
360
+ const memberChannelSet = this.memberChannels.get(memberName);
361
+ if (!memberChannelSet) return;
362
+
363
+ for (const channelName of memberChannelSet) {
364
+ const members = this.channels.get(channelName);
365
+ if (members) {
366
+ members.delete(memberName);
367
+ // Clean up empty channels
368
+ if (members.size === 0) {
369
+ this.channels.delete(channelName);
370
+ }
371
+ }
372
+ }
373
+ this.memberChannels.delete(memberName);
374
+ }
375
+
376
+ /**
377
+ * Subscribe an agent to a topic.
378
+ */
379
+ subscribe(agentName: string, topic: string): void {
380
+ let subscribers = this.subscriptions.get(topic);
381
+ if (!subscribers) {
382
+ subscribers = new Set();
383
+ this.subscriptions.set(topic, subscribers);
384
+ }
385
+ subscribers.add(agentName);
386
+ }
387
+
388
+ /**
389
+ * Unsubscribe an agent from a topic.
390
+ */
391
+ unsubscribe(agentName: string, topic: string): void {
392
+ const subscribers = this.subscriptions.get(topic);
393
+ if (subscribers) {
394
+ subscribers.delete(agentName);
395
+ if (subscribers.size === 0) {
396
+ this.subscriptions.delete(topic);
397
+ }
398
+ }
399
+ }
400
+
401
+ /**
402
+ * Bind a shadow agent to a primary agent.
403
+ * The shadow will receive copies of messages to/from the primary.
404
+ */
405
+ bindShadow(
406
+ shadowAgent: string,
407
+ primaryAgent: string,
408
+ options: {
409
+ speakOn?: SpeakOnTrigger[];
410
+ receiveIncoming?: boolean;
411
+ receiveOutgoing?: boolean;
412
+ } = {}
413
+ ): void {
414
+ // Clean up any existing shadow binding for this shadow
415
+ this.unbindShadow(shadowAgent);
416
+
417
+ const relationship: ShadowRelationship = {
418
+ shadowAgent,
419
+ primaryAgent,
420
+ speakOn: options.speakOn ?? ['EXPLICIT_ASK'],
421
+ receiveIncoming: options.receiveIncoming ?? true,
422
+ receiveOutgoing: options.receiveOutgoing ?? true,
423
+ };
424
+
425
+ // Add to primary's shadow list
426
+ let shadows = this.shadowsByPrimary.get(primaryAgent);
427
+ if (!shadows) {
428
+ shadows = [];
429
+ this.shadowsByPrimary.set(primaryAgent, shadows);
430
+ }
431
+ shadows.push(relationship);
432
+
433
+ // Set reverse lookup
434
+ this.primaryByShadow.set(shadowAgent, primaryAgent);
435
+
436
+ routerLog.info(`Shadow bound: ${shadowAgent} -> ${primaryAgent}`, { speakOn: relationship.speakOn });
437
+ }
438
+
439
+ /**
440
+ * Unbind a shadow agent from its primary.
441
+ */
442
+ unbindShadow(shadowAgent: string): void {
443
+ const primaryAgent = this.primaryByShadow.get(shadowAgent);
444
+ if (!primaryAgent) return;
445
+
446
+ // Remove from primary's shadow list
447
+ const shadows = this.shadowsByPrimary.get(primaryAgent);
448
+ if (shadows) {
449
+ const updatedShadows = shadows.filter(s => s.shadowAgent !== shadowAgent);
450
+ if (updatedShadows.length === 0) {
451
+ this.shadowsByPrimary.delete(primaryAgent);
452
+ } else {
453
+ this.shadowsByPrimary.set(primaryAgent, updatedShadows);
454
+ }
455
+ }
456
+
457
+ // Remove reverse lookup
458
+ this.primaryByShadow.delete(shadowAgent);
459
+
460
+ routerLog.info(`Shadow unbound: ${shadowAgent} from ${primaryAgent}`);
461
+ }
462
+
463
+ /**
464
+ * Get all shadows for a primary agent.
465
+ */
466
+ getShadowsForPrimary(primaryAgent: string): ShadowRelationship[] {
467
+ return this.shadowsByPrimary.get(primaryAgent) ?? [];
468
+ }
469
+
470
+ /**
471
+ * Get the primary agent for a shadow, if any.
472
+ */
473
+ getPrimaryForShadow(shadowAgent: string): string | undefined {
474
+ return this.primaryByShadow.get(shadowAgent);
475
+ }
476
+
477
+ /**
478
+ * Emit a trigger event for an agent's shadows.
479
+ * Shadows configured to speakOn this trigger will receive a notification.
480
+ * @param primaryAgent The agent whose shadows should be notified
481
+ * @param trigger The trigger event that occurred
482
+ * @param context Optional context data about the trigger
483
+ */
484
+ emitShadowTrigger(
485
+ primaryAgent: string,
486
+ trigger: SpeakOnTrigger,
487
+ context?: Record<string, unknown>
488
+ ): void {
489
+ const shadows = this.shadowsByPrimary.get(primaryAgent);
490
+ if (!shadows || shadows.length === 0) return;
491
+
492
+ for (const shadow of shadows) {
493
+ // Check if this shadow is configured to speak on this trigger
494
+ if (!shadow.speakOn.includes(trigger) && !shadow.speakOn.includes('ALL_MESSAGES')) {
495
+ continue;
496
+ }
497
+
498
+ const target = this.agents.get(shadow.shadowAgent);
499
+ if (!target) continue;
500
+
501
+ // Create a trigger notification envelope
502
+ const triggerEnvelope: SendEnvelope = {
503
+ v: PROTOCOL_VERSION,
504
+ type: 'SEND',
505
+ id: generateId(),
506
+ ts: Date.now(),
507
+ from: primaryAgent,
508
+ to: shadow.shadowAgent,
509
+ payload: {
510
+ kind: 'action',
511
+ body: `SHADOW_TRIGGER:${trigger}`,
512
+ data: {
513
+ _shadowTrigger: trigger,
514
+ _shadowOf: primaryAgent,
515
+ _triggerContext: context,
516
+ },
517
+ },
518
+ };
519
+
520
+ const deliver = this.createDeliverEnvelope(
521
+ primaryAgent,
522
+ shadow.shadowAgent,
523
+ triggerEnvelope,
524
+ target
525
+ );
526
+ const sent = target.send(deliver);
527
+ if (sent) {
528
+ this.trackDelivery(target, deliver);
529
+ routerLog.debug(`Shadow trigger ${trigger} sent to ${shadow.shadowAgent}`, { primary: primaryAgent });
530
+ // Set processing state for triggered shadows - they're expected to respond
531
+ this.setProcessing(shadow.shadowAgent, deliver.id);
532
+ }
533
+ }
534
+ }
535
+
536
+ /**
537
+ * Check if a shadow should speak based on a specific trigger.
538
+ */
539
+ shouldShadowSpeak(shadowAgent: string, trigger: SpeakOnTrigger): boolean {
540
+ const primaryAgent = this.primaryByShadow.get(shadowAgent);
541
+ if (!primaryAgent) return true; // Not a shadow, can always speak
542
+
543
+ const shadows = this.shadowsByPrimary.get(primaryAgent);
544
+ if (!shadows) return true;
545
+
546
+ const relationship = shadows.find(s => s.shadowAgent === shadowAgent);
547
+ if (!relationship) return true;
548
+
549
+ return relationship.speakOn.includes(trigger) || relationship.speakOn.includes('ALL_MESSAGES');
550
+ }
551
+
552
+ /**
553
+ * Route a SEND message to its destination(s).
554
+ */
555
+ route(from: RoutableConnection, envelope: SendEnvelope): void {
556
+ // Check if this is a cross-machine message (injected from cloud)
557
+ const isCrossMachine = envelope.payload?.data?._crossMachine === true;
558
+
559
+ // Use envelope.from for cross-machine messages where the connection is the recipient,
560
+ // otherwise use the connection's agent name (normal local messages)
561
+ const senderName = isCrossMachine ? envelope.from : (envelope.from || from.agentName);
562
+ if (!senderName) {
563
+ routerLog.warn('Dropping message - sender has no name');
564
+ return;
565
+ }
566
+
567
+ // Skip rate limiting, processing state, and send recording for cross-machine messages
568
+ // These only apply to local agents sending messages
569
+ if (!isCrossMachine) {
570
+ // Check rate limit
571
+ if (!this.rateLimiter.tryAcquire(senderName)) {
572
+ routerLog.warn(`Rate limited: ${senderName}`);
573
+ return;
574
+ }
575
+
576
+ // Agent is responding - clear their processing state
577
+ this.clearProcessing(senderName);
578
+
579
+ this.registry?.recordSend(senderName);
580
+ }
581
+
582
+ const to = envelope.to;
583
+ const topic = envelope.topic;
584
+
585
+ routerLog.debug(`Route ${senderName} -> ${to}`, { preview: typeof envelope.payload.body === 'string' ? envelope.payload.body.substring(0, 50) : JSON.stringify(envelope.payload.body)?.substring(0, 50) });
586
+
587
+ if (to === '*') {
588
+ // Broadcast to all (except sender)
589
+ this.broadcast(senderName, envelope, topic);
590
+ } else if (to === '@users') {
591
+ // Broadcast to all human users only (not agents)
592
+ this.broadcastToUsers(senderName, envelope);
593
+ } else if (to) {
594
+ // Direct message
595
+ this.sendDirect(senderName, to, envelope);
596
+ }
597
+
598
+ // Route copies to shadows of the sender (outgoing messages)
599
+ this.routeToShadows(senderName, envelope, 'outgoing');
600
+
601
+ // Route copies to shadows of the recipient (incoming messages)
602
+ if (to && to !== '*') {
603
+ this.routeToShadows(to, envelope, 'incoming', senderName);
604
+ }
605
+ }
606
+
607
+ /**
608
+ * Route a copy of a message to shadows of an agent.
609
+ * @param primaryAgent The primary agent whose shadows should receive the message
610
+ * @param envelope The original message envelope
611
+ * @param direction Whether this is an 'incoming' or 'outgoing' message for the primary
612
+ * @param actualFrom Override the 'from' field (for incoming messages, use original sender)
613
+ */
614
+ private routeToShadows(
615
+ primaryAgent: string,
616
+ envelope: SendEnvelope,
617
+ direction: 'incoming' | 'outgoing',
618
+ actualFrom?: string
619
+ ): void {
620
+ const shadows = this.shadowsByPrimary.get(primaryAgent);
621
+ if (!shadows || shadows.length === 0) return;
622
+
623
+ for (const shadow of shadows) {
624
+ // Check if shadow wants this direction
625
+ if (direction === 'incoming' && shadow.receiveIncoming === false) continue;
626
+ if (direction === 'outgoing' && shadow.receiveOutgoing === false) continue;
627
+
628
+ // Don't send to self
629
+ if (shadow.shadowAgent === (actualFrom ?? primaryAgent)) continue;
630
+
631
+ const target = this.agents.get(shadow.shadowAgent);
632
+ if (!target) continue;
633
+
634
+ // Create a shadow copy envelope with metadata indicating it's a shadow copy
635
+ const shadowEnvelope: SendEnvelope = {
636
+ ...envelope,
637
+ payload: {
638
+ ...envelope.payload,
639
+ data: {
640
+ ...envelope.payload.data,
641
+ _shadowCopy: true,
642
+ _shadowOf: primaryAgent,
643
+ _shadowDirection: direction,
644
+ },
645
+ },
646
+ };
647
+
648
+ const deliver = this.createDeliverEnvelope(
649
+ actualFrom ?? primaryAgent,
650
+ shadow.shadowAgent,
651
+ shadowEnvelope,
652
+ target
653
+ );
654
+ const sent = target.send(deliver);
655
+ if (sent) {
656
+ this.trackDelivery(target, deliver);
657
+ routerLog.debug(`Shadow copy to ${shadow.shadowAgent}`, { direction, primary: primaryAgent });
658
+ // Note: Don't set processing state for shadow copies - shadow stays passive
659
+ }
660
+ }
661
+ }
662
+
663
+ /**
664
+ * Send a direct message to a specific agent.
665
+ *
666
+ * If the target agent is offline but known (has connected before),
667
+ * the message is persisted for delivery when the agent reconnects.
668
+ * This prevents silent message drops during brief disconnections or spawn timing issues.
669
+ */
670
+ private sendDirect(
671
+ from: string,
672
+ to: string,
673
+ envelope: SendEnvelope
674
+ ): boolean {
675
+ // Check for agent first, then user connections
676
+ const agentTarget = this.agents.get(to);
677
+ const userConnections = this.users.get(to);
678
+ const hasTarget = agentTarget || (userConnections && userConnections.size > 0);
679
+
680
+ // If target not found locally, check remote
681
+ if (!hasTarget) {
682
+ const remoteAgent = this.crossMachineHandler?.isRemoteAgent(to);
683
+ if (remoteAgent) {
684
+ routerLog.info(`Routing to remote agent: ${to}`, { daemonName: remoteAgent.daemonName });
685
+ return this.sendToRemoteAgent(from, to, envelope, remoteAgent);
686
+ }
687
+ // Also check if it's a remote user (human connected via cloud dashboard)
688
+ const remoteUser = this.crossMachineHandler?.isRemoteUser?.(to);
689
+ if (remoteUser) {
690
+ routerLog.info(`Routing to remote user: ${to}`, { daemonName: remoteUser.daemonName });
691
+ return this.sendToRemoteAgent(from, to, envelope, remoteUser);
692
+ }
693
+
694
+ // Check if this is a known agent (has connected before) - queue for later delivery
695
+ // This prevents message drops during brief disconnections or spawn timing issues
696
+ if (this.registry?.has(to)) {
697
+ routerLog.info(`Target "${to}" offline but known, queueing message for delivery on reconnect`);
698
+ this.persistMessageForOfflineAgent(from, to, envelope);
699
+ return true; // Message accepted (queued), not dropped
700
+ }
701
+
702
+ // Check if agent is currently spawning (pre-HELLO) - queue for delivery after registration
703
+ // This handles the race condition between spawn completion and HELLO handshake
704
+ const spawning = this.isSpawning(to);
705
+ routerLog.debug(`Spawning check for "${to}": ${spawning}`, {
706
+ spawningAgents: Array.from(this.spawningAgents.keys()),
707
+ hasStorage: !!this.storage,
708
+ });
709
+ if (spawning) {
710
+ routerLog.info(`Target "${to}" is spawning, queueing message for delivery after registration`);
711
+ this.persistMessageForOfflineAgent(from, to, envelope);
712
+ return true; // Message accepted (queued), not dropped
713
+ }
714
+
715
+ routerLog.warn(`Target "${to}" not found and unknown`, { availableAgents: Array.from(this.agents.keys()), spawningAgents: Array.from(this.spawningAgents.keys()) });
716
+ return false;
717
+ }
718
+
719
+ // For user targets, send to ALL connections (multi-tab support)
720
+ if (userConnections && userConnections.size > 0) {
721
+ let anySent = false;
722
+ for (const userConn of userConnections) {
723
+ const deliver = this.createDeliverEnvelope(from, to, envelope, userConn);
724
+ const sent = userConn.send(deliver);
725
+ if (sent) anySent = true;
726
+ routerLog.debug(`Delivered ${from} -> ${to} (user connection ${userConn.id})`, {
727
+ success: sent,
728
+ preview: typeof envelope.payload.body === 'string' ? envelope.payload.body.substring(0, 40) : JSON.stringify(envelope.payload.body)?.substring(0, 40),
729
+ });
730
+ // Persist only once (for the first connection)
731
+ if (userConn === [...userConnections][0]) {
732
+ this.persistDeliverEnvelope(deliver);
733
+ }
734
+ if (sent) {
735
+ this.trackDelivery(userConn, deliver);
736
+ }
737
+ }
738
+ if (anySent) {
739
+ this.registry?.recordReceive(to);
740
+ }
741
+ return anySent;
742
+ }
743
+
744
+ // For agent targets, send to single connection
745
+ const target = agentTarget!;
746
+ const deliver = this.createDeliverEnvelope(from, to, envelope, target);
747
+ const sent = target.send(deliver);
748
+ routerLog.debug(`Delivered ${from} -> ${to}`, { success: sent, preview: typeof envelope.payload.body === 'string' ? envelope.payload.body.substring(0, 40) : JSON.stringify(envelope.payload.body)?.substring(0, 40) });
749
+ this.persistDeliverEnvelope(deliver);
750
+ if (sent) {
751
+ this.trackDelivery(target, deliver);
752
+ this.registry?.recordReceive(to);
753
+ this.setProcessing(to, deliver.id);
754
+ }
755
+ return sent;
756
+ }
757
+
758
+ /**
759
+ * Send a message to an agent on a remote machine via cloud.
760
+ */
761
+ private sendToRemoteAgent(
762
+ from: string,
763
+ to: string,
764
+ envelope: SendEnvelope,
765
+ remoteAgent: RemoteAgentInfo
766
+ ): boolean {
767
+ if (!this.crossMachineHandler) {
768
+ routerLog.warn('Cross-machine handler not available');
769
+ return false;
770
+ }
771
+
772
+ // Send asynchronously via cloud
773
+ this.crossMachineHandler.sendCrossMachineMessage(
774
+ remoteAgent.daemonId,
775
+ to,
776
+ from,
777
+ envelope.payload.body,
778
+ {
779
+ topic: envelope.topic,
780
+ thread: envelope.payload.thread,
781
+ kind: envelope.payload.kind,
782
+ data: envelope.payload.data,
783
+ originalId: envelope.id,
784
+ }
785
+ ).then((sent) => {
786
+ if (sent) {
787
+ routerLog.info(`Cross-machine message sent to ${to}`, { daemonName: remoteAgent.daemonName });
788
+ // Persist as cross-machine message
789
+ this.storage?.saveMessage({
790
+ id: envelope.id || `cross-${Date.now()}`,
791
+ ts: Date.now(),
792
+ from,
793
+ to,
794
+ topic: envelope.topic,
795
+ kind: envelope.payload.kind,
796
+ body: envelope.payload.body,
797
+ data: {
798
+ ...envelope.payload.data,
799
+ _crossMachine: true,
800
+ _targetDaemon: remoteAgent.daemonId,
801
+ _targetDaemonName: remoteAgent.daemonName,
802
+ },
803
+ thread: envelope.payload.thread,
804
+ status: 'unread',
805
+ is_urgent: false,
806
+ is_broadcast: false,
807
+ }).catch(err => routerLog.error('Failed to persist cross-machine message', { error: String(err) }));
808
+ } else {
809
+ routerLog.error(`Failed to send cross-machine message to ${to}`);
810
+ }
811
+ }).catch(err => {
812
+ routerLog.error('Cross-machine send error', { error: String(err) });
813
+ });
814
+
815
+ // Return true immediately - message is queued
816
+ return true;
817
+ }
818
+
819
+ /**
820
+ * Broadcast to all agents (optionally filtered by topic subscription).
821
+ */
822
+ private broadcast(
823
+ from: string,
824
+ envelope: SendEnvelope,
825
+ topic?: string
826
+ ): void {
827
+ // Build recipients list from both agents and users
828
+ const recipients = topic
829
+ ? this.subscriptions.get(topic) ?? new Set()
830
+ : new Set([...this.agents.keys(), ...this.users.keys()]);
831
+
832
+ for (const recipientName of recipients) {
833
+ if (recipientName === from) continue; // Don't send to self
834
+
835
+ // Check agents first
836
+ const agentTarget = this.agents.get(recipientName);
837
+ if (agentTarget) {
838
+ const deliver = this.createDeliverEnvelope(from, recipientName, envelope, agentTarget);
839
+ const sent = agentTarget.send(deliver);
840
+ this.persistDeliverEnvelope(deliver, true); // Mark as broadcast
841
+ if (sent) {
842
+ this.trackDelivery(agentTarget, deliver);
843
+ this.registry?.recordReceive(recipientName);
844
+ this.setProcessing(recipientName, deliver.id);
845
+ }
846
+ continue;
847
+ }
848
+
849
+ // Check user connections (send to all connections for multi-tab)
850
+ const userConnections = this.users.get(recipientName);
851
+ if (userConnections && userConnections.size > 0) {
852
+ let persisted = false;
853
+ for (const userConn of userConnections) {
854
+ const deliver = this.createDeliverEnvelope(from, recipientName, envelope, userConn);
855
+ const sent = userConn.send(deliver);
856
+ if (!persisted) {
857
+ this.persistDeliverEnvelope(deliver, true); // Mark as broadcast, persist once
858
+ persisted = true;
859
+ }
860
+ if (sent) {
861
+ this.trackDelivery(userConn, deliver);
862
+ this.registry?.recordReceive(recipientName);
863
+ }
864
+ }
865
+ }
866
+ }
867
+ }
868
+
869
+ /**
870
+ * Broadcast a message to all human users (not agents).
871
+ * Used for system notifications that only humans should see.
872
+ */
873
+ private broadcastToUsers(
874
+ from: string,
875
+ envelope: SendEnvelope
876
+ ): void {
877
+ for (const [userName, userConnections] of this.users) {
878
+ if (userName === from) continue; // Don't send to self
879
+ if (userConnections.size === 0) continue;
880
+
881
+ let persisted = false;
882
+ for (const userConn of userConnections) {
883
+ const deliver = this.createDeliverEnvelope(from, userName, envelope, userConn);
884
+ const sent = userConn.send(deliver);
885
+ if (!persisted) {
886
+ this.persistDeliverEnvelope(deliver, true); // Mark as broadcast, persist once
887
+ persisted = true;
888
+ }
889
+ if (sent) {
890
+ this.trackDelivery(userConn, deliver);
891
+ this.registry?.recordReceive(userName);
892
+ routerLog.debug(`Broadcast to user ${userName} (connection ${userConn.id})`);
893
+ }
894
+ }
895
+ }
896
+ }
897
+
898
+ /**
899
+ * Create a DELIVER envelope from a SEND.
900
+ */
901
+ private createDeliverEnvelope(
902
+ from: string,
903
+ to: string,
904
+ original: SendEnvelope,
905
+ target: RoutableConnection
906
+ ): DeliverEnvelope {
907
+ // Preserve the original 'to' field for broadcasts so agents know to reply to '*'
908
+ const originalTo = original.to;
909
+
910
+ return {
911
+ v: PROTOCOL_VERSION,
912
+ type: 'DELIVER',
913
+ id: generateId(),
914
+ ts: Date.now(),
915
+ from,
916
+ to,
917
+ topic: original.topic,
918
+ payload: original.payload,
919
+ payload_meta: original.payload_meta,
920
+ delivery: {
921
+ seq: target.getNextSeq(original.topic ?? 'default', from),
922
+ session_id: target.sessionId,
923
+ originalTo: originalTo !== to ? originalTo : undefined, // Only include if different
924
+ },
925
+ };
926
+ }
927
+
928
+ /**
929
+ * Persist a delivered message if storage is configured.
930
+ */
931
+ private persistDeliverEnvelope(envelope: DeliverEnvelope, isBroadcast: boolean = false): void {
932
+ if (!this.storage) return;
933
+
934
+ this.storage.saveMessage({
935
+ id: envelope.id,
936
+ ts: envelope.ts,
937
+ from: envelope.from ?? 'unknown',
938
+ to: envelope.to ?? 'unknown',
939
+ topic: envelope.topic,
940
+ kind: envelope.payload.kind,
941
+ body: envelope.payload.body,
942
+ data: envelope.payload.data,
943
+ payloadMeta: envelope.payload_meta,
944
+ thread: envelope.payload.thread,
945
+ deliverySeq: envelope.delivery.seq,
946
+ deliverySessionId: envelope.delivery.session_id,
947
+ sessionId: envelope.delivery.session_id,
948
+ status: 'unread',
949
+ is_urgent: false,
950
+ is_broadcast: isBroadcast || envelope.to === '*',
951
+ }).catch((err) => {
952
+ routerLog.error('Failed to persist message', { error: String(err) });
953
+ });
954
+ }
955
+
956
+ /**
957
+ * Persist a message for an offline agent.
958
+ * Called when a message is sent to a known agent that is not currently connected.
959
+ * The message is marked with _offlineQueued and will be delivered when the agent reconnects.
960
+ */
961
+ private persistMessageForOfflineAgent(from: string, to: string, envelope: SendEnvelope): void {
962
+ if (!this.storage) {
963
+ routerLog.warn('Cannot queue offline message: no storage configured');
964
+ return;
965
+ }
966
+
967
+ routerLog.info(`Persisting offline message for "${to}"`, {
968
+ from,
969
+ messageId: envelope.id,
970
+ bodyPreview: typeof envelope.payload.body === 'string' ? envelope.payload.body.substring(0, 50) : JSON.stringify(envelope.payload.body)?.substring(0, 50),
971
+ });
972
+
973
+ this.storage.saveMessage({
974
+ id: envelope.id || generateId(),
975
+ ts: Date.now(),
976
+ from,
977
+ to,
978
+ topic: envelope.topic,
979
+ kind: envelope.payload.kind,
980
+ body: envelope.payload.body,
981
+ data: {
982
+ ...envelope.payload.data,
983
+ _offlineQueued: true, // Mark as queued for offline delivery
984
+ _queuedAt: Date.now(),
985
+ },
986
+ payloadMeta: envelope.payload_meta,
987
+ thread: envelope.payload.thread,
988
+ status: 'unread', // Unread = pending delivery
989
+ is_urgent: false,
990
+ is_broadcast: false,
991
+ }).catch((err) => {
992
+ routerLog.error('Failed to persist offline message', { error: String(err), to });
993
+ });
994
+ }
995
+
996
+ /**
997
+ * Deliver pending messages to an agent that just connected.
998
+ * Queries for unread messages addressed to this agent that were queued while offline.
999
+ * This handles messages that were sent while the agent was offline.
1000
+ */
1001
+ async deliverPendingMessages(connection: RoutableConnection): Promise<void> {
1002
+ const agentName = connection.agentName;
1003
+ if (!agentName) return;
1004
+ if (!this.storage?.getMessages) return;
1005
+
1006
+ try {
1007
+ // Query for unread messages addressed to this agent
1008
+ const pendingMessages = await this.storage.getMessages({
1009
+ to: agentName,
1010
+ unreadOnly: true,
1011
+ order: 'asc', // Deliver oldest first
1012
+ });
1013
+
1014
+ // Filter to only include offline-queued messages (not already-delivered unacked messages)
1015
+ const offlineMessages = pendingMessages.filter(
1016
+ msg => msg.data?._offlineQueued === true
1017
+ ).sort((a, b) => a.ts - b.ts);
1018
+
1019
+ if (offlineMessages.length === 0) return;
1020
+
1021
+ routerLog.info(`Delivering ${offlineMessages.length} pending messages to ${agentName}`);
1022
+
1023
+ for (const msg of offlineMessages) {
1024
+ // Create deliver envelope
1025
+ const deliverEnvelope: DeliverEnvelope = {
1026
+ v: PROTOCOL_VERSION,
1027
+ type: 'DELIVER',
1028
+ id: generateId(),
1029
+ ts: Date.now(),
1030
+ from: msg.from,
1031
+ to: agentName,
1032
+ topic: msg.topic,
1033
+ payload: {
1034
+ body: msg.body,
1035
+ kind: msg.kind,
1036
+ data: msg.data,
1037
+ thread: msg.thread,
1038
+ },
1039
+ payload_meta: msg.payloadMeta,
1040
+ delivery: {
1041
+ seq: connection.getNextSeq(msg.topic ?? 'default', msg.from),
1042
+ session_id: connection.sessionId,
1043
+ },
1044
+ };
1045
+
1046
+ const sent = connection.send(deliverEnvelope);
1047
+ if (sent) {
1048
+ this.trackDelivery(connection, deliverEnvelope);
1049
+ this.registry?.recordReceive(agentName);
1050
+ this.setProcessing(agentName, deliverEnvelope.id);
1051
+
1052
+ // Mark original message as delivered (update status)
1053
+ if (this.storage.updateMessageStatus) {
1054
+ await this.storage.updateMessageStatus(msg.id, 'read');
1055
+ }
1056
+
1057
+ routerLog.info(`Delivered pending message to ${agentName}`, {
1058
+ from: msg.from,
1059
+ preview: msg.body.substring(0, 40),
1060
+ });
1061
+ } else {
1062
+ routerLog.warn(`Failed to deliver pending message to ${agentName}`);
1063
+ }
1064
+ }
1065
+ } catch (err) {
1066
+ routerLog.error('Failed to deliver pending messages', { error: String(err), agentName });
1067
+ }
1068
+ }
1069
+
1070
+ /**
1071
+ * Get list of connected agent names.
1072
+ */
1073
+ getAgents(): string[] {
1074
+ return Array.from(this.agents.keys());
1075
+ }
1076
+
1077
+ /**
1078
+ * Get connection by agent name.
1079
+ */
1080
+ getConnection(agentName: string): RoutableConnection | undefined {
1081
+ return this.agents.get(agentName);
1082
+ }
1083
+
1084
+ /**
1085
+ * Force remove an agent from the router (used when process dies without clean disconnect).
1086
+ * This cleans up the agent's connection and subscriptions without needing the connection object.
1087
+ */
1088
+ forceRemoveAgent(agentName: string): boolean {
1089
+ const connection = this.agents.get(agentName);
1090
+ if (!connection) {
1091
+ routerLog.debug(`forceRemoveAgent: agent ${agentName} not found in router`);
1092
+ return false;
1093
+ }
1094
+
1095
+ routerLog.info(`Force removing stale agent: ${agentName}`);
1096
+
1097
+ // Remove from agents map
1098
+ this.agents.delete(agentName);
1099
+
1100
+ // Remove from all channel subscriptions
1101
+ for (const [channel, subscribers] of this.subscriptions) {
1102
+ if (subscribers.delete(agentName)) {
1103
+ routerLog.debug(`Removed ${agentName} from channel ${channel}`);
1104
+ }
1105
+ }
1106
+
1107
+ // Remove from connections map
1108
+ this.connections.delete(connection.id);
1109
+
1110
+ // Clear any pending deliveries
1111
+ this.deliveryTracker.clearPendingForConnection(connection.id);
1112
+
1113
+ // Clean up channel memberships (same as unregister)
1114
+ this.removeFromAllChannels(agentName);
1115
+
1116
+ // Clean up shadow relationships
1117
+ this.unbindShadow(agentName);
1118
+
1119
+ // Clear processing state
1120
+ this.clearProcessing(agentName);
1121
+
1122
+ return true;
1123
+ }
1124
+
1125
+ /**
1126
+ * Get number of active connections.
1127
+ */
1128
+ get connectionCount(): number {
1129
+ return this.connections.size;
1130
+ }
1131
+
1132
+ get pendingDeliveryCount(): number {
1133
+ return this.deliveryTracker.pendingCount;
1134
+ }
1135
+
1136
+ /**
1137
+ * Get rate limiter statistics.
1138
+ */
1139
+ getRateLimiterStats(): { agentCount: number; config: RateLimitConfig } {
1140
+ return this.rateLimiter.getStats();
1141
+ }
1142
+
1143
+ /**
1144
+ * Reset rate limit for a specific agent (admin operation).
1145
+ */
1146
+ resetRateLimit(agentName: string): void {
1147
+ this.rateLimiter.reset(agentName);
1148
+ }
1149
+
1150
+ /**
1151
+ * Get list of agents currently processing (thinking).
1152
+ * Returns an object with agent names as keys and processing info as values.
1153
+ */
1154
+ getProcessingAgents(): Record<string, { startedAt: number; messageId: string }> {
1155
+ const result: Record<string, { startedAt: number; messageId: string }> = {};
1156
+ for (const [name, state] of this.processingAgents.entries()) {
1157
+ result[name] = { startedAt: state.startedAt, messageId: state.messageId };
1158
+ }
1159
+ return result;
1160
+ }
1161
+
1162
+ /**
1163
+ * Check if a specific agent is processing.
1164
+ */
1165
+ isAgentProcessing(agentName: string): boolean {
1166
+ return this.processingAgents.has(agentName);
1167
+ }
1168
+
1169
+ /**
1170
+ * Mark an agent as processing (called when they receive a message).
1171
+ */
1172
+ private setProcessing(agentName: string, messageId: string): void {
1173
+ // Clear any existing processing state
1174
+ this.clearProcessing(agentName);
1175
+
1176
+ const timer = setTimeout(() => {
1177
+ this.clearProcessing(agentName);
1178
+ routerLog.warn(`Processing timeout for ${agentName}`);
1179
+ }, Router.PROCESSING_TIMEOUT_MS);
1180
+
1181
+ this.processingAgents.set(agentName, {
1182
+ startedAt: Date.now(),
1183
+ messageId,
1184
+ timer,
1185
+ });
1186
+ routerLog.debug(`${agentName} started processing`, { messageId });
1187
+ this.onProcessingStateChange?.();
1188
+ }
1189
+
1190
+ /**
1191
+ * Clear processing state for an agent (called when they send a message).
1192
+ */
1193
+ private clearProcessing(agentName: string): void {
1194
+ const state = this.processingAgents.get(agentName);
1195
+ if (state) {
1196
+ if (state.timer) {
1197
+ clearTimeout(state.timer);
1198
+ }
1199
+ this.processingAgents.delete(agentName);
1200
+ routerLog.debug(`${agentName} finished processing`);
1201
+ this.onProcessingStateChange?.();
1202
+ }
1203
+ }
1204
+
1205
+ /**
1206
+ * Handle ACK for previously delivered messages.
1207
+ */
1208
+ handleAck(connection: RoutableConnection, envelope: Envelope<AckPayload>): void {
1209
+ const ackId = envelope.payload.ack_id;
1210
+ this.deliveryTracker.handleAck(connection.id, ackId);
1211
+ }
1212
+
1213
+ /**
1214
+ * Clear pending deliveries for a connection (e.g., on disconnect).
1215
+ */
1216
+ clearPendingForConnection(connectionId: string): void {
1217
+ this.deliveryTracker.clearPendingForConnection(connectionId);
1218
+ }
1219
+
1220
+ /**
1221
+ * Track a delivery and schedule retries until ACKed or TTL/attempts exhausted.
1222
+ */
1223
+ private trackDelivery(target: RoutableConnection, deliver: DeliverEnvelope): void {
1224
+ this.deliveryTracker.track(target, deliver);
1225
+ }
1226
+
1227
+ /**
1228
+ * Broadcast a system message to all connected agents.
1229
+ * Used for system notifications like agent death announcements.
1230
+ */
1231
+ broadcastSystemMessage(message: string, data?: Record<string, unknown>): void {
1232
+ const envelope: SendEnvelope = {
1233
+ v: PROTOCOL_VERSION,
1234
+ type: 'SEND',
1235
+ id: generateId(),
1236
+ ts: Date.now(),
1237
+ from: '_system',
1238
+ to: '*',
1239
+ payload: {
1240
+ kind: 'message',
1241
+ body: message,
1242
+ data: {
1243
+ ...data,
1244
+ _isSystemMessage: true,
1245
+ },
1246
+ },
1247
+ };
1248
+
1249
+ // Broadcast to all agents
1250
+ for (const [agentName, connection] of this.agents.entries()) {
1251
+ const deliver = this.createDeliverEnvelope('_system', agentName, envelope, connection);
1252
+ const sent = connection.send(deliver);
1253
+ if (sent) {
1254
+ routerLog.debug(`System broadcast sent to ${agentName}`);
1255
+ }
1256
+ }
1257
+ }
1258
+
1259
+ /**
1260
+ * Replay any pending (unacked) messages for a resumed session.
1261
+ */
1262
+ async replayPending(connection: RoutableConnection): Promise<void> {
1263
+ if (!this.storage?.getPendingMessagesForSession || !connection.agentName) {
1264
+ return;
1265
+ }
1266
+
1267
+ const pending = await this.storage.getPendingMessagesForSession(connection.agentName, connection.sessionId);
1268
+ if (!pending.length) return;
1269
+
1270
+ routerLog.info(`Replaying ${pending.length} messages to ${connection.agentName}`);
1271
+
1272
+ for (const msg of pending) {
1273
+ const deliver: DeliverEnvelope = {
1274
+ v: PROTOCOL_VERSION,
1275
+ type: 'DELIVER',
1276
+ id: msg.id,
1277
+ ts: msg.ts,
1278
+ from: msg.from,
1279
+ to: msg.to,
1280
+ topic: msg.topic,
1281
+ payload: {
1282
+ kind: msg.kind,
1283
+ body: msg.body,
1284
+ data: msg.data,
1285
+ thread: msg.thread,
1286
+ },
1287
+ payload_meta: msg.payloadMeta,
1288
+ delivery: {
1289
+ seq: msg.deliverySeq ?? connection.getNextSeq(msg.topic ?? 'default', msg.from),
1290
+ session_id: msg.deliverySessionId ?? connection.sessionId,
1291
+ },
1292
+ };
1293
+
1294
+ const sent = connection.send(deliver);
1295
+ if (sent) {
1296
+ this.trackDelivery(connection, deliver);
1297
+ }
1298
+ }
1299
+ }
1300
+
1301
+ // ==================== Channel Methods ====================
1302
+
1303
+ /**
1304
+ * Handle a CHANNEL_JOIN message.
1305
+ * Adds the member to the channel and notifies existing members.
1306
+ * If payload.member is set, adds that member (admin mode).
1307
+ * Otherwise, adds the connection's agent name.
1308
+ */
1309
+ handleChannelJoin(
1310
+ connection: RoutableConnection,
1311
+ envelope: Envelope<ChannelJoinPayload>
1312
+ ): void {
1313
+ // Use payload.member if provided (admin mode), otherwise use connection's name
1314
+ const memberName = envelope.payload.member ?? connection.agentName;
1315
+ if (!memberName) {
1316
+ routerLog.warn('CHANNEL_JOIN from connection without name and no member specified');
1317
+ return;
1318
+ }
1319
+
1320
+ const channel = envelope.payload.channel;
1321
+ const isAdminJoin = Boolean(envelope.payload.member);
1322
+
1323
+ // Get or create channel
1324
+ let members = this.channels.get(channel);
1325
+ if (!members) {
1326
+ members = new Set();
1327
+ this.channels.set(channel, members);
1328
+ }
1329
+
1330
+ // Check if already a member
1331
+ if (members.has(memberName)) {
1332
+ routerLog.debug(`${memberName} already in ${channel}`);
1333
+ return;
1334
+ }
1335
+
1336
+ // Only notify existing members for non-admin joins (agents joining themselves)
1337
+ // Admin joins are silent to avoid spamming notifications when syncing
1338
+ if (!isAdminJoin) {
1339
+ const existingMembers = members ? Array.from(members) : [];
1340
+ for (const existingMember of existingMembers) {
1341
+ const memberConn = this.getConnectionByName(existingMember);
1342
+ if (memberConn) {
1343
+ const joinNotification: Envelope<ChannelJoinPayload> = {
1344
+ v: PROTOCOL_VERSION,
1345
+ type: 'CHANNEL_JOIN',
1346
+ id: generateId(),
1347
+ ts: Date.now(),
1348
+ from: memberName,
1349
+ payload: envelope.payload,
1350
+ };
1351
+ memberConn.send(joinNotification);
1352
+ }
1353
+ }
1354
+ }
1355
+
1356
+ const added = this.addChannelMember(channel, memberName, { persist: true });
1357
+ if (!added) {
1358
+ routerLog.debug(`${memberName} already in ${channel}`);
1359
+ return;
1360
+ }
1361
+
1362
+ routerLog.info(`${memberName} joined ${channel} (${this.channels.get(channel)?.size ?? 0} members)${isAdminJoin ? ' [admin]' : ''}`);
1363
+ }
1364
+
1365
+ /**
1366
+ * Handle a CHANNEL_LEAVE message.
1367
+ * Removes the member from the channel and notifies remaining members.
1368
+ * If payload.member is provided, removes that member instead (admin mode).
1369
+ */
1370
+ handleChannelLeave(
1371
+ connection: RoutableConnection,
1372
+ envelope: Envelope<ChannelLeavePayload>
1373
+ ): void {
1374
+ // Use payload.member if provided (admin mode), otherwise use connection's name
1375
+ const memberName = envelope.payload.member ?? connection.agentName;
1376
+ if (!memberName) {
1377
+ routerLog.warn('CHANNEL_LEAVE from connection without name and no member specified');
1378
+ return;
1379
+ }
1380
+
1381
+ const channel = envelope.payload.channel;
1382
+ const isAdminRemove = Boolean(envelope.payload.member);
1383
+ const members = this.channels.get(channel);
1384
+
1385
+ if (!members || !members.has(memberName)) {
1386
+ routerLog.debug(`${memberName} not in ${channel}, ignoring leave`);
1387
+ return;
1388
+ }
1389
+
1390
+ const removed = this.removeChannelMember(channel, memberName, { persist: true });
1391
+ if (!removed) {
1392
+ routerLog.debug(`${memberName} not in ${channel}, ignoring leave`);
1393
+ return;
1394
+ }
1395
+
1396
+ // Only notify remaining members for non-admin removes
1397
+ // Admin removes are silent to avoid spamming notifications
1398
+ if (!isAdminRemove) {
1399
+ const remainingMembers = this.channels.get(channel);
1400
+ if (remainingMembers) {
1401
+ for (const remainingMember of remainingMembers) {
1402
+ const memberConn = this.getConnectionByName(remainingMember);
1403
+ if (memberConn) {
1404
+ const leaveNotification: Envelope<ChannelLeavePayload> = {
1405
+ v: PROTOCOL_VERSION,
1406
+ type: 'CHANNEL_LEAVE',
1407
+ id: generateId(),
1408
+ ts: Date.now(),
1409
+ from: memberName,
1410
+ payload: envelope.payload,
1411
+ };
1412
+ memberConn.send(leaveNotification);
1413
+ }
1414
+ }
1415
+ }
1416
+ }
1417
+
1418
+ routerLog.info(`${memberName} left ${channel}${isAdminRemove ? ' [admin]' : ''}`);
1419
+ }
1420
+
1421
+ /**
1422
+ * Route a channel message to all members except the sender.
1423
+ */
1424
+ routeChannelMessage(
1425
+ connection: RoutableConnection,
1426
+ envelope: Envelope<ChannelMessagePayload>
1427
+ ): void {
1428
+ const senderName = connection.agentName;
1429
+ if (!senderName) {
1430
+ routerLog.warn('CHANNEL_MESSAGE from connection without name');
1431
+ return;
1432
+ }
1433
+
1434
+ const channel = envelope.payload.channel;
1435
+ const members = this.channels.get(channel);
1436
+
1437
+ routerLog.info(`routeChannelMessage: channel=${channel} sender=${senderName} members=${members ? Array.from(members).join(',') : 'NONE'}`);
1438
+
1439
+ if (!members) {
1440
+ routerLog.warn(`Message to non-existent channel ${channel} (available channels: ${Array.from(this.channels.keys()).join(', ')})`);
1441
+ return;
1442
+ }
1443
+
1444
+ // Case-insensitive membership check
1445
+ const senderMemberName = this.findMemberInSet(members, senderName);
1446
+ if (!senderMemberName) {
1447
+ routerLog.warn(`${senderName} not a member of ${channel} (members: ${Array.from(members).join(', ')})`);
1448
+ return;
1449
+ }
1450
+
1451
+ // Route to all members except the sender (no echo)
1452
+ const allMembers = Array.from(members);
1453
+ routerLog.info(`Routing channel message from ${senderName} to ${channel}`, {
1454
+ totalMembers: allMembers.length,
1455
+ members: allMembers,
1456
+ });
1457
+
1458
+ let deliveredCount = 0;
1459
+ const undeliveredMembers: string[] = [];
1460
+ const connectedAgents = Array.from(this.agents.keys());
1461
+ const connectedUsers = Array.from(this.users.keys());
1462
+ routerLog.info(`Connected entities: agents=[${connectedAgents.join(',')}] users=[${connectedUsers.join(',')}]`);
1463
+
1464
+ for (const memberName of members) {
1465
+ // Case-insensitive comparison to skip sender
1466
+ if (this.namesMatch(memberName, senderName)) {
1467
+ continue;
1468
+ }
1469
+
1470
+ // Check for agent connection first
1471
+ const agentConn = this.agents.get(memberName);
1472
+ if (agentConn) {
1473
+ const deliverEnvelope: Envelope<ChannelMessagePayload> = {
1474
+ v: PROTOCOL_VERSION,
1475
+ type: 'CHANNEL_MESSAGE',
1476
+ id: generateId(),
1477
+ ts: Date.now(),
1478
+ from: senderName,
1479
+ payload: envelope.payload,
1480
+ };
1481
+ const sent = agentConn.send(deliverEnvelope);
1482
+ if (sent) {
1483
+ deliveredCount++;
1484
+ routerLog.info(`Delivered to ${memberName} (agent)`);
1485
+ } else {
1486
+ routerLog.warn(`Failed to send to ${memberName}`);
1487
+ undeliveredMembers.push(memberName);
1488
+ }
1489
+ continue;
1490
+ }
1491
+
1492
+ // Check for user connections (send to ALL for multi-tab support)
1493
+ const userConnections = this.users.get(memberName);
1494
+ if (userConnections && userConnections.size > 0) {
1495
+ let anyDelivered = false;
1496
+ for (const userConn of userConnections) {
1497
+ const deliverEnvelope: Envelope<ChannelMessagePayload> = {
1498
+ v: PROTOCOL_VERSION,
1499
+ type: 'CHANNEL_MESSAGE',
1500
+ id: generateId(),
1501
+ ts: Date.now(),
1502
+ from: senderName,
1503
+ payload: envelope.payload,
1504
+ };
1505
+ const sent = userConn.send(deliverEnvelope);
1506
+ if (sent) {
1507
+ anyDelivered = true;
1508
+ routerLog.info(`Delivered to ${memberName} (user connection ${userConn.id})`);
1509
+ }
1510
+ }
1511
+ if (anyDelivered) {
1512
+ deliveredCount++;
1513
+ } else {
1514
+ routerLog.warn(`Failed to send to any connection for user ${memberName}`);
1515
+ undeliveredMembers.push(memberName);
1516
+ }
1517
+ continue;
1518
+ }
1519
+
1520
+ // Member not connected
1521
+ routerLog.warn(`Member ${memberName} is registered in channel but NOT connected to daemon - message not delivered`);
1522
+ undeliveredMembers.push(memberName);
1523
+ }
1524
+
1525
+ // Persist channel message
1526
+ this.persistChannelMessage(envelope, senderName);
1527
+
1528
+ const recipientCount = allMembers.length - 1; // Exclude sender
1529
+ routerLog.info(`${senderName} -> ${channel}: delivered to ${deliveredCount}/${recipientCount} members`);
1530
+
1531
+ // Log warning if some members didn't receive the message
1532
+ if (undeliveredMembers.length > 0) {
1533
+ routerLog.warn(`Channel message undelivered to: [${undeliveredMembers.join(', ')}] - these agents may need to reconnect to the relay daemon`);
1534
+ }
1535
+ }
1536
+
1537
+ /**
1538
+ * Persist a channel message to storage.
1539
+ */
1540
+ private persistChannelMessage(
1541
+ envelope: Envelope<ChannelMessagePayload>,
1542
+ from: string
1543
+ ): void {
1544
+ if (!this.storage) return;
1545
+
1546
+ const payloadData = {
1547
+ ...envelope.payload.data,
1548
+ _isChannelMessage: true,
1549
+ _channel: envelope.payload.channel,
1550
+ _mentions: envelope.payload.mentions,
1551
+ };
1552
+
1553
+ this.storage.saveMessage({
1554
+ id: envelope.id,
1555
+ ts: envelope.ts,
1556
+ from,
1557
+ to: envelope.payload.channel, // Channel name as "to"
1558
+ topic: undefined,
1559
+ kind: 'message',
1560
+ body: envelope.payload.body,
1561
+ data: payloadData,
1562
+ thread: envelope.payload.thread,
1563
+ status: 'unread',
1564
+ is_urgent: false,
1565
+ is_broadcast: true, // Channel messages are effectively broadcasts
1566
+ }).catch((err) => {
1567
+ routerLog.error('Failed to persist channel message', { error: String(err) });
1568
+ });
1569
+ }
1570
+
1571
+ private persistChannelMembership(
1572
+ channel: string,
1573
+ member: string,
1574
+ action: 'join' | 'leave',
1575
+ opts?: { invitedBy?: string }
1576
+ ): void {
1577
+ if (this.storage) {
1578
+ this.storage.saveMessage({
1579
+ id: crypto.randomUUID(),
1580
+ ts: Date.now(),
1581
+ from: '__system__',
1582
+ to: channel,
1583
+ topic: undefined,
1584
+ kind: 'state', // membership events stored as state
1585
+ body: `${action}:${member}`,
1586
+ data: {
1587
+ _channelMembership: {
1588
+ member,
1589
+ action,
1590
+ invitedBy: opts?.invitedBy,
1591
+ },
1592
+ },
1593
+ status: 'read',
1594
+ is_urgent: false,
1595
+ is_broadcast: true,
1596
+ }).catch((err) => {
1597
+ routerLog.error('Failed to persist channel membership', { error: String(err) });
1598
+ });
1599
+ }
1600
+
1601
+ if (this.channelMembershipStore) {
1602
+ const persistPromise = action === 'leave'
1603
+ ? this.channelMembershipStore.removeMember(channel, member)
1604
+ : this.channelMembershipStore.addMember(channel, member);
1605
+
1606
+ persistPromise.catch((err) => {
1607
+ routerLog.error('Failed to sync channel membership to cloud store', {
1608
+ channel,
1609
+ member,
1610
+ action,
1611
+ error: err instanceof Error ? err.message : String(err),
1612
+ });
1613
+ });
1614
+ }
1615
+ }
1616
+
1617
+ /**
1618
+ * Get all members of a channel.
1619
+ */
1620
+ getChannelMembers(channel: string): string[] {
1621
+ const members = this.channels.get(channel);
1622
+ return members ? Array.from(members) : [];
1623
+ }
1624
+
1625
+ /**
1626
+ * Get all channels.
1627
+ */
1628
+ getChannels(): string[] {
1629
+ return Array.from(this.channels.keys());
1630
+ }
1631
+
1632
+ /**
1633
+ * Get all channels a member is in.
1634
+ */
1635
+ getChannelsForMember(memberName: string): string[] {
1636
+ const channels = this.memberChannels.get(memberName);
1637
+ return channels ? Array.from(channels) : [];
1638
+ }
1639
+
1640
+ /**
1641
+ * Check if a name belongs to a user (not an agent).
1642
+ */
1643
+ isUser(name: string): boolean {
1644
+ return this.users.has(name);
1645
+ }
1646
+
1647
+ /**
1648
+ * Check if a name belongs to an agent (not a user).
1649
+ */
1650
+ isAgent(name: string): boolean {
1651
+ return this.agents.has(name);
1652
+ }
1653
+
1654
+ /**
1655
+ * Get list of connected user names (human users only).
1656
+ */
1657
+ getUsers(): string[] {
1658
+ return Array.from(this.users.keys());
1659
+ }
1660
+
1661
+ /**
1662
+ * Get a connection by name (checks both agents and users).
1663
+ * Uses case-insensitive lookup to handle mismatched casing.
1664
+ * For users with multiple connections, returns the first connection.
1665
+ */
1666
+ private getConnectionByName(name: string): RoutableConnection | undefined {
1667
+ // Try exact match for agent first
1668
+ const agentExact = this.agents.get(name);
1669
+ if (agentExact) return agentExact;
1670
+
1671
+ // Try exact match for user (get first connection from set)
1672
+ const userConnections = this.users.get(name);
1673
+ if (userConnections && userConnections.size > 0) {
1674
+ return [...userConnections][0];
1675
+ }
1676
+
1677
+ // Fall back to case-insensitive search
1678
+ const lowerName = name.toLowerCase();
1679
+ for (const [key, conn] of this.agents) {
1680
+ if (key.toLowerCase() === lowerName) return conn;
1681
+ }
1682
+ for (const [key, conns] of this.users) {
1683
+ if (key.toLowerCase() === lowerName && conns.size > 0) {
1684
+ return [...conns][0];
1685
+ }
1686
+ }
1687
+ return undefined;
1688
+ }
1689
+
1690
+ /**
1691
+ * Check if a member is in a Set (case-insensitive).
1692
+ * Returns the actual stored name if found, undefined otherwise.
1693
+ */
1694
+ private findMemberInSet(members: Set<string>, name: string): string | undefined {
1695
+ // Try exact match first
1696
+ if (members.has(name)) return name;
1697
+
1698
+ // Fall back to case-insensitive search
1699
+ const lowerName = name.toLowerCase();
1700
+ for (const member of members) {
1701
+ if (member.toLowerCase() === lowerName) return member;
1702
+ }
1703
+ return undefined;
1704
+ }
1705
+
1706
+ /**
1707
+ * Check if two names match (case-insensitive).
1708
+ */
1709
+ private namesMatch(a: string, b: string): boolean {
1710
+ return a.toLowerCase() === b.toLowerCase();
1711
+ }
1712
+
1713
+ /**
1714
+ * Auto-join a member to a channel without notifications.
1715
+ * Used for default channel membership (e.g., #general).
1716
+ * @param memberName - The agent or user name to add
1717
+ * @param channel - The channel to join (e.g., '#general')
1718
+ */
1719
+ autoJoinChannel(memberName: string, channel: string, options?: { persist?: boolean }): void {
1720
+ // Get or create channel
1721
+ let members = this.channels.get(channel);
1722
+ if (!members) {
1723
+ members = new Set();
1724
+ this.channels.set(channel, members);
1725
+ }
1726
+
1727
+ // Check if already a member
1728
+ const added = this.addChannelMember(channel, memberName, { persist: options?.persist });
1729
+ if (added) {
1730
+ routerLog.debug(`Auto-joined ${memberName} to ${channel}`);
1731
+ }
1732
+ }
1733
+
1734
+ private addChannelMember(
1735
+ channel: string,
1736
+ memberName: string,
1737
+ options?: { persist?: boolean }
1738
+ ): boolean {
1739
+ let members = this.channels.get(channel);
1740
+ if (!members) {
1741
+ members = new Set();
1742
+ this.channels.set(channel, members);
1743
+ }
1744
+ // Case-insensitive check for existing membership
1745
+ const existingMember = this.findMemberInSet(members, memberName);
1746
+ if (existingMember) {
1747
+ return false;
1748
+ }
1749
+ members.add(memberName);
1750
+
1751
+ const memberChannelSet = this.memberChannels.get(memberName) ?? new Set();
1752
+ memberChannelSet.add(channel);
1753
+ this.memberChannels.set(memberName, memberChannelSet);
1754
+
1755
+ if (options?.persist ?? true) {
1756
+ this.persistChannelMembership(channel, memberName, 'join');
1757
+ }
1758
+
1759
+ return true;
1760
+ }
1761
+
1762
+ private removeChannelMember(
1763
+ channel: string,
1764
+ memberName: string,
1765
+ options?: { persist?: boolean }
1766
+ ): boolean {
1767
+ const members = this.channels.get(channel);
1768
+ if (!members) {
1769
+ return false;
1770
+ }
1771
+
1772
+ // Case-insensitive lookup to find actual stored name
1773
+ const actualMemberName = this.findMemberInSet(members, memberName);
1774
+ if (!actualMemberName) {
1775
+ return false;
1776
+ }
1777
+
1778
+ members.delete(actualMemberName);
1779
+ if (members.size === 0) {
1780
+ this.channels.delete(channel);
1781
+ }
1782
+
1783
+ // Also try case-insensitive for memberChannels cleanup
1784
+ const memberChannelSet = this.memberChannels.get(actualMemberName) ?? this.memberChannels.get(memberName);
1785
+ if (memberChannelSet) {
1786
+ memberChannelSet.delete(channel);
1787
+ if (memberChannelSet.size === 0) {
1788
+ this.memberChannels.delete(actualMemberName);
1789
+ this.memberChannels.delete(memberName); // Clean up both potential keys
1790
+ }
1791
+ }
1792
+
1793
+ if (options?.persist ?? true) {
1794
+ this.persistChannelMembership(channel, actualMemberName, 'leave');
1795
+ }
1796
+
1797
+ return true;
1798
+ }
1799
+
1800
+ handleMembershipUpdate(update: { channel: string; member: string; action: 'join' | 'leave' | 'invite' }) {
1801
+ if (!update.channel || !update.member) {
1802
+ return;
1803
+ }
1804
+
1805
+ if (update.action === 'leave') {
1806
+ this.removeChannelMember(update.channel, update.member, { persist: false });
1807
+ } else {
1808
+ this.addChannelMember(update.channel, update.member, { persist: false });
1809
+ }
1810
+ }
1811
+
1812
+ /**
1813
+ * Auto-rejoin an agent to their persisted channels on reconnect.
1814
+ * This handles daemon restarts where in-memory channel state is lost.
1815
+ * Queries both cloud DB (if available) and SQLite storage for memberships.
1816
+ * Uses silent/admin mode to avoid spamming join notifications.
1817
+ */
1818
+ async autoRejoinChannelsForAgent(agentName: string): Promise<void> {
1819
+ const channelsToJoin = new Set<string>();
1820
+
1821
+ // Query cloud DB if available
1822
+ if (this.channelMembershipStore?.loadMembershipsForAgent) {
1823
+ try {
1824
+ const cloudMemberships = await this.channelMembershipStore.loadMembershipsForAgent(agentName);
1825
+ for (const membership of cloudMemberships) {
1826
+ channelsToJoin.add(membership.channel);
1827
+ }
1828
+ if (cloudMemberships.length > 0) {
1829
+ routerLog.debug(`Found ${cloudMemberships.length} channel memberships for ${agentName} in cloud DB`);
1830
+ }
1831
+ } catch (err) {
1832
+ routerLog.error('Failed to query cloud DB for channel memberships', {
1833
+ agentName,
1834
+ error: String(err),
1835
+ });
1836
+ }
1837
+ }
1838
+
1839
+ // Query SQLite storage if available
1840
+ if (this.storage?.getChannelMembershipsForAgent) {
1841
+ try {
1842
+ const sqliteMemberships = await this.storage.getChannelMembershipsForAgent(agentName);
1843
+ for (const channel of sqliteMemberships) {
1844
+ channelsToJoin.add(channel);
1845
+ }
1846
+ if (sqliteMemberships.length > 0) {
1847
+ routerLog.debug(`Found ${sqliteMemberships.length} channel memberships for ${agentName} in SQLite`);
1848
+ }
1849
+ } catch (err) {
1850
+ routerLog.error('Failed to query SQLite for channel memberships', {
1851
+ agentName,
1852
+ error: String(err),
1853
+ });
1854
+ }
1855
+ }
1856
+
1857
+ if (channelsToJoin.size === 0) {
1858
+ routerLog.debug(`No persisted channel memberships found for ${agentName}`);
1859
+ return;
1860
+ }
1861
+
1862
+ // Rejoin channels silently (don't notify other members)
1863
+ let rejoinedCount = 0;
1864
+ for (const channel of channelsToJoin) {
1865
+ // Skip if already in channel (handles deduplication)
1866
+ const members = this.channels.get(channel);
1867
+ if (members && this.findMemberInSet(members, agentName)) {
1868
+ routerLog.debug(`${agentName} already in ${channel}, skipping auto-rejoin`);
1869
+ continue;
1870
+ }
1871
+
1872
+ // Add to channel without persisting (already persisted) or notifying
1873
+ const added = this.addChannelMember(channel, agentName, { persist: false });
1874
+ if (added) {
1875
+ rejoinedCount++;
1876
+ }
1877
+ }
1878
+
1879
+ if (rejoinedCount > 0) {
1880
+ routerLog.info(`Auto-rejoined ${agentName} to ${rejoinedCount} channels`, {
1881
+ channels: Array.from(channelsToJoin),
1882
+ });
1883
+ }
1884
+ }
1885
+ }