@agent-relay/cloud 0.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 (269) hide show
  1. package/dist/api/admin.d.ts +8 -0
  2. package/dist/api/admin.d.ts.map +1 -0
  3. package/dist/api/admin.js +225 -0
  4. package/dist/api/admin.js.map +1 -0
  5. package/dist/api/auth.d.ts +20 -0
  6. package/dist/api/auth.d.ts.map +1 -0
  7. package/dist/api/auth.js +136 -0
  8. package/dist/api/auth.js.map +1 -0
  9. package/dist/api/billing.d.ts +7 -0
  10. package/dist/api/billing.d.ts.map +1 -0
  11. package/dist/api/billing.js +564 -0
  12. package/dist/api/billing.js.map +1 -0
  13. package/dist/api/cli-pty-runner.d.ts +53 -0
  14. package/dist/api/cli-pty-runner.d.ts.map +1 -0
  15. package/dist/api/cli-pty-runner.js +193 -0
  16. package/dist/api/cli-pty-runner.js.map +1 -0
  17. package/dist/api/codex-auth-helper.d.ts +21 -0
  18. package/dist/api/codex-auth-helper.d.ts.map +1 -0
  19. package/dist/api/codex-auth-helper.js +327 -0
  20. package/dist/api/codex-auth-helper.js.map +1 -0
  21. package/dist/api/consensus.d.ts +13 -0
  22. package/dist/api/consensus.d.ts.map +1 -0
  23. package/dist/api/consensus.js +261 -0
  24. package/dist/api/consensus.js.map +1 -0
  25. package/dist/api/coordinators.d.ts +8 -0
  26. package/dist/api/coordinators.d.ts.map +1 -0
  27. package/dist/api/coordinators.js +750 -0
  28. package/dist/api/coordinators.js.map +1 -0
  29. package/dist/api/daemons.d.ts +12 -0
  30. package/dist/api/daemons.d.ts.map +1 -0
  31. package/dist/api/daemons.js +535 -0
  32. package/dist/api/daemons.js.map +1 -0
  33. package/dist/api/generic-webhooks.d.ts +8 -0
  34. package/dist/api/generic-webhooks.d.ts.map +1 -0
  35. package/dist/api/generic-webhooks.js +129 -0
  36. package/dist/api/generic-webhooks.js.map +1 -0
  37. package/dist/api/git.d.ts +8 -0
  38. package/dist/api/git.d.ts.map +1 -0
  39. package/dist/api/git.js +269 -0
  40. package/dist/api/git.js.map +1 -0
  41. package/dist/api/github-app.d.ts +11 -0
  42. package/dist/api/github-app.d.ts.map +1 -0
  43. package/dist/api/github-app.js +223 -0
  44. package/dist/api/github-app.js.map +1 -0
  45. package/dist/api/middleware/planLimits.d.ts +43 -0
  46. package/dist/api/middleware/planLimits.d.ts.map +1 -0
  47. package/dist/api/middleware/planLimits.js +202 -0
  48. package/dist/api/middleware/planLimits.js.map +1 -0
  49. package/dist/api/monitoring.d.ts +11 -0
  50. package/dist/api/monitoring.d.ts.map +1 -0
  51. package/dist/api/monitoring.js +578 -0
  52. package/dist/api/monitoring.js.map +1 -0
  53. package/dist/api/nango-auth.d.ts +9 -0
  54. package/dist/api/nango-auth.d.ts.map +1 -0
  55. package/dist/api/nango-auth.js +674 -0
  56. package/dist/api/nango-auth.js.map +1 -0
  57. package/dist/api/onboarding.d.ts +15 -0
  58. package/dist/api/onboarding.d.ts.map +1 -0
  59. package/dist/api/onboarding.js +679 -0
  60. package/dist/api/onboarding.js.map +1 -0
  61. package/dist/api/policy.d.ts +8 -0
  62. package/dist/api/policy.d.ts.map +1 -0
  63. package/dist/api/policy.js +229 -0
  64. package/dist/api/policy.js.map +1 -0
  65. package/dist/api/provider-env.d.ts +14 -0
  66. package/dist/api/provider-env.d.ts.map +1 -0
  67. package/dist/api/provider-env.js +75 -0
  68. package/dist/api/provider-env.js.map +1 -0
  69. package/dist/api/providers.d.ts +7 -0
  70. package/dist/api/providers.d.ts.map +1 -0
  71. package/dist/api/providers.js +564 -0
  72. package/dist/api/providers.js.map +1 -0
  73. package/dist/api/repos.d.ts +8 -0
  74. package/dist/api/repos.d.ts.map +1 -0
  75. package/dist/api/repos.js +577 -0
  76. package/dist/api/repos.js.map +1 -0
  77. package/dist/api/sessions.d.ts +11 -0
  78. package/dist/api/sessions.d.ts.map +1 -0
  79. package/dist/api/sessions.js +302 -0
  80. package/dist/api/sessions.js.map +1 -0
  81. package/dist/api/teams.d.ts +7 -0
  82. package/dist/api/teams.d.ts.map +1 -0
  83. package/dist/api/teams.js +281 -0
  84. package/dist/api/teams.js.map +1 -0
  85. package/dist/api/test-helpers.d.ts +10 -0
  86. package/dist/api/test-helpers.d.ts.map +1 -0
  87. package/dist/api/test-helpers.js +745 -0
  88. package/dist/api/test-helpers.js.map +1 -0
  89. package/dist/api/usage.d.ts +7 -0
  90. package/dist/api/usage.d.ts.map +1 -0
  91. package/dist/api/usage.js +111 -0
  92. package/dist/api/usage.js.map +1 -0
  93. package/dist/api/webhooks.d.ts +8 -0
  94. package/dist/api/webhooks.d.ts.map +1 -0
  95. package/dist/api/webhooks.js +645 -0
  96. package/dist/api/webhooks.js.map +1 -0
  97. package/dist/api/workspaces.d.ts +25 -0
  98. package/dist/api/workspaces.d.ts.map +1 -0
  99. package/dist/api/workspaces.js +1799 -0
  100. package/dist/api/workspaces.js.map +1 -0
  101. package/dist/billing/index.d.ts +9 -0
  102. package/dist/billing/index.d.ts.map +1 -0
  103. package/dist/billing/index.js +9 -0
  104. package/dist/billing/index.js.map +1 -0
  105. package/dist/billing/plans.d.ts +39 -0
  106. package/dist/billing/plans.d.ts.map +1 -0
  107. package/dist/billing/plans.js +245 -0
  108. package/dist/billing/plans.js.map +1 -0
  109. package/dist/billing/service.d.ts +80 -0
  110. package/dist/billing/service.d.ts.map +1 -0
  111. package/dist/billing/service.js +388 -0
  112. package/dist/billing/service.js.map +1 -0
  113. package/dist/billing/types.d.ts +141 -0
  114. package/dist/billing/types.d.ts.map +1 -0
  115. package/dist/billing/types.js +7 -0
  116. package/dist/billing/types.js.map +1 -0
  117. package/dist/config.d.ts +5 -0
  118. package/dist/config.d.ts.map +1 -0
  119. package/dist/config.js +5 -0
  120. package/dist/config.js.map +1 -0
  121. package/dist/db/bulk-ingest.d.ts +89 -0
  122. package/dist/db/bulk-ingest.d.ts.map +1 -0
  123. package/dist/db/bulk-ingest.js +268 -0
  124. package/dist/db/bulk-ingest.js.map +1 -0
  125. package/dist/db/drizzle.d.ts +256 -0
  126. package/dist/db/drizzle.d.ts.map +1 -0
  127. package/dist/db/drizzle.js +1286 -0
  128. package/dist/db/drizzle.js.map +1 -0
  129. package/dist/db/index.d.ts +55 -0
  130. package/dist/db/index.d.ts.map +1 -0
  131. package/dist/db/index.js +68 -0
  132. package/dist/db/index.js.map +1 -0
  133. package/dist/db/schema.d.ts +4873 -0
  134. package/dist/db/schema.d.ts.map +1 -0
  135. package/dist/db/schema.js +620 -0
  136. package/dist/db/schema.js.map +1 -0
  137. package/dist/index.d.ts +11 -0
  138. package/dist/index.d.ts.map +1 -0
  139. package/dist/index.js +38 -0
  140. package/dist/index.js.map +1 -0
  141. package/dist/provisioner/index.d.ts +207 -0
  142. package/dist/provisioner/index.d.ts.map +1 -0
  143. package/dist/provisioner/index.js +2114 -0
  144. package/dist/provisioner/index.js.map +1 -0
  145. package/dist/server.d.ts +17 -0
  146. package/dist/server.d.ts.map +1 -0
  147. package/dist/server.js +1924 -0
  148. package/dist/server.js.map +1 -0
  149. package/dist/services/auto-scaler.d.ts +152 -0
  150. package/dist/services/auto-scaler.d.ts.map +1 -0
  151. package/dist/services/auto-scaler.js +439 -0
  152. package/dist/services/auto-scaler.js.map +1 -0
  153. package/dist/services/capacity-manager.d.ts +148 -0
  154. package/dist/services/capacity-manager.d.ts.map +1 -0
  155. package/dist/services/capacity-manager.js +449 -0
  156. package/dist/services/capacity-manager.js.map +1 -0
  157. package/dist/services/ci-agent-spawner.d.ts +49 -0
  158. package/dist/services/ci-agent-spawner.d.ts.map +1 -0
  159. package/dist/services/ci-agent-spawner.js +373 -0
  160. package/dist/services/ci-agent-spawner.js.map +1 -0
  161. package/dist/services/cloud-message-bus.d.ts +28 -0
  162. package/dist/services/cloud-message-bus.d.ts.map +1 -0
  163. package/dist/services/cloud-message-bus.js +19 -0
  164. package/dist/services/cloud-message-bus.js.map +1 -0
  165. package/dist/services/compute-enforcement.d.ts +57 -0
  166. package/dist/services/compute-enforcement.d.ts.map +1 -0
  167. package/dist/services/compute-enforcement.js +175 -0
  168. package/dist/services/compute-enforcement.js.map +1 -0
  169. package/dist/services/coordinator.d.ts +62 -0
  170. package/dist/services/coordinator.d.ts.map +1 -0
  171. package/dist/services/coordinator.js +389 -0
  172. package/dist/services/coordinator.js.map +1 -0
  173. package/dist/services/index.d.ts +17 -0
  174. package/dist/services/index.d.ts.map +1 -0
  175. package/dist/services/index.js +25 -0
  176. package/dist/services/index.js.map +1 -0
  177. package/dist/services/intro-expiration.d.ts +60 -0
  178. package/dist/services/intro-expiration.d.ts.map +1 -0
  179. package/dist/services/intro-expiration.js +252 -0
  180. package/dist/services/intro-expiration.js.map +1 -0
  181. package/dist/services/mention-handler.d.ts +65 -0
  182. package/dist/services/mention-handler.d.ts.map +1 -0
  183. package/dist/services/mention-handler.js +405 -0
  184. package/dist/services/mention-handler.js.map +1 -0
  185. package/dist/services/nango.d.ts +201 -0
  186. package/dist/services/nango.d.ts.map +1 -0
  187. package/dist/services/nango.js +392 -0
  188. package/dist/services/nango.js.map +1 -0
  189. package/dist/services/persistence.d.ts +131 -0
  190. package/dist/services/persistence.d.ts.map +1 -0
  191. package/dist/services/persistence.js +200 -0
  192. package/dist/services/persistence.js.map +1 -0
  193. package/dist/services/planLimits.d.ts +147 -0
  194. package/dist/services/planLimits.d.ts.map +1 -0
  195. package/dist/services/planLimits.js +335 -0
  196. package/dist/services/planLimits.js.map +1 -0
  197. package/dist/services/presence-registry.d.ts +56 -0
  198. package/dist/services/presence-registry.d.ts.map +1 -0
  199. package/dist/services/presence-registry.js +91 -0
  200. package/dist/services/presence-registry.js.map +1 -0
  201. package/dist/services/scaling-orchestrator.d.ts +159 -0
  202. package/dist/services/scaling-orchestrator.d.ts.map +1 -0
  203. package/dist/services/scaling-orchestrator.js +502 -0
  204. package/dist/services/scaling-orchestrator.js.map +1 -0
  205. package/dist/services/scaling-policy.d.ts +121 -0
  206. package/dist/services/scaling-policy.d.ts.map +1 -0
  207. package/dist/services/scaling-policy.js +415 -0
  208. package/dist/services/scaling-policy.js.map +1 -0
  209. package/dist/services/ssh-security.d.ts +31 -0
  210. package/dist/services/ssh-security.d.ts.map +1 -0
  211. package/dist/services/ssh-security.js +63 -0
  212. package/dist/services/ssh-security.js.map +1 -0
  213. package/dist/services/workspace-keepalive.d.ts +76 -0
  214. package/dist/services/workspace-keepalive.d.ts.map +1 -0
  215. package/dist/services/workspace-keepalive.js +234 -0
  216. package/dist/services/workspace-keepalive.js.map +1 -0
  217. package/dist/shims/consensus.d.ts +23 -0
  218. package/dist/shims/consensus.d.ts.map +1 -0
  219. package/dist/shims/consensus.js +5 -0
  220. package/dist/shims/consensus.js.map +1 -0
  221. package/dist/webhooks/index.d.ts +24 -0
  222. package/dist/webhooks/index.d.ts.map +1 -0
  223. package/dist/webhooks/index.js +29 -0
  224. package/dist/webhooks/index.js.map +1 -0
  225. package/dist/webhooks/parsers/github.d.ts +8 -0
  226. package/dist/webhooks/parsers/github.d.ts.map +1 -0
  227. package/dist/webhooks/parsers/github.js +234 -0
  228. package/dist/webhooks/parsers/github.js.map +1 -0
  229. package/dist/webhooks/parsers/index.d.ts +23 -0
  230. package/dist/webhooks/parsers/index.d.ts.map +1 -0
  231. package/dist/webhooks/parsers/index.js +30 -0
  232. package/dist/webhooks/parsers/index.js.map +1 -0
  233. package/dist/webhooks/parsers/linear.d.ts +9 -0
  234. package/dist/webhooks/parsers/linear.d.ts.map +1 -0
  235. package/dist/webhooks/parsers/linear.js +258 -0
  236. package/dist/webhooks/parsers/linear.js.map +1 -0
  237. package/dist/webhooks/parsers/slack.d.ts +9 -0
  238. package/dist/webhooks/parsers/slack.d.ts.map +1 -0
  239. package/dist/webhooks/parsers/slack.js +214 -0
  240. package/dist/webhooks/parsers/slack.js.map +1 -0
  241. package/dist/webhooks/responders/github.d.ts +8 -0
  242. package/dist/webhooks/responders/github.d.ts.map +1 -0
  243. package/dist/webhooks/responders/github.js +73 -0
  244. package/dist/webhooks/responders/github.js.map +1 -0
  245. package/dist/webhooks/responders/index.d.ts +23 -0
  246. package/dist/webhooks/responders/index.d.ts.map +1 -0
  247. package/dist/webhooks/responders/index.js +30 -0
  248. package/dist/webhooks/responders/index.js.map +1 -0
  249. package/dist/webhooks/responders/linear.d.ts +9 -0
  250. package/dist/webhooks/responders/linear.d.ts.map +1 -0
  251. package/dist/webhooks/responders/linear.js +149 -0
  252. package/dist/webhooks/responders/linear.js.map +1 -0
  253. package/dist/webhooks/responders/slack.d.ts +20 -0
  254. package/dist/webhooks/responders/slack.d.ts.map +1 -0
  255. package/dist/webhooks/responders/slack.js +178 -0
  256. package/dist/webhooks/responders/slack.js.map +1 -0
  257. package/dist/webhooks/router.d.ts +25 -0
  258. package/dist/webhooks/router.d.ts.map +1 -0
  259. package/dist/webhooks/router.js +504 -0
  260. package/dist/webhooks/router.js.map +1 -0
  261. package/dist/webhooks/rules-engine.d.ts +24 -0
  262. package/dist/webhooks/rules-engine.d.ts.map +1 -0
  263. package/dist/webhooks/rules-engine.js +287 -0
  264. package/dist/webhooks/rules-engine.js.map +1 -0
  265. package/dist/webhooks/types.d.ts +186 -0
  266. package/dist/webhooks/types.d.ts.map +1 -0
  267. package/dist/webhooks/types.js +8 -0
  268. package/dist/webhooks/types.js.map +1 -0
  269. package/package.json +55 -0
