agent-relay 1.0.21 → 1.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (283) hide show
  1. package/dist/bridge/shadow-cli.d.ts +17 -0
  2. package/dist/bridge/shadow-cli.d.ts.map +1 -0
  3. package/dist/bridge/shadow-cli.js +75 -0
  4. package/dist/bridge/shadow-cli.js.map +1 -0
  5. package/dist/bridge/shadow-config.d.ts +87 -0
  6. package/dist/bridge/shadow-config.d.ts.map +1 -0
  7. package/dist/bridge/shadow-config.js +134 -0
  8. package/dist/bridge/shadow-config.js.map +1 -0
  9. package/dist/bridge/spawner.d.ts +15 -1
  10. package/dist/bridge/spawner.d.ts.map +1 -1
  11. package/dist/bridge/spawner.js +164 -4
  12. package/dist/bridge/spawner.js.map +1 -1
  13. package/dist/bridge/types.d.ts +55 -0
  14. package/dist/bridge/types.d.ts.map +1 -1
  15. package/dist/cli/index.js +796 -11
  16. package/dist/cli/index.js.map +1 -1
  17. package/dist/cloud/api/auth.d.ts +19 -0
  18. package/dist/cloud/api/auth.d.ts.map +1 -0
  19. package/dist/cloud/api/auth.js +216 -0
  20. package/dist/cloud/api/auth.js.map +1 -0
  21. package/dist/cloud/api/billing.d.ts +17 -0
  22. package/dist/cloud/api/billing.d.ts.map +1 -0
  23. package/dist/cloud/api/billing.js +353 -0
  24. package/dist/cloud/api/billing.js.map +1 -0
  25. package/dist/cloud/api/coordinators.d.ts +8 -0
  26. package/dist/cloud/api/coordinators.d.ts.map +1 -0
  27. package/dist/cloud/api/coordinators.js +347 -0
  28. package/dist/cloud/api/coordinators.js.map +1 -0
  29. package/dist/cloud/api/daemons.d.ts +12 -0
  30. package/dist/cloud/api/daemons.d.ts.map +1 -0
  31. package/dist/cloud/api/daemons.js +320 -0
  32. package/dist/cloud/api/daemons.js.map +1 -0
  33. package/dist/cloud/api/middleware/planLimits.d.ts +36 -0
  34. package/dist/cloud/api/middleware/planLimits.d.ts.map +1 -0
  35. package/dist/cloud/api/middleware/planLimits.js +164 -0
  36. package/dist/cloud/api/middleware/planLimits.js.map +1 -0
  37. package/dist/cloud/api/onboarding.d.ts +8 -0
  38. package/dist/cloud/api/onboarding.d.ts.map +1 -0
  39. package/dist/cloud/api/onboarding.js +407 -0
  40. package/dist/cloud/api/onboarding.js.map +1 -0
  41. package/dist/cloud/api/providers.d.ts +7 -0
  42. package/dist/cloud/api/providers.d.ts.map +1 -0
  43. package/dist/cloud/api/providers.js +435 -0
  44. package/dist/cloud/api/providers.js.map +1 -0
  45. package/dist/cloud/api/repos.d.ts +7 -0
  46. package/dist/cloud/api/repos.d.ts.map +1 -0
  47. package/dist/cloud/api/repos.js +314 -0
  48. package/dist/cloud/api/repos.js.map +1 -0
  49. package/dist/cloud/api/teams.d.ts +7 -0
  50. package/dist/cloud/api/teams.d.ts.map +1 -0
  51. package/dist/cloud/api/teams.js +279 -0
  52. package/dist/cloud/api/teams.js.map +1 -0
  53. package/dist/cloud/api/usage.d.ts +7 -0
  54. package/dist/cloud/api/usage.d.ts.map +1 -0
  55. package/dist/cloud/api/usage.js +98 -0
  56. package/dist/cloud/api/usage.js.map +1 -0
  57. package/dist/cloud/api/workspaces.d.ts +7 -0
  58. package/dist/cloud/api/workspaces.d.ts.map +1 -0
  59. package/dist/cloud/api/workspaces.js +510 -0
  60. package/dist/cloud/api/workspaces.js.map +1 -0
  61. package/dist/cloud/billing/index.d.ts +9 -0
  62. package/dist/cloud/billing/index.d.ts.map +1 -0
  63. package/dist/cloud/billing/index.js +9 -0
  64. package/dist/cloud/billing/index.js.map +1 -0
  65. package/dist/cloud/billing/plans.d.ts +39 -0
  66. package/dist/cloud/billing/plans.d.ts.map +1 -0
  67. package/dist/cloud/billing/plans.js +232 -0
  68. package/dist/cloud/billing/plans.js.map +1 -0
  69. package/dist/cloud/billing/service.d.ts +80 -0
  70. package/dist/cloud/billing/service.d.ts.map +1 -0
  71. package/dist/cloud/billing/service.js +388 -0
  72. package/dist/cloud/billing/service.js.map +1 -0
  73. package/dist/cloud/billing/types.d.ts +135 -0
  74. package/dist/cloud/billing/types.d.ts.map +1 -0
  75. package/dist/cloud/billing/types.js +7 -0
  76. package/dist/cloud/billing/types.js.map +1 -0
  77. package/dist/cloud/config.d.ts +59 -0
  78. package/dist/cloud/config.d.ts.map +1 -0
  79. package/dist/cloud/config.js +83 -0
  80. package/dist/cloud/config.js.map +1 -0
  81. package/dist/cloud/db/drizzle.d.ts +132 -0
  82. package/dist/cloud/db/drizzle.d.ts.map +1 -0
  83. package/dist/cloud/db/drizzle.js +613 -0
  84. package/dist/cloud/db/drizzle.js.map +1 -0
  85. package/dist/cloud/db/index.d.ts +30 -0
  86. package/dist/cloud/db/index.d.ts.map +1 -0
  87. package/dist/cloud/db/index.js +44 -0
  88. package/dist/cloud/db/index.js.map +1 -0
  89. package/dist/cloud/db/schema.d.ts +1792 -0
  90. package/dist/cloud/db/schema.d.ts.map +1 -0
  91. package/dist/cloud/db/schema.js +234 -0
  92. package/dist/cloud/db/schema.js.map +1 -0
  93. package/dist/cloud/index.d.ts +11 -0
  94. package/dist/cloud/index.d.ts.map +1 -0
  95. package/dist/cloud/index.js +37 -0
  96. package/dist/cloud/index.js.map +1 -0
  97. package/dist/cloud/provisioner/index.d.ts +51 -0
  98. package/dist/cloud/provisioner/index.d.ts.map +1 -0
  99. package/dist/cloud/provisioner/index.js +676 -0
  100. package/dist/cloud/provisioner/index.js.map +1 -0
  101. package/dist/cloud/server.d.ts +16 -0
  102. package/dist/cloud/server.d.ts.map +1 -0
  103. package/dist/cloud/server.js +190 -0
  104. package/dist/cloud/server.js.map +1 -0
  105. package/dist/cloud/services/coordinator.d.ts +62 -0
  106. package/dist/cloud/services/coordinator.d.ts.map +1 -0
  107. package/dist/cloud/services/coordinator.js +389 -0
  108. package/dist/cloud/services/coordinator.js.map +1 -0
  109. package/dist/cloud/services/planLimits.d.ts +110 -0
  110. package/dist/cloud/services/planLimits.d.ts.map +1 -0
  111. package/dist/cloud/services/planLimits.js +254 -0
  112. package/dist/cloud/services/planLimits.js.map +1 -0
  113. package/dist/cloud/vault/index.d.ts +76 -0
  114. package/dist/cloud/vault/index.d.ts.map +1 -0
  115. package/dist/cloud/vault/index.js +219 -0
  116. package/dist/cloud/vault/index.js.map +1 -0
  117. package/dist/daemon/agent-manager.d.ts +87 -0
  118. package/dist/daemon/agent-manager.d.ts.map +1 -0
  119. package/dist/daemon/agent-manager.js +412 -0
  120. package/dist/daemon/agent-manager.js.map +1 -0
  121. package/dist/daemon/agent-registry.d.ts +2 -0
  122. package/dist/daemon/agent-registry.d.ts.map +1 -1
  123. package/dist/daemon/agent-registry.js +3 -0
  124. package/dist/daemon/agent-registry.js.map +1 -1
  125. package/dist/daemon/api.d.ts +69 -0
  126. package/dist/daemon/api.d.ts.map +1 -0
  127. package/dist/daemon/api.js +425 -0
  128. package/dist/daemon/api.js.map +1 -0
  129. package/dist/daemon/cloud-sync.d.ts +101 -0
  130. package/dist/daemon/cloud-sync.d.ts.map +1 -0
  131. package/dist/daemon/cloud-sync.js +261 -0
  132. package/dist/daemon/cloud-sync.js.map +1 -0
  133. package/dist/daemon/index.d.ts +4 -0
  134. package/dist/daemon/index.d.ts.map +1 -1
  135. package/dist/daemon/index.js +6 -0
  136. package/dist/daemon/index.js.map +1 -1
  137. package/dist/daemon/orchestrator.d.ts +155 -0
  138. package/dist/daemon/orchestrator.d.ts.map +1 -0
  139. package/dist/daemon/orchestrator.js +736 -0
  140. package/dist/daemon/orchestrator.js.map +1 -0
  141. package/dist/daemon/router.d.ts +24 -0
  142. package/dist/daemon/router.d.ts.map +1 -1
  143. package/dist/daemon/router.js +71 -1
  144. package/dist/daemon/router.js.map +1 -1
  145. package/dist/daemon/server.d.ts +37 -0
  146. package/dist/daemon/server.d.ts.map +1 -1
  147. package/dist/daemon/server.js +191 -16
  148. package/dist/daemon/server.js.map +1 -1
  149. package/dist/daemon/types.d.ts +127 -0
  150. package/dist/daemon/types.d.ts.map +1 -0
  151. package/dist/daemon/types.js +6 -0
  152. package/dist/daemon/types.js.map +1 -0
  153. package/dist/daemon/workspace-manager.d.ts +75 -0
  154. package/dist/daemon/workspace-manager.d.ts.map +1 -0
  155. package/dist/daemon/workspace-manager.js +289 -0
  156. package/dist/daemon/workspace-manager.js.map +1 -0
  157. package/dist/dashboard/out/404.html +1 -1
  158. package/dist/dashboard/out/_next/static/chunks/693-7b3301d8f6bc5014.js +1 -0
  159. package/dist/dashboard/out/_next/static/chunks/713-f78477eb185f1f4d.js +1 -0
  160. package/dist/dashboard/out/_next/static/chunks/766-e53e1cfe39b0b5b5.js +1 -0
  161. package/dist/dashboard/out/_next/static/chunks/900-037c64bfd797fb2a.js +1 -0
  162. package/dist/dashboard/out/_next/static/chunks/app/app/page-e3d9e1f4466b9bae.js +1 -0
  163. package/dist/dashboard/out/_next/static/chunks/app/history/page-b6edd4dde8d08194.js +1 -0
  164. package/dist/dashboard/out/_next/static/chunks/app/layout-2433bb48965f4333.js +1 -0
  165. package/dist/dashboard/out/_next/static/chunks/app/metrics/page-e68825a81db67ba1.js +1 -0
  166. package/dist/dashboard/out/_next/static/chunks/app/page-cc108bf68c8a657f.js +1 -0
  167. package/dist/dashboard/out/_next/static/chunks/app/pricing/page-d80e03a5297f95b6.js +1 -0
  168. package/dist/dashboard/out/_next/static/chunks/main-app-5d692157a8eb1fd9.js +1 -0
  169. package/dist/dashboard/out/_next/static/chunks/{main-e0a1f53fe0617a63.js → main-c2f423b9c9f4591b.js} +1 -1
  170. package/dist/dashboard/out/_next/static/chunks/{webpack-c81f7fd28659d64f.js → webpack-a5acc2831d094776.js} +1 -1
  171. package/dist/dashboard/out/_next/static/css/79b80143647a07d7.css +1 -0
  172. package/dist/dashboard/out/_next/static/css/8cf277370ad48cfe.css +1 -0
  173. package/dist/dashboard/out/alt-logos/agent-relay-logo-128.png +0 -0
  174. package/dist/dashboard/out/alt-logos/agent-relay-logo-256.png +0 -0
  175. package/dist/dashboard/out/alt-logos/agent-relay-logo-32.png +0 -0
  176. package/dist/dashboard/out/alt-logos/agent-relay-logo-512.png +0 -0
  177. package/dist/dashboard/out/alt-logos/agent-relay-logo-64.png +0 -0
  178. package/dist/dashboard/out/alt-logos/agent-relay-logo.svg +45 -0
  179. package/dist/dashboard/out/alt-logos/logo.svg +38 -0
  180. package/dist/dashboard/out/alt-logos/monogram-logo-128.png +0 -0
  181. package/dist/dashboard/out/alt-logos/monogram-logo-256.png +0 -0
  182. package/dist/dashboard/out/alt-logos/monogram-logo-32.png +0 -0
  183. package/dist/dashboard/out/alt-logos/monogram-logo-512.png +0 -0
  184. package/dist/dashboard/out/alt-logos/monogram-logo-64.png +0 -0
  185. package/dist/dashboard/out/alt-logos/monogram-logo.svg +38 -0
  186. package/dist/dashboard/out/app.html +14 -0
  187. package/dist/dashboard/out/app.txt +7 -0
  188. package/dist/dashboard/out/history.html +1 -0
  189. package/dist/dashboard/out/history.txt +7 -0
  190. package/dist/dashboard/out/index.html +1 -1
  191. package/dist/dashboard/out/index.txt +2 -2
  192. package/dist/dashboard/out/metrics.html +1 -515
  193. package/dist/dashboard/out/metrics.txt +2 -2
  194. package/dist/dashboard/out/pricing.html +13 -0
  195. package/dist/dashboard/out/pricing.txt +7 -0
  196. package/dist/dashboard-server/metrics.d.ts.map +1 -1
  197. package/dist/dashboard-server/metrics.js +3 -2
  198. package/dist/dashboard-server/metrics.js.map +1 -1
  199. package/dist/dashboard-server/server.d.ts.map +1 -1
  200. package/dist/dashboard-server/server.js +1279 -56
  201. package/dist/dashboard-server/server.js.map +1 -1
  202. package/dist/protocol/types.d.ts +10 -1
  203. package/dist/protocol/types.d.ts.map +1 -1
  204. package/dist/resiliency/context-persistence.d.ts +140 -0
  205. package/dist/resiliency/context-persistence.d.ts.map +1 -0
  206. package/dist/resiliency/context-persistence.js +397 -0
  207. package/dist/resiliency/context-persistence.js.map +1 -0
  208. package/dist/resiliency/health-monitor.d.ts +97 -0
  209. package/dist/resiliency/health-monitor.d.ts.map +1 -0
  210. package/dist/resiliency/health-monitor.js +291 -0
  211. package/dist/resiliency/health-monitor.js.map +1 -0
  212. package/dist/resiliency/index.d.ts +63 -0
  213. package/dist/resiliency/index.d.ts.map +1 -0
  214. package/dist/resiliency/index.js +63 -0
  215. package/dist/resiliency/index.js.map +1 -0
  216. package/dist/resiliency/logger.d.ts +114 -0
  217. package/dist/resiliency/logger.d.ts.map +1 -0
  218. package/dist/resiliency/logger.js +250 -0
  219. package/dist/resiliency/logger.js.map +1 -0
  220. package/dist/resiliency/metrics.d.ts +115 -0
  221. package/dist/resiliency/metrics.d.ts.map +1 -0
  222. package/dist/resiliency/metrics.js +239 -0
  223. package/dist/resiliency/metrics.js.map +1 -0
  224. package/dist/resiliency/provider-context.d.ts +100 -0
  225. package/dist/resiliency/provider-context.d.ts.map +1 -0
  226. package/dist/resiliency/provider-context.js +360 -0
  227. package/dist/resiliency/provider-context.js.map +1 -0
  228. package/dist/resiliency/supervisor.d.ts +109 -0
  229. package/dist/resiliency/supervisor.d.ts.map +1 -0
  230. package/dist/resiliency/supervisor.js +337 -0
  231. package/dist/resiliency/supervisor.js.map +1 -0
  232. package/dist/storage/adapter.d.ts +2 -0
  233. package/dist/storage/adapter.d.ts.map +1 -1
  234. package/dist/storage/adapter.js +12 -2
  235. package/dist/storage/adapter.js.map +1 -1
  236. package/dist/storage/sqlite-adapter.d.ts.map +1 -1
  237. package/dist/storage/sqlite-adapter.js +18 -14
  238. package/dist/storage/sqlite-adapter.js.map +1 -1
  239. package/dist/utils/index.d.ts +1 -0
  240. package/dist/utils/index.d.ts.map +1 -1
  241. package/dist/utils/index.js +1 -0
  242. package/dist/utils/index.js.map +1 -1
  243. package/dist/utils/logger.d.ts +40 -0
  244. package/dist/utils/logger.d.ts.map +1 -0
  245. package/dist/utils/logger.js +84 -0
  246. package/dist/utils/logger.js.map +1 -0
  247. package/dist/wrapper/client.d.ts +16 -1
  248. package/dist/wrapper/client.d.ts.map +1 -1
  249. package/dist/wrapper/client.js +32 -1
  250. package/dist/wrapper/client.js.map +1 -1
  251. package/dist/wrapper/parser.d.ts +3 -0
  252. package/dist/wrapper/parser.d.ts.map +1 -1
  253. package/dist/wrapper/parser.js +121 -18
  254. package/dist/wrapper/parser.js.map +1 -1
  255. package/dist/wrapper/pty-wrapper.d.ts +28 -1
  256. package/dist/wrapper/pty-wrapper.d.ts.map +1 -1
  257. package/dist/wrapper/pty-wrapper.js +166 -30
  258. package/dist/wrapper/pty-wrapper.js.map +1 -1
  259. package/dist/wrapper/tmux-wrapper.d.ts +5 -0
  260. package/dist/wrapper/tmux-wrapper.d.ts.map +1 -1
  261. package/dist/wrapper/tmux-wrapper.js +58 -18
  262. package/dist/wrapper/tmux-wrapper.js.map +1 -1
  263. package/docs/CLOUD-ARCHITECTURE.md +652 -0
  264. package/docs/CLOUD-ONBOARDING-DESIGN.md +1983 -0
  265. package/docs/TESTING_PRESENCE_FEATURES.md +327 -0
  266. package/docs/agent-relay-snippet.md +107 -4
  267. package/docs/guides/CLOUD.md +236 -0
  268. package/docs/guides/LOCAL.md +535 -0
  269. package/docs/guides/SELF-HOSTED.md +494 -0
  270. package/docs/proposals/shadow-as-subagent.md +765 -0
  271. package/docs/proposals/slack-bot-integration.md +1457 -0
  272. package/package.json +33 -4
  273. package/dist/dashboard/out/_next/static/chunks/app/layout-c9d8c5d95e48c6bf.js +0 -1
  274. package/dist/dashboard/out/_next/static/chunks/app/metrics/page-8aa9936bc6c771ab.js +0 -1
  275. package/dist/dashboard/out/_next/static/chunks/app/page-49055e5d2b5e34ec.js +0 -1
  276. package/dist/dashboard/out/_next/static/chunks/main-app-bae2e535de00de50.js +0 -1
  277. package/dist/dashboard/out/_next/static/css/50ed6996e3df7bdd.css +0 -1
  278. /package/dist/dashboard/out/_next/static/{gZXwjIKGDKJ0hiTH-HMeJ → 6HHWb2ZmnJ4OSm0zUP7h4}/_buildManifest.js +0 -0
  279. /package/dist/dashboard/out/_next/static/{gZXwjIKGDKJ0hiTH-HMeJ → 6HHWb2ZmnJ4OSm0zUP7h4}/_ssgManifest.js +0 -0
  280. /package/dist/dashboard/out/_next/static/chunks/{117-3bef7b19f3e60751.js → 117-b2cd8d6485aacf2b.js} +0 -0
  281. /package/dist/dashboard/out/_next/static/chunks/{648-6cf686106c891ad3.js → 648-8f3f26864ce515e5.js} +0 -0
  282. /package/dist/dashboard/out/_next/static/chunks/app/_not-found/{page-8ff6572bc7c9bc61.js → page-0b990dbb71d72a98.js} +0 -0
  283. /package/dist/dashboard/out/_next/static/chunks/{fd9d1056-26bd8d656b496dba.js → fd9d1056-bf46c09eb57e019c.js} +0 -0
