agent-relay 2.0.23 → 2.0.25

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 (168) hide show
  1. package/dist/src/cli/index.js +160 -17
  2. package/package.json +18 -52
  3. package/packages/api-types/package.json +1 -1
  4. package/packages/bridge/package.json +8 -8
  5. package/packages/cli-tester/package.json +1 -1
  6. package/packages/config/package.json +2 -2
  7. package/packages/continuity/package.json +1 -1
  8. package/packages/daemon/package.json +12 -12
  9. package/packages/hooks/package.json +4 -4
  10. package/packages/mcp/package.json +2 -2
  11. package/packages/memory/package.json +2 -2
  12. package/packages/policy/package.json +2 -2
  13. package/packages/protocol/package.json +1 -1
  14. package/packages/resiliency/package.json +1 -1
  15. package/packages/sdk/package.json +2 -2
  16. package/packages/spawner/package.json +1 -1
  17. package/packages/state/package.json +1 -1
  18. package/packages/storage/package.json +2 -2
  19. package/packages/telemetry/package.json +1 -1
  20. package/packages/trajectory/package.json +2 -2
  21. package/packages/user-directory/package.json +2 -2
  22. package/packages/utils/package.json +1 -1
  23. package/packages/wrapper/package.json +6 -6
  24. package/deploy/init-db.sql +0 -5
  25. package/deploy/scripts/setup-fly-workspaces.sh +0 -69
  26. package/deploy/scripts/setup-railway.sh +0 -75
  27. package/dist/src/cloud/index.d.ts +0 -8
  28. package/dist/src/cloud/index.js +0 -8
  29. package/packages/cloud/dist/api/admin.d.ts +0 -8
  30. package/packages/cloud/dist/api/admin.js +0 -225
  31. package/packages/cloud/dist/api/auth.d.ts +0 -20
  32. package/packages/cloud/dist/api/auth.js +0 -138
  33. package/packages/cloud/dist/api/billing.d.ts +0 -7
  34. package/packages/cloud/dist/api/billing.js +0 -564
  35. package/packages/cloud/dist/api/cli-pty-runner.d.ts +0 -53
  36. package/packages/cloud/dist/api/cli-pty-runner.js +0 -175
  37. package/packages/cloud/dist/api/codex-auth-helper.d.ts +0 -21
  38. package/packages/cloud/dist/api/codex-auth-helper.js +0 -327
  39. package/packages/cloud/dist/api/consensus.d.ts +0 -13
  40. package/packages/cloud/dist/api/consensus.js +0 -261
  41. package/packages/cloud/dist/api/coordinators.d.ts +0 -8
  42. package/packages/cloud/dist/api/coordinators.js +0 -750
  43. package/packages/cloud/dist/api/daemons.d.ts +0 -12
  44. package/packages/cloud/dist/api/daemons.js +0 -535
  45. package/packages/cloud/dist/api/email-auth.d.ts +0 -11
  46. package/packages/cloud/dist/api/email-auth.js +0 -347
  47. package/packages/cloud/dist/api/generic-webhooks.d.ts +0 -8
  48. package/packages/cloud/dist/api/generic-webhooks.js +0 -129
  49. package/packages/cloud/dist/api/git.d.ts +0 -8
  50. package/packages/cloud/dist/api/git.js +0 -269
  51. package/packages/cloud/dist/api/github-app.d.ts +0 -11
  52. package/packages/cloud/dist/api/github-app.js +0 -223
  53. package/packages/cloud/dist/api/middleware/planLimits.d.ts +0 -43
  54. package/packages/cloud/dist/api/middleware/planLimits.js +0 -202
  55. package/packages/cloud/dist/api/monitoring.d.ts +0 -11
  56. package/packages/cloud/dist/api/monitoring.js +0 -578
  57. package/packages/cloud/dist/api/nango-auth.d.ts +0 -9
  58. package/packages/cloud/dist/api/nango-auth.js +0 -741
  59. package/packages/cloud/dist/api/onboarding.d.ts +0 -15
  60. package/packages/cloud/dist/api/onboarding.js +0 -679
  61. package/packages/cloud/dist/api/policy.d.ts +0 -8
  62. package/packages/cloud/dist/api/policy.js +0 -229
  63. package/packages/cloud/dist/api/provider-env.d.ts +0 -26
  64. package/packages/cloud/dist/api/provider-env.js +0 -141
  65. package/packages/cloud/dist/api/providers.d.ts +0 -7
  66. package/packages/cloud/dist/api/providers.js +0 -574
  67. package/packages/cloud/dist/api/repos.d.ts +0 -8
  68. package/packages/cloud/dist/api/repos.js +0 -577
  69. package/packages/cloud/dist/api/sessions.d.ts +0 -11
  70. package/packages/cloud/dist/api/sessions.js +0 -302
  71. package/packages/cloud/dist/api/teams.d.ts +0 -7
  72. package/packages/cloud/dist/api/teams.js +0 -281
  73. package/packages/cloud/dist/api/test-helpers.d.ts +0 -10
  74. package/packages/cloud/dist/api/test-helpers.js +0 -745
  75. package/packages/cloud/dist/api/usage.d.ts +0 -7
  76. package/packages/cloud/dist/api/usage.js +0 -111
  77. package/packages/cloud/dist/api/webhooks.d.ts +0 -8
  78. package/packages/cloud/dist/api/webhooks.js +0 -645
  79. package/packages/cloud/dist/api/workspaces.d.ts +0 -25
  80. package/packages/cloud/dist/api/workspaces.js +0 -1799
  81. package/packages/cloud/dist/billing/index.d.ts +0 -9
  82. package/packages/cloud/dist/billing/index.js +0 -9
  83. package/packages/cloud/dist/billing/plans.d.ts +0 -39
  84. package/packages/cloud/dist/billing/plans.js +0 -245
  85. package/packages/cloud/dist/billing/service.d.ts +0 -80
  86. package/packages/cloud/dist/billing/service.js +0 -388
  87. package/packages/cloud/dist/billing/types.d.ts +0 -141
  88. package/packages/cloud/dist/billing/types.js +0 -7
  89. package/packages/cloud/dist/config.d.ts +0 -5
  90. package/packages/cloud/dist/config.js +0 -5
  91. package/packages/cloud/dist/db/bulk-ingest.d.ts +0 -89
  92. package/packages/cloud/dist/db/bulk-ingest.js +0 -268
  93. package/packages/cloud/dist/db/drizzle.d.ts +0 -290
  94. package/packages/cloud/dist/db/drizzle.js +0 -1422
  95. package/packages/cloud/dist/db/index.d.ts +0 -56
  96. package/packages/cloud/dist/db/index.js +0 -70
  97. package/packages/cloud/dist/db/schema.d.ts +0 -5117
  98. package/packages/cloud/dist/db/schema.js +0 -656
  99. package/packages/cloud/dist/index.d.ts +0 -11
  100. package/packages/cloud/dist/index.js +0 -38
  101. package/packages/cloud/dist/provisioner/index.d.ts +0 -207
  102. package/packages/cloud/dist/provisioner/index.js +0 -2118
  103. package/packages/cloud/dist/server.d.ts +0 -17
  104. package/packages/cloud/dist/server.js +0 -2055
  105. package/packages/cloud/dist/services/auto-scaler.d.ts +0 -152
  106. package/packages/cloud/dist/services/auto-scaler.js +0 -439
  107. package/packages/cloud/dist/services/capacity-manager.d.ts +0 -148
  108. package/packages/cloud/dist/services/capacity-manager.js +0 -449
  109. package/packages/cloud/dist/services/ci-agent-spawner.d.ts +0 -49
  110. package/packages/cloud/dist/services/ci-agent-spawner.js +0 -373
  111. package/packages/cloud/dist/services/cloud-message-bus.d.ts +0 -28
  112. package/packages/cloud/dist/services/cloud-message-bus.js +0 -19
  113. package/packages/cloud/dist/services/compute-enforcement.d.ts +0 -57
  114. package/packages/cloud/dist/services/compute-enforcement.js +0 -175
  115. package/packages/cloud/dist/services/coordinator.d.ts +0 -62
  116. package/packages/cloud/dist/services/coordinator.js +0 -389
  117. package/packages/cloud/dist/services/index.d.ts +0 -17
  118. package/packages/cloud/dist/services/index.js +0 -25
  119. package/packages/cloud/dist/services/intro-expiration.d.ts +0 -60
  120. package/packages/cloud/dist/services/intro-expiration.js +0 -252
  121. package/packages/cloud/dist/services/mention-handler.d.ts +0 -65
  122. package/packages/cloud/dist/services/mention-handler.js +0 -405
  123. package/packages/cloud/dist/services/nango.d.ts +0 -219
  124. package/packages/cloud/dist/services/nango.js +0 -424
  125. package/packages/cloud/dist/services/persistence.d.ts +0 -131
  126. package/packages/cloud/dist/services/persistence.js +0 -200
  127. package/packages/cloud/dist/services/planLimits.d.ts +0 -147
  128. package/packages/cloud/dist/services/planLimits.js +0 -335
  129. package/packages/cloud/dist/services/presence-registry.d.ts +0 -56
  130. package/packages/cloud/dist/services/presence-registry.js +0 -91
  131. package/packages/cloud/dist/services/scaling-orchestrator.d.ts +0 -159
  132. package/packages/cloud/dist/services/scaling-orchestrator.js +0 -502
  133. package/packages/cloud/dist/services/scaling-policy.d.ts +0 -121
  134. package/packages/cloud/dist/services/scaling-policy.js +0 -415
  135. package/packages/cloud/dist/services/ssh-security.d.ts +0 -31
  136. package/packages/cloud/dist/services/ssh-security.js +0 -63
  137. package/packages/cloud/dist/services/workspace-keepalive.d.ts +0 -76
  138. package/packages/cloud/dist/services/workspace-keepalive.js +0 -234
  139. package/packages/cloud/dist/shims/consensus.d.ts +0 -23
  140. package/packages/cloud/dist/shims/consensus.js +0 -5
  141. package/packages/cloud/dist/webhooks/index.d.ts +0 -24
  142. package/packages/cloud/dist/webhooks/index.js +0 -29
  143. package/packages/cloud/dist/webhooks/parsers/github.d.ts +0 -8
  144. package/packages/cloud/dist/webhooks/parsers/github.js +0 -234
  145. package/packages/cloud/dist/webhooks/parsers/index.d.ts +0 -23
  146. package/packages/cloud/dist/webhooks/parsers/index.js +0 -30
  147. package/packages/cloud/dist/webhooks/parsers/linear.d.ts +0 -9
  148. package/packages/cloud/dist/webhooks/parsers/linear.js +0 -258
  149. package/packages/cloud/dist/webhooks/parsers/slack.d.ts +0 -9
  150. package/packages/cloud/dist/webhooks/parsers/slack.js +0 -214
  151. package/packages/cloud/dist/webhooks/responders/github.d.ts +0 -8
  152. package/packages/cloud/dist/webhooks/responders/github.js +0 -73
  153. package/packages/cloud/dist/webhooks/responders/index.d.ts +0 -23
  154. package/packages/cloud/dist/webhooks/responders/index.js +0 -30
  155. package/packages/cloud/dist/webhooks/responders/linear.d.ts +0 -9
  156. package/packages/cloud/dist/webhooks/responders/linear.js +0 -149
  157. package/packages/cloud/dist/webhooks/responders/slack.d.ts +0 -20
  158. package/packages/cloud/dist/webhooks/responders/slack.js +0 -178
  159. package/packages/cloud/dist/webhooks/router.d.ts +0 -25
  160. package/packages/cloud/dist/webhooks/router.js +0 -504
  161. package/packages/cloud/dist/webhooks/rules-engine.d.ts +0 -24
  162. package/packages/cloud/dist/webhooks/rules-engine.js +0 -287
  163. package/packages/cloud/dist/webhooks/types.d.ts +0 -186
  164. package/packages/cloud/dist/webhooks/types.js +0 -8
  165. package/packages/cloud/package.json +0 -60
  166. package/scripts/run-migrations.js +0 -43
  167. package/scripts/setup-stripe-products.ts +0 -312
  168. package/scripts/verify-schema.js +0 -134