@@ -0,0 +1,750 @@
1
+ /**
2
+ * Coordinator Agent API Routes
3
+ *
4
+ * Manage coordinator agents for project groups.
5
+ * Coordinators oversee and orchestrate work across repositories in a group.
6
+ */
7
+ import { Router } from 'express';
8
+ import { requireAuth } from './auth.js';
9
+ import { checkCoordinatorAccess } from './middleware/planLimits.js';
10
+ import { db } from '../db/index.js';
11
+ import { getCoordinatorService, sendToWorkspace, broadcastToGroup, routeToCoordinator, getActiveCoordinators, } from '../services/coordinator.js';
12
+ export const coordinatorsRouter = Router();
13
+ // All routes require authentication
14
+ coordinatorsRouter.use(requireAuth);
15
+ // Coordinator modification routes require Pro plan or higher
16
+ const coordinatorWriteRoutes = [
17
+ '/:groupId/coordinator/enable',
18
+ '/:groupId/coordinator/disable',
19
+ ];
20
+ coordinatorWriteRoutes.forEach(route => {
21
+ coordinatorsRouter.use(route, checkCoordinatorAccess);
22
+ });
23
+ // ============================================================================
24
+ // Project Group CRUD Routes
25
+ // These must come BEFORE the /:groupId/coordinator routes
26
+ // ============================================================================
27
+ /**
28
+ * GET /api/project-groups
29
+ * List all project groups for the authenticated user
30
+ */
31
+ coordinatorsRouter.get('/', async (req, res) => {
32
+ const userId = req.session.userId;
33
+ try {
34
+ const result = await db.projectGroups.findAllWithRepositories(userId);
35
+ res.json({
36
+ groups: result.groups.map(group => ({
37
+ id: group.id,
38
+ name: group.name,
39
+ description: group.description,
40
+ color: group.color,
41
+ icon: group.icon,
42
+ coordinatorAgent: group.coordinatorAgent,
43
+ sortOrder: group.sortOrder,
44
+ repositoryCount: group.repositories.length,
45
+ repositories: group.repositories.map(repo => ({
46
+ id: repo.id,
47
+ githubFullName: repo.githubFullName,
48
+ defaultBranch: repo.defaultBranch,
49
+ isPrivate: repo.isPrivate,
50
+ })),
51
+ createdAt: group.createdAt,
52
+ updatedAt: group.updatedAt,
53
+ })),
54
+ ungroupedRepositories: result.ungroupedRepositories.map(repo => ({
55
+ id: repo.id,
56
+ githubFullName: repo.githubFullName,
57
+ defaultBranch: repo.defaultBranch,
58
+ isPrivate: repo.isPrivate,
59
+ workspaceId: repo.workspaceId,
60
+ })),
61
+ });
62
+ }
63
+ catch (error) {
64
+ console.error('Error listing project groups:', error);
65
+ res.status(500).json({ error: 'Failed to list project groups' });
66
+ }
67
+ });
68
+ /**
69
+ * POST /api/project-groups
70
+ * Create a new project group
71
+ */
72
+ coordinatorsRouter.post('/', async (req, res) => {
73
+ const userId = req.session.userId;
74
+ const { name, description, color, icon, repositoryIds } = req.body;
75
+ if (!name || typeof name !== 'string' || name.trim().length === 0) {
76
+ return res.status(400).json({ error: 'Name is required' });
77
+ }
78
+ if (name.length > 255) {
79
+ return res.status(400).json({ error: 'Name must be 255 characters or less' });
80
+ }
81
+ try {
82
+ // Check for duplicate name
83
+ const existing = await db.projectGroups.findByName(userId, name.trim());
84
+ if (existing) {
85
+ return res.status(409).json({ error: 'A project group with this name already exists' });
86
+ }
87
+ // Create the group
88
+ const group = await db.projectGroups.create({
89
+ userId,
90
+ name: name.trim(),
91
+ description: description?.trim() || null,
92
+ color: color || null,
93
+ icon: icon || null,
94
+ coordinatorAgent: { enabled: false },
95
+ sortOrder: 0,
96
+ });
97
+ // Assign repositories to the group if provided
98
+ if (repositoryIds && Array.isArray(repositoryIds) && repositoryIds.length > 0) {
99
+ // Verify all repositories belong to the user
100
+ const userRepos = await db.repositories.findByUserId(userId);
101
+ const userRepoIds = new Set(userRepos.map(r => r.id));
102
+ for (const repoId of repositoryIds) {
103
+ if (!userRepoIds.has(repoId)) {
104
+ return res.status(400).json({
105
+ error: `Repository ${repoId} not found or not owned by user`,
106
+ });
107
+ }
108
+ }
109
+ // Assign repositories to the group
110
+ for (const repoId of repositoryIds) {
111
+ await db.repositories.assignToGroup(repoId, group.id);
112
+ }
113
+ }
114
+ // Fetch the group with repositories for response
115
+ const groupWithRepos = await db.projectGroups.findWithRepositories(group.id);
116
+ res.status(201).json({
117
+ success: true,
118
+ group: {
119
+ id: groupWithRepos.id,
120
+ name: groupWithRepos.name,
121
+ description: groupWithRepos.description,
122
+ color: groupWithRepos.color,
123
+ icon: groupWithRepos.icon,
124
+ coordinatorAgent: groupWithRepos.coordinatorAgent,
125
+ repositories: groupWithRepos.repositories.map(repo => ({
126
+ id: repo.id,
127
+ githubFullName: repo.githubFullName,
128
+ defaultBranch: repo.defaultBranch,
129
+ isPrivate: repo.isPrivate,
130
+ })),
131
+ createdAt: groupWithRepos.createdAt,
132
+ updatedAt: groupWithRepos.updatedAt,
133
+ },
134
+ });
135
+ }
136
+ catch (error) {
137
+ console.error('Error creating project group:', error);
138
+ res.status(500).json({ error: 'Failed to create project group' });
139
+ }
140
+ });
141
+ /**
142
+ * GET /api/project-groups/:id
143
+ * Get a specific project group with its repositories
144
+ */
145
+ coordinatorsRouter.get('/:id', async (req, res) => {
146
+ const userId = req.session.userId;
147
+ const id = req.params.id;
148
+ // Skip if this looks like a coordinator route
149
+ if (id === 'coordinators') {
150
+ return res.status(404).json({ error: 'Not found' });
151
+ }
152
+ try {
153
+ const group = await db.projectGroups.findWithRepositories(id);
154
+ if (!group) {
155
+ return res.status(404).json({ error: 'Project group not found' });
156
+ }
157
+ if (group.userId !== userId) {
158
+ return res.status(403).json({ error: 'Unauthorized' });
159
+ }
160
+ res.json({
161
+ id: group.id,
162
+ name: group.name,
163
+ description: group.description,
164
+ color: group.color,
165
+ icon: group.icon,
166
+ coordinatorAgent: group.coordinatorAgent,
167
+ sortOrder: group.sortOrder,
168
+ repositories: group.repositories.map(repo => ({
169
+ id: repo.id,
170
+ githubFullName: repo.githubFullName,
171
+ defaultBranch: repo.defaultBranch,
172
+ isPrivate: repo.isPrivate,
173
+ syncStatus: repo.syncStatus,
174
+ lastSyncedAt: repo.lastSyncedAt,
175
+ workspaceId: repo.workspaceId,
176
+ })),
177
+ createdAt: group.createdAt,
178
+ updatedAt: group.updatedAt,
179
+ });
180
+ }
181
+ catch (error) {
182
+ console.error('Error getting project group:', error);
183
+ res.status(500).json({ error: 'Failed to get project group' });
184
+ }
185
+ });
186
+ /**
187
+ * PATCH /api/project-groups/:id
188
+ * Update a project group's metadata
189
+ */
190
+ coordinatorsRouter.patch('/:id', async (req, res) => {
191
+ const userId = req.session.userId;
192
+ const id = req.params.id;
193
+ const { name, description, color, icon } = req.body;
194
+ try {
195
+ const group = await db.projectGroups.findById(id);
196
+ if (!group) {
197
+ return res.status(404).json({ error: 'Project group not found' });
198
+ }
199
+ if (group.userId !== userId) {
200
+ return res.status(403).json({ error: 'Unauthorized' });
201
+ }
202
+ // Build update object with only provided fields
203
+ const updates = {};
204
+ if (name !== undefined) {
205
+ if (typeof name !== 'string' || name.trim().length === 0) {
206
+ return res.status(400).json({ error: 'Name cannot be empty' });
207
+ }
208
+ if (name.length > 255) {
209
+ return res.status(400).json({ error: 'Name must be 255 characters or less' });
210
+ }
211
+ // Check for duplicate name (excluding current group)
212
+ const existing = await db.projectGroups.findByName(userId, name.trim());
213
+ if (existing && existing.id !== id) {
214
+ return res.status(409).json({ error: 'A project group with this name already exists' });
215
+ }
216
+ updates.name = name.trim();
217
+ }
218
+ if (description !== undefined) {
219
+ updates.description = description?.trim() || null;
220
+ }
221
+ if (color !== undefined) {
222
+ // Validate hex color format if provided
223
+ if (color && !/^#[0-9A-Fa-f]{6}$/.test(color)) {
224
+ return res.status(400).json({ error: 'Color must be a valid hex color (e.g., #3B82F6)' });
225
+ }
226
+ updates.color = color || null;
227
+ }
228
+ if (icon !== undefined) {
229
+ updates.icon = icon || null;
230
+ }
231
+ if (Object.keys(updates).length === 0) {
232
+ return res.status(400).json({ error: 'No valid fields to update' });
233
+ }
234
+ await db.projectGroups.update(id, updates);
235
+ // Fetch updated group
236
+ const updatedGroup = await db.projectGroups.findWithRepositories(id);
237
+ res.json({
238
+ success: true,
239
+ group: {
240
+ id: updatedGroup.id,
241
+ name: updatedGroup.name,
242
+ description: updatedGroup.description,
243
+ color: updatedGroup.color,
244
+ icon: updatedGroup.icon,
245
+ coordinatorAgent: updatedGroup.coordinatorAgent,
246
+ repositories: updatedGroup.repositories.map(repo => ({
247
+ id: repo.id,
248
+ githubFullName: repo.githubFullName,
249
+ defaultBranch: repo.defaultBranch,
250
+ isPrivate: repo.isPrivate,
251
+ })),
252
+ updatedAt: updatedGroup.updatedAt,
253
+ },
254
+ });
255
+ }
256
+ catch (error) {
257
+ console.error('Error updating project group:', error);
258
+ res.status(500).json({ error: 'Failed to update project group' });
259
+ }
260
+ });
261
+ /**
262
+ * DELETE /api/project-groups/:id
263
+ * Delete a project group (repositories are unassigned, not deleted)
264
+ */
265
+ coordinatorsRouter.delete('/:id', async (req, res) => {
266
+ const userId = req.session.userId;
267
+ const id = req.params.id;
268
+ try {
269
+ const group = await db.projectGroups.findById(id);
270
+ if (!group) {
271
+ return res.status(404).json({ error: 'Project group not found' });
272
+ }
273
+ if (group.userId !== userId) {
274
+ return res.status(403).json({ error: 'Unauthorized' });
275
+ }
276
+ // Stop coordinator if running
277
+ if (group.coordinatorAgent?.enabled) {
278
+ try {
279
+ const coordinatorService = getCoordinatorService();
280
+ await coordinatorService.stop(id);
281
+ }
282
+ catch (err) {
283
+ console.warn('Error stopping coordinator during group deletion:', err);
284
+ }
285
+ }
286
+ // Delete the group (repositories will have projectGroupId set to null due to ON DELETE SET NULL)
287
+ await db.projectGroups.delete(id);
288
+ res.json({
289
+ success: true,
290
+ message: 'Project group deleted',
291
+ });
292
+ }
293
+ catch (error) {
294
+ console.error('Error deleting project group:', error);
295
+ res.status(500).json({ error: 'Failed to delete project group' });
296
+ }
297
+ });
298
+ /**
299
+ * POST /api/project-groups/:id/repositories
300
+ * Add repositories to a project group
301
+ */
302
+ coordinatorsRouter.post('/:id/repositories', async (req, res) => {
303
+ const userId = req.session.userId;
304
+ const id = req.params.id;
305
+ const { repositoryIds } = req.body;
306
+ if (!repositoryIds || !Array.isArray(repositoryIds) || repositoryIds.length === 0) {
307
+ return res.status(400).json({ error: 'repositoryIds array is required' });
308
+ }
309
+ try {
310
+ const group = await db.projectGroups.findById(id);
311
+ if (!group) {
312
+ return res.status(404).json({ error: 'Project group not found' });
313
+ }
314
+ if (group.userId !== userId) {
315
+ return res.status(403).json({ error: 'Unauthorized' });
316
+ }
317
+ // Verify all repositories belong to the user
318
+ const userRepos = await db.repositories.findByUserId(userId);
319
+ const userRepoIds = new Set(userRepos.map(r => r.id));
320
+ for (const repoId of repositoryIds) {
321
+ if (!userRepoIds.has(repoId)) {
322
+ return res.status(400).json({
323
+ error: `Repository ${repoId} not found or not owned by user`,
324
+ });
325
+ }
326
+ }
327
+ // Assign repositories to the group
328
+ for (const repoId of repositoryIds) {
329
+ await db.repositories.assignToGroup(repoId, id);
330
+ }
331
+ // Fetch updated group
332
+ const updatedGroup = await db.projectGroups.findWithRepositories(id);
333
+ res.json({
334
+ success: true,
335
+ group: {
336
+ id: updatedGroup.id,
337
+ name: updatedGroup.name,
338
+ repositories: updatedGroup.repositories.map(repo => ({
339
+ id: repo.id,
340
+ githubFullName: repo.githubFullName,
341
+ defaultBranch: repo.defaultBranch,
342
+ isPrivate: repo.isPrivate,
343
+ })),
344
+ },
345
+ });
346
+ }
347
+ catch (error) {
348
+ console.error('Error adding repositories to group:', error);
349
+ res.status(500).json({ error: 'Failed to add repositories to group' });
350
+ }
351
+ });
352
+ /**
353
+ * DELETE /api/project-groups/:id/repositories/:repoId
354
+ * Remove a repository from a project group
355
+ */
356
+ coordinatorsRouter.delete('/:id/repositories/:repoId', async (req, res) => {
357
+ const userId = req.session.userId;
358
+ const id = req.params.id;
359
+ const repoId = req.params.repoId;
360
+ try {
361
+ const group = await db.projectGroups.findById(id);
362
+ if (!group) {
363
+ return res.status(404).json({ error: 'Project group not found' });
364
+ }
365
+ if (group.userId !== userId) {
366
+ return res.status(403).json({ error: 'Unauthorized' });
367
+ }
368
+ // Verify repository exists and belongs to this group
369
+ const repo = await db.repositories.findById(repoId);
370
+ if (!repo) {
371
+ return res.status(404).json({ error: 'Repository not found' });
372
+ }
373
+ if (repo.userId !== userId) {
374
+ return res.status(403).json({ error: 'Unauthorized' });
375
+ }
376
+ if (repo.projectGroupId !== id) {
377
+ return res.status(400).json({ error: 'Repository is not in this group' });
378
+ }
379
+ // Remove repository from group (set projectGroupId to null)
380
+ await db.repositories.assignToGroup(repoId, null);
381
+ res.json({
382
+ success: true,
383
+ message: 'Repository removed from group',
384
+ });
385
+ }
386
+ catch (error) {
387
+ console.error('Error removing repository from group:', error);
388
+ res.status(500).json({ error: 'Failed to remove repository from group' });
389
+ }
390
+ });
391
+ /**
392
+ * PUT /api/project-groups/reorder
393
+ * Reorder project groups
394
+ */
395
+ coordinatorsRouter.put('/reorder', async (req, res) => {
396
+ const userId = req.session.userId;
397
+ const { orderedIds } = req.body;
398
+ if (!orderedIds || !Array.isArray(orderedIds)) {
399
+ return res.status(400).json({ error: 'orderedIds array is required' });
400
+ }
401
+ try {
402
+ // Verify all groups belong to user
403
+ const userGroups = await db.projectGroups.findByUserId(userId);
404
+ const userGroupIds = new Set(userGroups.map(g => g.id));
405
+ for (const groupId of orderedIds) {
406
+ if (!userGroupIds.has(groupId)) {
407
+ return res.status(400).json({
408
+ error: `Group ${groupId} not found or not owned by user`,
409
+ });
410
+ }
411
+ }
412
+ await db.projectGroups.reorder(userId, orderedIds);
413
+ res.json({
414
+ success: true,
415
+ message: 'Groups reordered',
416
+ });
417
+ }
418
+ catch (error) {
419
+ console.error('Error reordering project groups:', error);
420
+ res.status(500).json({ error: 'Failed to reorder project groups' });
421
+ }
422
+ });
423
+ // ============================================================================
424
+ // Coordinator Agent Routes
425
+ // ============================================================================
426
+ /**
427
+ * GET /api/project-groups/:groupId/coordinator
428
+ * Get coordinator agent configuration
429
+ */
430
+ coordinatorsRouter.get('/:groupId/coordinator', async (req, res) => {
431
+ const userId = req.session.userId;
432
+ const groupId = req.params.groupId;
433
+ try {
434
+ const group = await db.projectGroups.findById(groupId);
435
+ if (!group) {
436
+ return res.status(404).json({ error: 'Project group not found' });
437
+ }
438
+ if (group.userId !== userId) {
439
+ return res.status(403).json({ error: 'Unauthorized' });
440
+ }
441
+ res.json({
442
+ groupId: group.id,
443
+ groupName: group.name,
444
+ coordinator: group.coordinatorAgent || { enabled: false },
445
+ });
446
+ }
447
+ catch (error) {
448
+ console.error('Error getting coordinator config:', error);
449
+ res.status(500).json({ error: 'Failed to get coordinator configuration' });
450
+ }
451
+ });
452
+ /**
453
+ * PUT /api/project-groups/:groupId/coordinator
454
+ * Update coordinator agent configuration
455
+ */
456
+ coordinatorsRouter.put('/:groupId/coordinator', async (req, res) => {
457
+ const userId = req.session.userId;
458
+ const groupId = req.params.groupId;
459
+ const { name, model, systemPrompt, capabilities } = req.body;
460
+ try {
461
+ const group = await db.projectGroups.findById(groupId);
462
+ if (!group) {
463
+ return res.status(404).json({ error: 'Project group not found' });
464
+ }
465
+ if (group.userId !== userId) {
466
+ return res.status(403).json({ error: 'Unauthorized' });
467
+ }
468
+ // Build updated config, preserving enabled state
469
+ const currentConfig = group.coordinatorAgent || { enabled: false };
470
+ const updatedConfig = {
471
+ enabled: currentConfig.enabled,
472
+ name: name !== undefined ? name : currentConfig.name,
473
+ model: model !== undefined ? model : currentConfig.model,
474
+ systemPrompt: systemPrompt !== undefined ? systemPrompt : currentConfig.systemPrompt,
475
+ capabilities: capabilities !== undefined ? capabilities : currentConfig.capabilities,
476
+ };
477
+ await db.projectGroups.updateCoordinatorAgent(groupId, updatedConfig);
478
+ // If coordinator is currently enabled, restart it with new config
479
+ if (updatedConfig.enabled) {
480
+ const coordinatorService = getCoordinatorService();
481
+ await coordinatorService.restart(groupId);
482
+ }
483
+ res.json({
484
+ success: true,
485
+ coordinator: updatedConfig,
486
+ });
487
+ }
488
+ catch (error) {
489
+ console.error('Error updating coordinator config:', error);
490
+ res.status(500).json({ error: 'Failed to update coordinator configuration' });
491
+ }
492
+ });
493
+ /**
494
+ * POST /api/project-groups/:groupId/coordinator/enable
495
+ * Enable coordinator agent for a project group
496
+ */
497
+ coordinatorsRouter.post('/:groupId/coordinator/enable', async (req, res) => {
498
+ const userId = req.session.userId;
499
+ const groupId = req.params.groupId;
500
+ try {
501
+ const group = await db.projectGroups.findById(groupId);
502
+ if (!group) {
503
+ return res.status(404).json({ error: 'Project group not found' });
504
+ }
505
+ if (group.userId !== userId) {
506
+ return res.status(403).json({ error: 'Unauthorized' });
507
+ }
508
+ // Plan check is handled by checkCoordinatorAccess middleware
509
+ // Get repositories in the group
510
+ const repositories = await db.repositories.findByProjectGroupId(groupId);
511
+ if (repositories.length === 0) {
512
+ return res.status(400).json({
513
+ error: 'Cannot enable coordinator for empty group',
514
+ message: 'Add at least one repository to this group first',
515
+ });
516
+ }
517
+ // Enable coordinator
518
+ const currentConfig = group.coordinatorAgent || { enabled: false };
519
+ const updatedConfig = {
520
+ ...currentConfig,
521
+ enabled: true,
522
+ name: currentConfig.name || `${group.name} Coordinator`,
523
+ };
524
+ await db.projectGroups.updateCoordinatorAgent(groupId, updatedConfig);
525
+ // Start the coordinator agent
526
+ const coordinatorService = getCoordinatorService();
527
+ await coordinatorService.start(groupId);
528
+ res.json({
529
+ success: true,
530
+ message: 'Coordinator agent enabled',
531
+ coordinator: updatedConfig,
532
+ });
533
+ }
534
+ catch (error) {
535
+ console.error('Error enabling coordinator:', error);
536
+ res.status(500).json({ error: 'Failed to enable coordinator agent' });
537
+ }
538
+ });
539
+ /**
540
+ * POST /api/project-groups/:groupId/coordinator/disable
541
+ * Disable coordinator agent for a project group
542
+ */
543
+ coordinatorsRouter.post('/:groupId/coordinator/disable', async (req, res) => {
544
+ const userId = req.session.userId;
545
+ const groupId = req.params.groupId;
546
+ try {
547
+ const group = await db.projectGroups.findById(groupId);
548
+ if (!group) {
549
+ return res.status(404).json({ error: 'Project group not found' });
550
+ }
551
+ if (group.userId !== userId) {
552
+ return res.status(403).json({ error: 'Unauthorized' });
553
+ }
554
+ // Disable coordinator
555
+ const currentConfig = group.coordinatorAgent || { enabled: false };
556
+ const updatedConfig = {
557
+ ...currentConfig,
558
+ enabled: false,
559
+ };
560
+ await db.projectGroups.updateCoordinatorAgent(groupId, updatedConfig);
561
+ // Stop the coordinator agent
562
+ const coordinatorService = getCoordinatorService();
563
+ await coordinatorService.stop(groupId);
564
+ res.json({
565
+ success: true,
566
+ message: 'Coordinator agent disabled',
567
+ coordinator: updatedConfig,
568
+ });
569
+ }
570
+ catch (error) {
571
+ console.error('Error disabling coordinator:', error);
572
+ res.status(500).json({ error: 'Failed to disable coordinator agent' });
573
+ }
574
+ });
575
+ /**
576
+ * GET /api/project-groups/coordinators/active
577
+ * List all active coordinators for the user
578
+ */
579
+ coordinatorsRouter.get('/coordinators/active', async (req, res) => {
580
+ const userId = req.session.userId;
581
+ try {
582
+ // Get all coordinators
583
+ const activeCoordinators = await getActiveCoordinators();
584
+ // Filter to only user's project groups
585
+ const userGroups = await db.projectGroups.findByUserId(userId);
586
+ const userGroupIds = new Set(userGroups.map((g) => g.id));
587
+ const userCoordinators = activeCoordinators.filter((c) => userGroupIds.has(c.groupId));
588
+ res.json({
589
+ coordinators: userCoordinators,
590
+ });
591
+ }
592
+ catch (error) {
593
+ console.error('Error listing active coordinators:', error);
594
+ res.status(500).json({ error: 'Failed to list coordinators' });
595
+ }
596
+ });
597
+ /**
598
+ * POST /api/project-groups/:groupId/coordinator/message
599
+ * Send a message from coordinator to a specific workspace/agent
600
+ */
601
+ coordinatorsRouter.post('/:groupId/coordinator/message', async (req, res) => {
602
+ const userId = req.session.userId;
603
+ const groupId = req.params.groupId;
604
+ const { workspaceId, agentName, message, thread } = req.body;
605
+ if (!workspaceId || !agentName || !message) {
606
+ return res.status(400).json({
607
+ error: 'workspaceId, agentName, and message are required',
608
+ });
609
+ }
610
+ try {
611
+ const group = await db.projectGroups.findById(groupId);
612
+ if (!group) {
613
+ return res.status(404).json({ error: 'Project group not found' });
614
+ }
615
+ if (group.userId !== userId) {
616
+ return res.status(403).json({ error: 'Unauthorized' });
617
+ }
618
+ if (!group.coordinatorAgent?.enabled) {
619
+ return res.status(400).json({ error: 'Coordinator is not enabled' });
620
+ }
621
+ await sendToWorkspace(groupId, workspaceId, agentName, message, thread);
622
+ res.json({
623
+ success: true,
624
+ message: 'Message sent',
625
+ });
626
+ }
627
+ catch (error) {
628
+ console.error('Error sending coordinator message:', error);
629
+ res.status(500).json({ error: 'Failed to send message' });
630
+ }
631
+ });
632
+ /**
633
+ * POST /api/project-groups/:groupId/coordinator/broadcast
634
+ * Broadcast a message from coordinator to all workspaces in the group
635
+ */
636
+ coordinatorsRouter.post('/:groupId/coordinator/broadcast', async (req, res) => {
637
+ const userId = req.session.userId;
638
+ const groupId = req.params.groupId;
639
+ const { message, thread } = req.body;
640
+ if (!message) {
641
+ return res.status(400).json({ error: 'message is required' });
642
+ }
643
+ try {
644
+ const group = await db.projectGroups.findById(groupId);
645
+ if (!group) {
646
+ return res.status(404).json({ error: 'Project group not found' });
647
+ }
648
+ if (group.userId !== userId) {
649
+ return res.status(403).json({ error: 'Unauthorized' });
650
+ }
651
+ if (!group.coordinatorAgent?.enabled) {
652
+ return res.status(400).json({ error: 'Coordinator is not enabled' });
653
+ }
654
+ await broadcastToGroup(groupId, message, thread);
655
+ res.json({
656
+ success: true,
657
+ message: 'Broadcast sent',
658
+ });
659
+ }
660
+ catch (error) {
661
+ console.error('Error broadcasting coordinator message:', error);
662
+ res.status(500).json({ error: 'Failed to broadcast message' });
663
+ }
664
+ });
665
+ /**
666
+ * POST /api/project-groups/coordinator/route
667
+ * Route a message from a workspace to its coordinator
668
+ * (Called by workspace daemons)
669
+ */
670
+ coordinatorsRouter.post('/coordinator/route', async (req, res) => {
671
+ const { workspaceId, agentName, message, thread } = req.body;
672
+ if (!workspaceId || !agentName || !message) {
673
+ return res.status(400).json({
674
+ error: 'workspaceId, agentName, and message are required',
675
+ });
676
+ }
677
+ try {
678
+ // Verify workspace exists and get its owner
679
+ const workspace = await db.workspaces.findById(workspaceId);
680
+ if (!workspace) {
681
+ return res.status(404).json({ error: 'Workspace not found' });
682
+ }
683
+ await routeToCoordinator(workspaceId, agentName, message, thread);
684
+ res.json({
685
+ success: true,
686
+ message: 'Message routed to coordinator',
687
+ });
688
+ }
689
+ catch (error) {
690
+ console.error('Error routing to coordinator:', error);
691
+ res.status(500).json({ error: 'Failed to route message' });
692
+ }
693
+ });
694
+ /**
695
+ * GET /api/project-groups/:groupId/coordinator/status
696
+ * Get detailed coordinator status including connected workspaces
697
+ */
698
+ coordinatorsRouter.get('/:groupId/coordinator/status', async (req, res) => {
699
+ const userId = req.session.userId;
700
+ const groupId = req.params.groupId;
701
+ try {
702
+ const group = await db.projectGroups.findById(groupId);
703
+ if (!group) {
704
+ return res.status(404).json({ error: 'Project group not found' });
705
+ }
706
+ if (group.userId !== userId) {
707
+ return res.status(403).json({ error: 'Unauthorized' });
708
+ }
709
+ const coordinatorService = getCoordinatorService();
710
+ const status = await coordinatorService.getStatus(groupId);
711
+ // Get connected workspaces info
712
+ const repositories = await db.repositories.findByProjectGroupId(groupId);
713
+ const workspaceIds = new Set();
714
+ for (const repo of repositories) {
715
+ if (repo.workspaceId) {
716
+ workspaceIds.add(repo.workspaceId);
717
+ }
718
+ }
719
+ const workspaces = await Promise.all(Array.from(workspaceIds).map(async (id) => {
720
+ const ws = await db.workspaces.findById(id);
721
+ return ws
722
+ ? {
723
+ id: ws.id,
724
+ name: ws.name,
725
+ status: ws.status,
726
+ publicUrl: ws.publicUrl,
727
+ }
728
+ : null;
729
+ }));
730
+ res.json({
731
+ groupId,
732
+ groupName: group.name,
733
+ coordinator: {
734
+ enabled: group.coordinatorAgent?.enabled || false,
735
+ name: group.coordinatorAgent?.name,
736
+ model: group.coordinatorAgent?.model,
737
+ status: status?.status || 'stopped',
738
+ startedAt: status?.startedAt,
739
+ error: status?.error,
740
+ },
741
+ workspaces: workspaces.filter(Boolean),
742
+ repositoryCount: repositories.length,
743
+ });
744
+ }
745
+ catch (error) {
746
+ console.error('Error getting coordinator status:', error);
747
+ res.status(500).json({ error: 'Failed to get coordinator status' });
748
+ }
749
+ });
750
+ //# sourceMappingURL=coordinators.js.map