@@ -0,0 +1,389 @@
1
+ /**
2
+ * Coordinator Agent Service
3
+ *
4
+ * Manages lifecycle of coordinator agents for project groups.
5
+ * Coordinators oversee and orchestrate work across repositories in a group.
6
+ */
7
+ import { db } from '../db/index.js';
8
+ import { createClient } from 'redis';
9
+ import { getConfig } from '../config.js';
10
+ /**
11
+ * In-memory coordinator state tracker
12
+ * In production, this would be persisted to database or Redis
13
+ */
14
+ const coordinatorStates = new Map();
15
+ /**
16
+ * Start a coordinator agent for a project group
17
+ */
18
+ async function start(groupId) {
19
+ const group = await db.projectGroups.findById(groupId);
20
+ if (!group) {
21
+ throw new Error('Project group not found');
22
+ }
23
+ if (!group.coordinatorAgent?.enabled) {
24
+ throw new Error('Coordinator is not enabled for this group');
25
+ }
26
+ const repositories = await db.repositories.findByProjectGroupId(groupId);
27
+ if (repositories.length === 0) {
28
+ throw new Error('Cannot start coordinator for empty group');
29
+ }
30
+ const config = group.coordinatorAgent;
31
+ const agentName = config.name || `${group.name} Coordinator`;
32
+ const model = config.model || 'claude-sonnet-4-5';
33
+ // Check if already running
34
+ const existing = coordinatorStates.get(groupId);
35
+ if (existing && existing.status === 'running') {
36
+ console.log(`Coordinator for group ${groupId} is already running`);
37
+ return;
38
+ }
39
+ // Update state to starting
40
+ const state = {
41
+ groupId,
42
+ groupName: group.name,
43
+ agentName,
44
+ model,
45
+ status: 'starting',
46
+ repositories,
47
+ };
48
+ coordinatorStates.set(groupId, state);
49
+ try {
50
+ // Spawn the coordinator agent
51
+ // In a real implementation, this would:
52
+ // 1. Connect to agent-relay daemon or cloud workspace
53
+ // 2. Spawn agent with configured name and model
54
+ // 3. Provide system prompt with group context
55
+ // 4. Configure capabilities (read repos, create PRs, etc.)
56
+ await spawnCoordinatorAgent(group, config, repositories);
57
+ // Update state to running
58
+ state.status = 'running';
59
+ state.startedAt = new Date();
60
+ coordinatorStates.set(groupId, state);
61
+ console.log(`Coordinator agent started for group ${groupId}: ${agentName}`);
62
+ }
63
+ catch (error) {
64
+ state.status = 'error';
65
+ state.error = error instanceof Error ? error.message : 'Unknown error';
66
+ coordinatorStates.set(groupId, state);
67
+ throw error;
68
+ }
69
+ }
70
+ /**
71
+ * Stop a coordinator agent for a project group
72
+ */
73
+ async function stop(groupId) {
74
+ const state = coordinatorStates.get(groupId);
75
+ if (!state) {
76
+ // Not running, nothing to do
77
+ return;
78
+ }
79
+ if (state.status === 'stopped') {
80
+ return;
81
+ }
82
+ // Update state to stopping
83
+ state.status = 'stopping';
84
+ coordinatorStates.set(groupId, state);
85
+ try {
86
+ // Stop the coordinator agent
87
+ // In a real implementation, this would:
88
+ // 1. Send stop signal to the agent
89
+ // 2. Wait for graceful shutdown
90
+ // 3. Clean up resources
91
+ await stopCoordinatorAgent(groupId, state);
92
+ // Update state to stopped
93
+ state.status = 'stopped';
94
+ state.stoppedAt = new Date();
95
+ coordinatorStates.set(groupId, state);
96
+ console.log(`Coordinator agent stopped for group ${groupId}`);
97
+ }
98
+ catch (error) {
99
+ state.status = 'error';
100
+ state.error = error instanceof Error ? error.message : 'Unknown error';
101
+ coordinatorStates.set(groupId, state);
102
+ throw error;
103
+ }
104
+ }
105
+ /**
106
+ * Restart a coordinator agent
107
+ */
108
+ async function restart(groupId) {
109
+ await stop(groupId);
110
+ await start(groupId);
111
+ }
112
+ /**
113
+ * Get status of a coordinator agent
114
+ */
115
+ async function getStatus(groupId) {
116
+ return coordinatorStates.get(groupId) || null;
117
+ }
118
+ /**
119
+ * List all active coordinators
120
+ */
121
+ async function listActive() {
122
+ return Array.from(coordinatorStates.values()).filter((state) => state.status === 'running' || state.status === 'starting');
123
+ }
124
+ /**
125
+ * Redis pub/sub for cross-workspace messaging
126
+ */
127
+ let pubClient = null;
128
+ let subClient = null;
129
+ const messageHandlers = new Map();
130
+ /**
131
+ * Initialize Redis pub/sub clients
132
+ */
133
+ async function initRedisClients() {
134
+ if (pubClient && subClient)
135
+ return;
136
+ const config = getConfig();
137
+ pubClient = createClient({ url: config.redisUrl });
138
+ subClient = createClient({ url: config.redisUrl });
139
+ await pubClient.connect();
140
+ await subClient.connect();
141
+ // Subscribe to coordinator channel
142
+ await subClient.subscribe('coordinator:messages', (message) => {
143
+ try {
144
+ const parsed = JSON.parse(message);
145
+ const handler = messageHandlers.get(parsed.toWorkspace || '*');
146
+ if (handler) {
147
+ handler(parsed);
148
+ }
149
+ }
150
+ catch (error) {
151
+ console.error('Failed to parse coordinator message:', error);
152
+ }
153
+ });
154
+ console.log('[coordinator] Redis pub/sub initialized');
155
+ }
156
+ /**
157
+ * Send a message from coordinator to a workspace
158
+ */
159
+ export async function sendToWorkspace(coordinatorGroupId, targetWorkspaceId, agentName, message, thread) {
160
+ if (!pubClient) {
161
+ await initRedisClients();
162
+ }
163
+ const crossMessage = {
164
+ type: 'relay',
165
+ from: 'Coordinator',
166
+ fromWorkspace: `coordinator:${coordinatorGroupId}`,
167
+ to: agentName,
168
+ toWorkspace: targetWorkspaceId,
169
+ body: message,
170
+ thread,
171
+ timestamp: new Date().toISOString(),
172
+ };
173
+ await pubClient.publish('workspace:messages', JSON.stringify(crossMessage));
174
+ console.log(`[coordinator] Sent message to ${targetWorkspaceId}:${agentName}`);
175
+ }
176
+ /**
177
+ * Broadcast a message to all workspaces in a project group
178
+ */
179
+ export async function broadcastToGroup(coordinatorGroupId, message, thread) {
180
+ if (!pubClient) {
181
+ await initRedisClients();
182
+ }
183
+ const group = await db.projectGroups.findById(coordinatorGroupId);
184
+ if (!group) {
185
+ throw new Error('Project group not found');
186
+ }
187
+ const repositories = await db.repositories.findByProjectGroupId(coordinatorGroupId);
188
+ // Get all workspaces containing these repositories
189
+ const workspaceIds = new Set();
190
+ for (const repo of repositories) {
191
+ if (repo.workspaceId) {
192
+ workspaceIds.add(repo.workspaceId);
193
+ }
194
+ }
195
+ const crossMessage = {
196
+ type: 'relay',
197
+ from: 'Coordinator',
198
+ fromWorkspace: `coordinator:${coordinatorGroupId}`,
199
+ to: '*',
200
+ body: message,
201
+ thread,
202
+ timestamp: new Date().toISOString(),
203
+ };
204
+ // Broadcast to each workspace
205
+ for (const workspaceId of workspaceIds) {
206
+ crossMessage.toWorkspace = workspaceId;
207
+ await pubClient.publish('workspace:messages', JSON.stringify(crossMessage));
208
+ }
209
+ console.log(`[coordinator] Broadcast to ${workspaceIds.size} workspace(s)`);
210
+ }
211
+ /**
212
+ * Spawn the actual coordinator agent
213
+ */
214
+ async function spawnCoordinatorAgent(group, config, repositories) {
215
+ // Initialize Redis for cross-workspace messaging
216
+ await initRedisClients();
217
+ // Build system prompt for the coordinator
218
+ const systemPrompt = buildCoordinatorSystemPrompt(group, config, repositories);
219
+ // Get workspaces for the repositories
220
+ const workspaceIds = new Set();
221
+ for (const repo of repositories) {
222
+ if (repo.workspaceId) {
223
+ workspaceIds.add(repo.workspaceId);
224
+ }
225
+ }
226
+ const workspaces = await Promise.all(Array.from(workspaceIds).map(id => db.workspaces.findById(id)));
227
+ console.log(`[coordinator] Spawning coordinator agent: ${config.name || group.name}`);
228
+ console.log(`[coordinator] Model: ${config.model || 'claude-sonnet-4-5'}`);
229
+ console.log(`[coordinator] Repositories: ${repositories.map((r) => r.githubFullName).join(', ')}`);
230
+ console.log(`[coordinator] Connected workspaces: ${workspaces.filter(Boolean).length}`);
231
+ // Register message handler for this coordinator
232
+ messageHandlers.set(`coordinator:${group.id}`, async (message) => {
233
+ console.log(`[coordinator:${group.id}] Received: ${message.body.substring(0, 100)}...`);
234
+ // In a full implementation, this would:
235
+ // 1. Parse the message for coordinator commands
236
+ // 2. Route to appropriate workspace(s)
237
+ // 3. Track conversation state
238
+ });
239
+ // Store coordinator connection info in Redis for workspace discovery
240
+ if (pubClient) {
241
+ await pubClient.hSet(`coordinator:${group.id}`, {
242
+ groupId: group.id,
243
+ groupName: group.name,
244
+ agentName: config.name || `${group.name} Coordinator`,
245
+ model: config.model || 'claude-sonnet-4-5',
246
+ startedAt: new Date().toISOString(),
247
+ workspaces: JSON.stringify(Array.from(workspaceIds)),
248
+ systemPrompt,
249
+ });
250
+ await pubClient.expire(`coordinator:${group.id}`, 86400); // 24h TTL
251
+ }
252
+ }
253
+ /**
254
+ * Stop the actual coordinator agent
255
+ */
256
+ async function stopCoordinatorAgent(groupId, state) {
257
+ console.log(`[coordinator] Stopping coordinator agent for group ${groupId}: ${state.agentName}`);
258
+ // Remove message handler
259
+ messageHandlers.delete(`coordinator:${groupId}`);
260
+ // Remove from Redis
261
+ if (pubClient) {
262
+ await pubClient.del(`coordinator:${groupId}`);
263
+ }
264
+ }
265
+ /**
266
+ * Route a message from a workspace to the coordinator
267
+ */
268
+ export async function routeToCoordinator(workspaceId, agentName, message, thread) {
269
+ if (!pubClient) {
270
+ await initRedisClients();
271
+ }
272
+ // Find which coordinator is managing this workspace
273
+ // by scanning all coordinator keys
274
+ const coordinatorKeys = await pubClient.keys('coordinator:*');
275
+ for (const key of coordinatorKeys) {
276
+ if (key === 'coordinator:messages')
277
+ continue; // Skip the channel
278
+ const data = await pubClient.hGetAll(key);
279
+ if (data.workspaces) {
280
+ const workspaces = JSON.parse(data.workspaces);
281
+ if (workspaces.includes(workspaceId)) {
282
+ // Found the coordinator for this workspace
283
+ const crossMessage = {
284
+ type: 'relay',
285
+ from: agentName,
286
+ fromWorkspace: workspaceId,
287
+ to: 'Coordinator',
288
+ toWorkspace: key,
289
+ body: message,
290
+ thread,
291
+ timestamp: new Date().toISOString(),
292
+ };
293
+ await pubClient.publish('coordinator:messages', JSON.stringify(crossMessage));
294
+ console.log(`[coordinator] Routed message from ${workspaceId}:${agentName} to ${key}`);
295
+ return;
296
+ }
297
+ }
298
+ }
299
+ console.warn(`[coordinator] No coordinator found for workspace ${workspaceId}`);
300
+ }
301
+ /**
302
+ * Get all active coordinators
303
+ */
304
+ export async function getActiveCoordinators() {
305
+ if (!pubClient) {
306
+ await initRedisClients();
307
+ }
308
+ const coordinatorKeys = await pubClient.keys('coordinator:*');
309
+ const coordinators = [];
310
+ for (const key of coordinatorKeys) {
311
+ if (key === 'coordinator:messages')
312
+ continue;
313
+ const data = await pubClient.hGetAll(key);
314
+ if (data.groupId) {
315
+ coordinators.push({
316
+ groupId: data.groupId,
317
+ groupName: data.groupName,
318
+ agentName: data.agentName,
319
+ model: data.model,
320
+ startedAt: data.startedAt,
321
+ workspaces: data.workspaces ? JSON.parse(data.workspaces) : [],
322
+ });
323
+ }
324
+ }
325
+ return coordinators;
326
+ }
327
+ /**
328
+ * Build system prompt for coordinator agent
329
+ */
330
+ function buildCoordinatorSystemPrompt(group, config, repositories) {
331
+ const repoList = repositories.map((r) => r.githubFullName).join('\n- ');
332
+ let prompt = `You are the coordinator agent for the "${group.name}" project group.
333
+
334
+ Your role is to oversee and orchestrate work across the following repositories:
335
+ - ${repoList}
336
+
337
+ `;
338
+ if (config.capabilities && config.capabilities.length > 0) {
339
+ prompt += `You have the following capabilities:
340
+ ${config.capabilities.map((c) => `- ${c}`).join('\n')}
341
+
342
+ `;
343
+ }
344
+ if (config.systemPrompt) {
345
+ prompt += `${config.systemPrompt}\n\n`;
346
+ }
347
+ prompt += `When coordinating work:
348
+ 1. Monitor all repositories in your group
349
+ 2. Identify dependencies and coordination points
350
+ 3. Delegate tasks to project-specific agents when appropriate
351
+ 4. Ensure consistency across repositories
352
+ 5. Report status and blockers to the team
353
+
354
+ Use the Agent Relay messaging system to communicate with other agents and team members.`;
355
+ return prompt;
356
+ }
357
+ /**
358
+ * Singleton instance
359
+ */
360
+ let coordinatorServiceInstance = null;
361
+ /**
362
+ * Get the coordinator service singleton
363
+ */
364
+ export function getCoordinatorService() {
365
+ if (!coordinatorServiceInstance) {
366
+ coordinatorServiceInstance = {
367
+ start,
368
+ stop,
369
+ restart,
370
+ getStatus,
371
+ listActive,
372
+ };
373
+ }
374
+ return coordinatorServiceInstance;
375
+ }
376
+ /**
377
+ * Initialize coordinator service
378
+ * Restarts any coordinators that should be running
379
+ */
380
+ export async function initializeCoordinatorService() {
381
+ console.log('Initializing coordinator service...');
382
+ // In a production system, this would:
383
+ // 1. Query database for all enabled coordinators
384
+ // 2. Check their expected state
385
+ // 3. Restart any that should be running
386
+ // For now, just log initialization
387
+ console.log('Coordinator service initialized');
388
+ }
389
+ //# sourceMappingURL=coordinator.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"coordinator.js","sourceRoot":"","sources":["../../../src/cloud/services/coordinator.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,110 @@
1
+ /**
2
+ * Plan Limits Service
3
+ *
4
+ * Defines resource limits for each plan tier and provides
5
+ * functions to check if users are within their limits.
6
+ */
7
+ import { PlanType } from '../db/index.js';
8
+ /**
9
+ * Resource limits for each plan tier
10
+ */
11
+ export interface PlanLimits {
12
+ maxWorkspaces: number;
13
+ maxRepos: number;
14
+ maxConcurrentAgents: number;
15
+ maxComputeHoursPerMonth: number;
16
+ coordinatorsEnabled: boolean;
17
+ }
18
+ /**
19
+ * Plan limits configuration
20
+ *
21
+ * Free: Try it out on a side project
22
+ * Pro: Professional developers, coordinators enabled
23
+ * Team: Growing teams with advanced needs
24
+ * Enterprise: Unlimited everything
25
+ */
26
+ export declare const PLAN_LIMITS: Record<PlanType, PlanLimits>;
27
+ /**
28
+ * Get plan limits for a given plan type
29
+ */
30
+ export declare function getPlanLimits(plan: PlanType): PlanLimits;
31
+ /**
32
+ * Current usage for a user
33
+ */
34
+ export interface UserUsage {
35
+ workspaceCount: number;
36
+ repoCount: number;
37
+ concurrentAgents: number;
38
+ computeHoursThisMonth: number;
39
+ }
40
+ /**
41
+ * Get current usage for a user
42
+ */
43
+ export declare function getUserUsage(userId: string): Promise<UserUsage>;
44
+ /**
45
+ * Check if user can create a new workspace
46
+ */
47
+ export declare function canCreateWorkspace(userId: string): Promise<{
48
+ allowed: boolean;
49
+ reason?: string;
50
+ limit?: number;
51
+ current?: number;
52
+ }>;
53
+ /**
54
+ * Check if user can add a new repo
55
+ */
56
+ export declare function canAddRepo(userId: string): Promise<{
57
+ allowed: boolean;
58
+ reason?: string;
59
+ limit?: number;
60
+ current?: number;
61
+ }>;
62
+ /**
63
+ * Check if user can spawn another agent (concurrent agent limit)
64
+ */
65
+ export declare function canSpawnAgent(userId: string, currentRunningAgents?: number): Promise<{
66
+ allowed: boolean;
67
+ reason?: string;
68
+ limit?: number;
69
+ current?: number;
70
+ }>;
71
+ /**
72
+ * Check if user can use coordinator agents
73
+ */
74
+ export declare function canUseCoordinator(userId: string): Promise<{
75
+ allowed: boolean;
76
+ reason?: string;
77
+ requiredPlan?: string;
78
+ }>;
79
+ /**
80
+ * Check if user has compute hours available
81
+ */
82
+ export declare function hasComputeHoursAvailable(userId: string): Promise<{
83
+ available: boolean;
84
+ reason?: string;
85
+ limit?: number;
86
+ current?: number;
87
+ }>;
88
+ /**
89
+ * Get remaining quota for a user
90
+ */
91
+ export declare function getRemainingQuota(userId: string): Promise<{
92
+ plan: PlanType;
93
+ limits: PlanLimits;
94
+ usage: UserUsage;
95
+ remaining: {
96
+ workspaces: number;
97
+ repos: number;
98
+ concurrentAgents: number;
99
+ computeHours: number;
100
+ };
101
+ }>;
102
+ /**
103
+ * Record compute usage
104
+ */
105
+ export declare function recordComputeUsage(userId: string, workspaceId: string, hours: number): Promise<void>;
106
+ /**
107
+ * Update active agent count for a user
108
+ */
109
+ export declare function updateActiveAgentCount(userId: string, workspaceId: string, count: number): Promise<void>;
110
+ //# sourceMappingURL=planLimits.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"planLimits.d.ts","sourceRoot":"","sources":["../../../src/cloud/services/planLimits.ts"],"names":[],"mappings":"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"}