@@ -1,269 +0,0 @@
1
- /**
2
- * Git Gateway API Routes
3
- *
4
- * Provides fresh GitHub tokens to workspace containers for git operations.
5
- * This gateway pattern ensures tokens are always valid (Nango handles refresh).
6
- */
7
- import crypto from 'crypto';
8
- import { Router } from 'express';
9
- import { db } from '../db/index.js';
10
- import { nangoService } from '../services/nango.js';
11
- import { getConfig } from '../config.js';
12
- export const gitRouter = Router();
13
- /**
14
- * Generate expected workspace token using HMAC
15
- */
16
- function generateExpectedToken(workspaceId) {
17
- const config = getConfig();
18
- return crypto
19
- .createHmac('sha256', config.sessionSecret)
20
- .update(`workspace:${workspaceId}`)
21
- .digest('hex');
22
- }
23
- /**
24
- * Verify workspace access token
25
- * Workspaces authenticate with a secret passed at provisioning time
26
- *
27
- * Returns:
28
- * - { valid: true } if token matches
29
- * - { valid: false, reason: string } if token is invalid or missing
30
- */
31
- function verifyWorkspaceToken(req, workspaceId) {
32
- const authHeader = req.get('authorization');
33
- if (!authHeader) {
34
- return { valid: false, reason: 'No Authorization header. WORKSPACE_TOKEN may not be set in the container.' };
35
- }
36
- if (!authHeader.startsWith('Bearer ')) {
37
- return { valid: false, reason: 'Invalid Authorization header format. Expected: Bearer <token>' };
38
- }
39
- const providedToken = authHeader.slice(7);
40
- if (!providedToken) {
41
- return { valid: false, reason: 'Empty bearer token provided.' };
42
- }
43
- const expectedToken = generateExpectedToken(workspaceId);
44
- // Use timing-safe comparison to prevent timing attacks
45
- try {
46
- const isValid = crypto.timingSafeEqual(Buffer.from(providedToken), Buffer.from(expectedToken));
47
- if (!isValid) {
48
- return { valid: false, reason: 'Token mismatch. Workspace may need reprovisioning or SESSION_SECRET changed.' };
49
- }
50
- return { valid: true };
51
- }
52
- catch {
53
- return { valid: false, reason: 'Token comparison failed (length mismatch). Workspace may need reprovisioning.' };
54
- }
55
- }
56
- /**
57
- * GET /api/git/token
58
- * Get a fresh GitHub token for git operations
59
- *
60
- * Query params:
61
- * - workspaceId: The workspace requesting the token
62
- *
63
- * Returns: { token: string, expiresAt?: string }
64
- *
65
- * This endpoint is called by the git credential helper in workspace containers.
66
- * It fetches a fresh GitHub App installation token via Nango.
67
- */
68
- gitRouter.get('/token', async (req, res) => {
69
- const { workspaceId } = req.query;
70
- if (!workspaceId || typeof workspaceId !== 'string') {
71
- return res.status(400).json({ error: 'workspaceId is required' });
72
- }
73
- // Verify the request is from a valid workspace
74
- const tokenVerification = verifyWorkspaceToken(req, workspaceId);
75
- if (!tokenVerification.valid) {
76
- console.warn(`[git] Token verification failed for workspace ${workspaceId.substring(0, 8)}: ${tokenVerification.reason}`);
77
- return res.status(401).json({
78
- error: 'Invalid workspace token',
79
- code: 'INVALID_WORKSPACE_TOKEN',
80
- hint: tokenVerification.reason,
81
- });
82
- }
83
- try {
84
- // Get workspace to find the user
85
- const workspace = await db.workspaces.findById(workspaceId);
86
- if (!workspace) {
87
- console.warn(`[git] Workspace not found: ${workspaceId}`);
88
- return res.status(404).json({
89
- error: 'Workspace not found',
90
- code: 'WORKSPACE_NOT_FOUND',
91
- hint: 'The workspace may have been deleted. Try reprovisioning.',
92
- });
93
- }
94
- const userId = workspace.userId;
95
- console.log(`[git] Token request for workspace ${workspaceId.substring(0, 8)}, user ${userId.substring(0, 8)}`);
96
- // Find a repository with a Nango connection for this user
97
- const repos = await db.repositories.findByUserId(userId);
98
- const repoWithConnection = repos.find(r => r.nangoConnectionId);
99
- if (!repoWithConnection?.nangoConnectionId) {
100
- console.warn(`[git] No Nango connection found for user ${userId.substring(0, 8)}. Repos: ${repos.length}, with connections: ${repos.filter(r => r.nangoConnectionId).length}`);
101
- return res.status(404).json({
102
- error: 'No GitHub App connection found',
103
- code: 'NO_GITHUB_APP_CONNECTION',
104
- hint: 'Install the GitHub App on your repositories at https://github.com/apps/agent-relay',
105
- repoCount: repos.length,
106
- });
107
- }
108
- console.log(`[git] Fetching token from Nango for connection ${repoWithConnection.nangoConnectionId.substring(0, 8)}...`);
109
- // Get fresh tokens from Nango (auto-refreshes if needed)
110
- // - installationToken: for git operations (clone, push, pull)
111
- // - userToken: for gh CLI operations (requires user context)
112
- let installationToken;
113
- try {
114
- installationToken = await nangoService.getGithubAppToken(repoWithConnection.nangoConnectionId);
115
- }
116
- catch (nangoError) {
117
- const errorMessage = nangoError instanceof Error ? nangoError.message : 'Unknown error';
118
- console.error(`[git] Nango token fetch failed for connection ${repoWithConnection.nangoConnectionId}:`, errorMessage);
119
- // Provide specific hints based on error type
120
- if (errorMessage.includes('not found') || errorMessage.includes('404')) {
121
- return res.status(500).json({
122
- error: 'GitHub App connection expired or revoked',
123
- code: 'NANGO_CONNECTION_EXPIRED',
124
- hint: 'Reconnect your GitHub App at https://github.com/apps/agent-relay',
125
- details: errorMessage,
126
- });
127
- }
128
- return res.status(500).json({
129
- error: 'Failed to fetch GitHub token from Nango',
130
- code: 'NANGO_TOKEN_FETCH_FAILED',
131
- hint: 'This may be a temporary issue. Try again in a few seconds.',
132
- details: errorMessage,
133
- });
134
- }
135
- // Try to get user OAuth token from github-app-oauth connection_config first
136
- // Fall back to separate 'github' user connection if available
137
- let userToken = null;
138
- try {
139
- userToken = await nangoService.getGithubUserOAuthToken(repoWithConnection.nangoConnectionId);
140
- }
141
- catch {
142
- // Try the separate github user connection if available
143
- const userRepo = repos.find(r => r.nangoConnectionId && r.nangoConnectionId !== repoWithConnection.nangoConnectionId);
144
- if (userRepo?.nangoConnectionId) {
145
- try {
146
- userToken = await nangoService.getGithubUserToken(userRepo.nangoConnectionId);
147
- }
148
- catch {
149
- console.log('[git] No github user token available');
150
- }
151
- }
152
- }
153
- // GitHub App installation tokens expire after 1 hour
154
- const expiresAt = new Date(Date.now() + 55 * 60 * 1000).toISOString(); // 55 min buffer
155
- // Prefer userToken for git operations - installation tokens (ghs_*) are API-only
156
- // and don't work with git credential helpers. User OAuth tokens work for both
157
- // git operations (clone, push, pull) AND gh CLI commands.
158
- const primaryToken = userToken || installationToken;
159
- const tokenType = userToken ? 'user' : 'installation';
160
- console.log(`[git] Token fetched successfully for workspace ${workspaceId.substring(0, 8)} (type: ${tokenType})`);
161
- res.json({
162
- token: primaryToken, // Primary token for git/gh operations (prefer user token)
163
- userToken, // Explicit user token field (may be null)
164
- installationToken, // GitHub App installation token for API operations
165
- expiresAt,
166
- username: 'x-access-token', // Works with both token types
167
- tokenType, // 'user' or 'installation' - helps clients know what they got
168
- });
169
- }
170
- catch (error) {
171
- const errorMessage = error instanceof Error ? error.message : 'Unknown error';
172
- console.error('[git] Unexpected error getting token:', error);
173
- res.status(500).json({
174
- error: 'Failed to get GitHub token',
175
- code: 'UNEXPECTED_ERROR',
176
- details: errorMessage,
177
- });
178
- }
179
- });
180
- /**
181
- * POST /api/git/token
182
- * Same as GET but accepts body params (for compatibility with some git credential helpers)
183
- */
184
- gitRouter.post('/token', async (req, res) => {
185
- const workspaceId = req.body.workspaceId || req.query.workspaceId;
186
- if (!workspaceId || typeof workspaceId !== 'string') {
187
- return res.status(400).json({ error: 'workspaceId is required' });
188
- }
189
- const tokenVerification = verifyWorkspaceToken(req, workspaceId);
190
- if (!tokenVerification.valid) {
191
- console.warn(`[git] POST: Token verification failed for workspace ${workspaceId.substring(0, 8)}: ${tokenVerification.reason}`);
192
- return res.status(401).json({
193
- error: 'Invalid workspace token',
194
- code: 'INVALID_WORKSPACE_TOKEN',
195
- hint: tokenVerification.reason,
196
- });
197
- }
198
- try {
199
- const workspace = await db.workspaces.findById(workspaceId);
200
- if (!workspace) {
201
- console.warn(`[git] POST: Workspace not found: ${workspaceId}`);
202
- return res.status(404).json({
203
- error: 'Workspace not found',
204
- code: 'WORKSPACE_NOT_FOUND',
205
- });
206
- }
207
- const repos = await db.repositories.findByUserId(workspace.userId);
208
- const repoWithConnection = repos.find(r => r.nangoConnectionId);
209
- if (!repoWithConnection?.nangoConnectionId) {
210
- console.warn(`[git] POST: No Nango connection for user ${workspace.userId.substring(0, 8)}`);
211
- return res.status(404).json({
212
- error: 'No GitHub App connection found',
213
- code: 'NO_GITHUB_APP_CONNECTION',
214
- });
215
- }
216
- let installationToken;
217
- try {
218
- installationToken = await nangoService.getGithubAppToken(repoWithConnection.nangoConnectionId);
219
- }
220
- catch (nangoError) {
221
- const errorMessage = nangoError instanceof Error ? nangoError.message : 'Unknown error';
222
- console.error(`[git] POST: Nango token fetch failed:`, errorMessage);
223
- return res.status(500).json({
224
- error: 'Failed to fetch GitHub token',
225
- code: 'NANGO_TOKEN_FETCH_FAILED',
226
- details: errorMessage,
227
- });
228
- }
229
- // Try to get user OAuth token (preferred for git operations)
230
- let userToken = null;
231
- try {
232
- userToken = await nangoService.getGithubUserOAuthToken(repoWithConnection.nangoConnectionId);
233
- }
234
- catch {
235
- // Try the separate github user connection if available
236
- const userRepo = repos.find(r => r.nangoConnectionId && r.nangoConnectionId !== repoWithConnection.nangoConnectionId);
237
- if (userRepo?.nangoConnectionId) {
238
- try {
239
- userToken = await nangoService.getGithubUserToken(userRepo.nangoConnectionId);
240
- }
241
- catch {
242
- console.log('[git] POST: No github user token available');
243
- }
244
- }
245
- }
246
- const expiresAt = new Date(Date.now() + 55 * 60 * 1000).toISOString();
247
- // Prefer userToken for git operations
248
- const primaryToken = userToken || installationToken;
249
- const tokenType = userToken ? 'user' : 'installation';
250
- res.json({
251
- token: primaryToken,
252
- userToken,
253
- installationToken,
254
- expiresAt,
255
- username: 'x-access-token',
256
- tokenType,
257
- });
258
- }
259
- catch (error) {
260
- const errorMessage = error instanceof Error ? error.message : 'Unknown error';
261
- console.error('[git] POST: Unexpected error:', error);
262
- res.status(500).json({
263
- error: 'Failed to get GitHub token',
264
- code: 'UNEXPECTED_ERROR',
265
- details: errorMessage,
266
- });
267
- }
268
- });
269
- //# sourceMappingURL=git.js.map
@@ -1,11 +0,0 @@
1
- /**
2
- * GitHub App API Routes
3
- *
4
- * Repo operations via Nango's github-app-oauth connection:
5
- * - Get clone token for repositories
6
- * - Create issues, PRs, and comments
7
- *
8
- * Auth flow is handled by nango-auth.ts
9
- */
10
- export declare const githubAppRouter: import("express-serve-static-core").Router;
11
- //# sourceMappingURL=github-app.d.ts.map
@@ -1,223 +0,0 @@
1
- /**
2
- * GitHub App API Routes
3
- *
4
- * Repo operations via Nango's github-app-oauth connection:
5
- * - Get clone token for repositories
6
- * - Create issues, PRs, and comments
7
- *
8
- * Auth flow is handled by nango-auth.ts
9
- */
10
- import { Router } from 'express';
11
- import { requireAuth } from './auth.js';
12
- import { db } from '../db/index.js';
13
- import { nangoService, NANGO_INTEGRATIONS } from '../services/nango.js';
14
- export const githubAppRouter = Router();
15
- // All routes require authentication
16
- githubAppRouter.use(requireAuth);
17
- /**
18
- * GET /api/github-app/status
19
- * Check if Nango GitHub App OAuth is configured
20
- */
21
- githubAppRouter.get('/status', (_req, res) => {
22
- res.json({
23
- configured: true,
24
- integrations: NANGO_INTEGRATIONS,
25
- connectUrl: '/connect-repos',
26
- });
27
- });
28
- /**
29
- * GET /api/github-app/repos
30
- * List repositories the user has access to
31
- *
32
- * First tries database (populated by GitHub App OAuth).
33
- * If empty, queries GitHub directly via user OAuth connection.
34
- */
35
- githubAppRouter.get('/repos', async (req, res) => {
36
- const userId = req.session.userId;
37
- try {
38
- // Try database first (from GitHub App OAuth flow)
39
- const dbRepos = await db.repositories.findByUserId(userId);
40
- if (dbRepos.length > 0) {
41
- // Return repos from database
42
- return res.json({
43
- repositories: dbRepos.map((r) => ({
44
- id: r.id,
45
- fullName: r.githubFullName,
46
- isPrivate: r.isPrivate,
47
- defaultBranch: r.defaultBranch,
48
- syncStatus: r.syncStatus,
49
- hasNangoConnection: !!r.nangoConnectionId,
50
- lastSyncedAt: r.lastSyncedAt,
51
- })),
52
- source: 'database',
53
- });
54
- }
55
- // Database empty - query GitHub directly via user OAuth
56
- const user = await db.users.findById(userId);
57
- if (!user?.nangoConnectionId) {
58
- return res.json({
59
- repositories: [],
60
- source: 'none',
61
- hint: 'User not connected to GitHub',
62
- });
63
- }
64
- console.log(`[github-app/repos] Database empty, querying GitHub for user ${user.githubUsername}`);
65
- const { repositories } = await nangoService.listUserAccessibleRepos(user.nangoConnectionId, {
66
- perPage: 100,
67
- type: 'all',
68
- });
69
- res.json({
70
- repositories: repositories.map((r) => ({
71
- id: null, // No database ID yet
72
- fullName: r.fullName,
73
- isPrivate: r.isPrivate,
74
- defaultBranch: r.defaultBranch,
75
- syncStatus: 'live', // Queried from GitHub, not cached
76
- hasNangoConnection: true,
77
- lastSyncedAt: null,
78
- })),
79
- source: 'github-api',
80
- });
81
- }
82
- catch (error) {
83
- console.error('Error listing repos:', error);
84
- res.status(500).json({ error: 'Failed to list repositories' });
85
- }
86
- });
87
- /**
88
- * GET /api/github-app/clone-token
89
- * Get a clone token for a repository
90
- * Used by workspace provisioner to clone private repos
91
- */
92
- githubAppRouter.get('/clone-token', async (req, res) => {
93
- const userId = req.session.userId;
94
- const { repo } = req.query;
95
- if (!repo || typeof repo !== 'string') {
96
- return res.status(400).json({ error: 'Repository name is required (owner/repo)' });
97
- }
98
- try {
99
- // Find the repository in our database
100
- const userRepos = await db.repositories.findByUserId(userId);
101
- const repository = userRepos.find((r) => r.githubFullName === repo);
102
- if (!repository) {
103
- return res.status(404).json({ error: 'Repository not found' });
104
- }
105
- if (!repository.nangoConnectionId) {
106
- return res.status(400).json({
107
- error: 'Repository not connected via Nango',
108
- hint: 'Connect your GitHub repos through the Nango flow first',
109
- });
110
- }
111
- // Get token from Nango connection
112
- const token = await nangoService.getGithubAppToken(repository.nangoConnectionId);
113
- const cloneUrl = `https://x-access-token:${token}@github.com/${repo}.git`;
114
- res.json({
115
- token,
116
- cloneUrl,
117
- expiresIn: '1 hour',
118
- });
119
- }
120
- catch (error) {
121
- console.error('Error getting clone token:', error);
122
- res.status(500).json({ error: 'Failed to get clone token' });
123
- }
124
- });
125
- /**
126
- * POST /api/github-app/repos/:id/issues
127
- * Create an issue on a repository
128
- */
129
- githubAppRouter.post('/repos/:id/issues', async (req, res) => {
130
- const userId = req.session.userId;
131
- const id = req.params.id;
132
- const { title, body, labels } = req.body;
133
- if (!title || typeof title !== 'string') {
134
- return res.status(400).json({ error: 'Issue title is required' });
135
- }
136
- try {
137
- // Find the repository
138
- const repository = await db.repositories.findById(id);
139
- if (!repository || repository.userId !== userId) {
140
- return res.status(404).json({ error: 'Repository not found' });
141
- }
142
- if (!repository.nangoConnectionId) {
143
- return res.status(400).json({ error: 'Repository not connected via Nango' });
144
- }
145
- // Create issue via Nango Proxy (handles token injection automatically)
146
- const [owner, repo] = repository.githubFullName.split('/');
147
- const issue = await nangoService.createGithubIssue(repository.nangoConnectionId, owner, repo, { title, body: body || '', labels });
148
- res.json({
149
- number: issue.number,
150
- url: issue.html_url,
151
- });
152
- }
153
- catch (error) {
154
- console.error('Error creating issue:', error);
155
- res.status(500).json({ error: 'Failed to create issue' });
156
- }
157
- });
158
- /**
159
- * POST /api/github-app/repos/:id/pulls
160
- * Create a pull request on a repository
161
- */
162
- githubAppRouter.post('/repos/:id/pulls', async (req, res) => {
163
- const userId = req.session.userId;
164
- const id = req.params.id;
165
- const { title, body, head, base } = req.body;
166
- if (!title || !head || !base) {
167
- return res.status(400).json({ error: 'title, head, and base are required' });
168
- }
169
- try {
170
- // Find the repository
171
- const repository = await db.repositories.findById(id);
172
- if (!repository || repository.userId !== userId) {
173
- return res.status(404).json({ error: 'Repository not found' });
174
- }
175
- if (!repository.nangoConnectionId) {
176
- return res.status(400).json({ error: 'Repository not connected via Nango' });
177
- }
178
- // Create PR via Nango Proxy (handles token injection automatically)
179
- const [owner, repo] = repository.githubFullName.split('/');
180
- const pr = await nangoService.createGithubPullRequest(repository.nangoConnectionId, owner, repo, { title, body: body || '', head, base });
181
- res.json({
182
- number: pr.number,
183
- url: pr.html_url,
184
- });
185
- }
186
- catch (error) {
187
- console.error('Error creating PR:', error);
188
- res.status(500).json({ error: 'Failed to create pull request' });
189
- }
190
- });
191
- /**
192
- * POST /api/github-app/repos/:id/comments
193
- * Add a comment to an issue or PR
194
- */
195
- githubAppRouter.post('/repos/:id/comments', async (req, res) => {
196
- const userId = req.session.userId;
197
- const id = req.params.id;
198
- const { issueNumber, body } = req.body;
199
- if (!issueNumber || !body) {
200
- return res.status(400).json({ error: 'issueNumber and body are required' });
201
- }
202
- try {
203
- const repository = await db.repositories.findById(id);
204
- if (!repository || repository.userId !== userId) {
205
- return res.status(404).json({ error: 'Repository not found' });
206
- }
207
- if (!repository.nangoConnectionId) {
208
- return res.status(400).json({ error: 'Repository not connected via Nango' });
209
- }
210
- // Add comment via Nango Proxy (handles token injection automatically)
211
- const [owner, repo] = repository.githubFullName.split('/');
212
- const comment = await nangoService.addGithubIssueComment(repository.nangoConnectionId, owner, repo, issueNumber, body);
213
- res.json({
214
- id: comment.id,
215
- url: comment.html_url,
216
- });
217
- }
218
- catch (error) {
219
- console.error('Error adding comment:', error);
220
- res.status(500).json({ error: 'Failed to add comment' });
221
- }
222
- });
223
- //# sourceMappingURL=github-app.js.map
@@ -1,43 +0,0 @@
1
- /**
2
- * Plan Limits Middleware
3
- *
4
- * Express middleware to enforce plan-based resource limits.
5
- */
6
- import { Request, Response, NextFunction } from 'express';
7
- /**
8
- * Middleware to check workspace creation limit
9
- *
10
- * Use this middleware on workspace creation endpoints.
11
- * Requires userId in session (use after requireAuth).
12
- */
13
- export declare function checkWorkspaceLimit(req: Request, res: Response, next: NextFunction): Promise<void>;
14
- /**
15
- * Middleware to check repository limit
16
- *
17
- * Use this middleware on repo connection endpoints.
18
- * Requires userId in session (use after requireAuth).
19
- */
20
- export declare function checkRepoLimit(req: Request, res: Response, next: NextFunction): Promise<void>;
21
- /**
22
- * Middleware to check concurrent agent limit
23
- *
24
- * Use this middleware on agent spawn endpoints.
25
- * Requires userId in session (use after requireAuth).
26
- * Optionally pass currentRunningAgents in request body for accurate count.
27
- */
28
- export declare function checkAgentLimit(req: Request, res: Response, next: NextFunction): Promise<void>;
29
- /**
30
- * Middleware to check coordinator access
31
- *
32
- * Use this middleware on coordinator-related endpoints.
33
- * Coordinators are only available on Pro plan and above.
34
- */
35
- export declare function checkCoordinatorAccess(req: Request, res: Response, next: NextFunction): Promise<void>;
36
- /**
37
- * Middleware to check session persistence access
38
- *
39
- * Use this middleware on endpoints that enable cloud session persistence.
40
- * Session persistence is only available on Pro plan and above.
41
- */
42
- export declare function checkSessionPersistenceAccess(req: Request, res: Response, next: NextFunction): Promise<void>;
43
- //# sourceMappingURL=planLimits.d.ts.map