@chaaskit/server 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 (189) hide show
  1. package/dist/api/admin.js +438 -0
  2. package/dist/api/admin.js.map +1 -0
  3. package/dist/api/agents.js +21 -0
  4. package/dist/api/agents.js.map +1 -0
  5. package/dist/api/api-keys.js +122 -0
  6. package/dist/api/api-keys.js.map +1 -0
  7. package/dist/api/auth.js +399 -0
  8. package/dist/api/auth.js.map +1 -0
  9. package/dist/api/chat.js +900 -0
  10. package/dist/api/chat.js.map +1 -0
  11. package/dist/api/config.js +91 -0
  12. package/dist/api/config.js.map +1 -0
  13. package/dist/api/documents.js +237 -0
  14. package/dist/api/documents.js.map +1 -0
  15. package/dist/api/export.js +107 -0
  16. package/dist/api/export.js.map +1 -0
  17. package/dist/api/health.js +25 -0
  18. package/dist/api/health.js.map +1 -0
  19. package/dist/api/mcp-server.js +84 -0
  20. package/dist/api/mcp-server.js.map +1 -0
  21. package/dist/api/mcp.js +400 -0
  22. package/dist/api/mcp.js.map +1 -0
  23. package/dist/api/mentions.js +94 -0
  24. package/dist/api/mentions.js.map +1 -0
  25. package/dist/api/oauth.js +366 -0
  26. package/dist/api/oauth.js.map +1 -0
  27. package/dist/api/payments.js +473 -0
  28. package/dist/api/payments.js.map +1 -0
  29. package/dist/api/projects.js +301 -0
  30. package/dist/api/projects.js.map +1 -0
  31. package/dist/api/scheduled-prompts.js +617 -0
  32. package/dist/api/scheduled-prompts.js.map +1 -0
  33. package/dist/api/search.js +85 -0
  34. package/dist/api/search.js.map +1 -0
  35. package/dist/api/share.js +188 -0
  36. package/dist/api/share.js.map +1 -0
  37. package/dist/api/slack.js +468 -0
  38. package/dist/api/slack.js.map +1 -0
  39. package/dist/api/teams.js +693 -0
  40. package/dist/api/teams.js.map +1 -0
  41. package/dist/api/templates.js +134 -0
  42. package/dist/api/templates.js.map +1 -0
  43. package/dist/api/threads.js +323 -0
  44. package/dist/api/threads.js.map +1 -0
  45. package/dist/api/upload.js +57 -0
  46. package/dist/api/upload.js.map +1 -0
  47. package/dist/api/user.js +111 -0
  48. package/dist/api/user.js.map +1 -0
  49. package/dist/api/v1/openai.js +245 -0
  50. package/dist/api/v1/openai.js.map +1 -0
  51. package/dist/app.js +168 -0
  52. package/dist/app.js.map +1 -0
  53. package/dist/bin/cli.js +57 -0
  54. package/dist/bin/cli.js.map +1 -0
  55. package/dist/commands/db-sync.js +108 -0
  56. package/dist/commands/db-sync.js.map +1 -0
  57. package/dist/config/loader.js +374 -0
  58. package/dist/config/loader.js.map +1 -0
  59. package/dist/documents/extractors.js +136 -0
  60. package/dist/documents/extractors.js.map +1 -0
  61. package/dist/extensions/glob.js +53 -0
  62. package/dist/extensions/glob.js.map +1 -0
  63. package/dist/extensions/loader.js +72 -0
  64. package/dist/extensions/loader.js.map +1 -0
  65. package/dist/index.js +25 -0
  66. package/dist/index.js.map +1 -0
  67. package/dist/loaders/index.js +75 -0
  68. package/dist/loaders/index.js.map +1 -0
  69. package/dist/mcp/client.js +551 -0
  70. package/dist/mcp/client.js.map +1 -0
  71. package/dist/mcp/server.js +335 -0
  72. package/dist/mcp/server.js.map +1 -0
  73. package/dist/middleware/apiKeyAuth.js +136 -0
  74. package/dist/middleware/apiKeyAuth.js.map +1 -0
  75. package/dist/middleware/auth.js +192 -0
  76. package/dist/middleware/auth.js.map +1 -0
  77. package/dist/middleware/errorHandler.js +41 -0
  78. package/dist/middleware/errorHandler.js.map +1 -0
  79. package/dist/middleware/mcpServerAuth.js +164 -0
  80. package/dist/middleware/mcpServerAuth.js.map +1 -0
  81. package/dist/middleware/requestLogger.js +9 -0
  82. package/dist/middleware/requestLogger.js.map +1 -0
  83. package/dist/middleware/team.js +132 -0
  84. package/dist/middleware/team.js.map +1 -0
  85. package/dist/oauth/server.js +410 -0
  86. package/dist/oauth/server.js.map +1 -0
  87. package/dist/queue/cli.js +93 -0
  88. package/dist/queue/cli.js.map +1 -0
  89. package/dist/queue/handlers/index.js +91 -0
  90. package/dist/queue/handlers/index.js.map +1 -0
  91. package/dist/queue/handlers/scheduled-prompt.js +270 -0
  92. package/dist/queue/handlers/scheduled-prompt.js.map +1 -0
  93. package/dist/queue/index.js +91 -0
  94. package/dist/queue/index.js.map +1 -0
  95. package/dist/queue/providers/memory.js +296 -0
  96. package/dist/queue/providers/memory.js.map +1 -0
  97. package/dist/queue/providers/sqs.js +275 -0
  98. package/dist/queue/providers/sqs.js.map +1 -0
  99. package/dist/queue/scheduler.js +355 -0
  100. package/dist/queue/scheduler.js.map +1 -0
  101. package/dist/queue/types.js +5 -0
  102. package/dist/queue/types.js.map +1 -0
  103. package/dist/queue/worker.js +230 -0
  104. package/dist/queue/worker.js.map +1 -0
  105. package/dist/registry/index.js +40 -0
  106. package/dist/registry/index.js.map +1 -0
  107. package/dist/server.js +207 -0
  108. package/dist/server.js.map +1 -0
  109. package/dist/services/agent.js +530 -0
  110. package/dist/services/agent.js.map +1 -0
  111. package/dist/services/agents.js +194 -0
  112. package/dist/services/agents.js.map +1 -0
  113. package/dist/services/documents.js +507 -0
  114. package/dist/services/documents.js.map +1 -0
  115. package/dist/services/email/index.js +91 -0
  116. package/dist/services/email/index.js.map +1 -0
  117. package/dist/services/email/providers/ses.js +97 -0
  118. package/dist/services/email/providers/ses.js.map +1 -0
  119. package/dist/services/email/templates.js +194 -0
  120. package/dist/services/email/templates.js.map +1 -0
  121. package/dist/services/email/types.js +5 -0
  122. package/dist/services/email/types.js.map +1 -0
  123. package/dist/services/encryption.js +69 -0
  124. package/dist/services/encryption.js.map +1 -0
  125. package/dist/services/oauth-discovery.js +226 -0
  126. package/dist/services/oauth-discovery.js.map +1 -0
  127. package/dist/services/pendingConfirmation.js +105 -0
  128. package/dist/services/pendingConfirmation.js.map +1 -0
  129. package/dist/services/scheduledPrompts.js +70 -0
  130. package/dist/services/scheduledPrompts.js.map +1 -0
  131. package/dist/services/slack/client.js +174 -0
  132. package/dist/services/slack/client.js.map +1 -0
  133. package/dist/services/slack/events.js +189 -0
  134. package/dist/services/slack/events.js.map +1 -0
  135. package/dist/services/slack/index.js +6 -0
  136. package/dist/services/slack/index.js.map +1 -0
  137. package/dist/services/slack/notifications.js +124 -0
  138. package/dist/services/slack/notifications.js.map +1 -0
  139. package/dist/services/slack/signature.js +74 -0
  140. package/dist/services/slack/signature.js.map +1 -0
  141. package/dist/services/slack/thread-context.js +191 -0
  142. package/dist/services/slack/thread-context.js.map +1 -0
  143. package/dist/services/toolConfirmation.js +55 -0
  144. package/dist/services/toolConfirmation.js.map +1 -0
  145. package/dist/services/usage.js +241 -0
  146. package/dist/services/usage.js.map +1 -0
  147. package/dist/ssr/build.js +90 -0
  148. package/dist/ssr/build.js.map +1 -0
  149. package/dist/ssr/components/SSRMessageList.js +120 -0
  150. package/dist/ssr/components/SSRMessageList.js.map +1 -0
  151. package/dist/ssr/entry.client.js +8 -0
  152. package/dist/ssr/entry.client.js.map +1 -0
  153. package/dist/ssr/entry.server.js +71 -0
  154. package/dist/ssr/entry.server.js.map +1 -0
  155. package/dist/ssr/handler.js +51 -0
  156. package/dist/ssr/handler.js.map +1 -0
  157. package/dist/ssr/root.js +184 -0
  158. package/dist/ssr/root.js.map +1 -0
  159. package/dist/ssr/routes/login.js +140 -0
  160. package/dist/ssr/routes/login.js.map +1 -0
  161. package/dist/ssr/routes/pricing.js +195 -0
  162. package/dist/ssr/routes/pricing.js.map +1 -0
  163. package/dist/ssr/routes/privacy.js +39 -0
  164. package/dist/ssr/routes/privacy.js.map +1 -0
  165. package/dist/ssr/routes/register.js +148 -0
  166. package/dist/ssr/routes/register.js.map +1 -0
  167. package/dist/ssr/routes/shared.$shareId.js +153 -0
  168. package/dist/ssr/routes/shared.$shareId.js.map +1 -0
  169. package/dist/ssr/routes/terms.js +39 -0
  170. package/dist/ssr/routes/terms.js.map +1 -0
  171. package/dist/storage/index.js +43 -0
  172. package/dist/storage/index.js.map +1 -0
  173. package/dist/storage/providers/database.js +38 -0
  174. package/dist/storage/providers/database.js.map +1 -0
  175. package/dist/storage/providers/filesystem.js +51 -0
  176. package/dist/storage/providers/filesystem.js.map +1 -0
  177. package/dist/storage/types.js +2 -0
  178. package/dist/storage/types.js.map +1 -0
  179. package/dist/tools/documents.js +336 -0
  180. package/dist/tools/documents.js.map +1 -0
  181. package/dist/tools/get-plan-usage.js +82 -0
  182. package/dist/tools/get-plan-usage.js.map +1 -0
  183. package/dist/tools/index.js +106 -0
  184. package/dist/tools/index.js.map +1 -0
  185. package/dist/tools/types.js +2 -0
  186. package/dist/tools/types.js.map +1 -0
  187. package/dist/tools/web-scrape.js +145 -0
  188. package/dist/tools/web-scrape.js.map +1 -0
  189. package/package.json +93 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mcp-server.js","sourceRoot":"","sources":["../../src/api/mcp-server.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,400 @@
1
+ import { Router } from 'express';
2
+ import { db } from '@chaaskit/db';
3
+ import { HTTP_STATUS } from '@chaaskit/shared';
4
+ import { requireAuth } from '../middleware/auth.js';
5
+ import { AppError } from '../middleware/errorHandler.js';
6
+ import { getConfig } from '../config/loader.js';
7
+ import { mcpManager } from '../mcp/client.js';
8
+ import { encryptCredential, generateOAuthState, generatePKCE, } from '../services/encryption.js';
9
+ import { discoverOAuthConfig } from '../services/oauth-discovery.js';
10
+ export const mcpRouter = Router();
11
+ // List available MCP servers
12
+ mcpRouter.get('/servers', requireAuth, async (req, res, next) => {
13
+ try {
14
+ const config = getConfig();
15
+ if (!config.mcp) {
16
+ res.json({ servers: [] });
17
+ return;
18
+ }
19
+ const servers = config.mcp.servers.map((server) => ({
20
+ id: server.id,
21
+ name: server.name,
22
+ transport: server.transport,
23
+ enabled: server.enabled,
24
+ connected: mcpManager.isConnected(server.id),
25
+ }));
26
+ res.json({ servers });
27
+ }
28
+ catch (error) {
29
+ next(error);
30
+ }
31
+ });
32
+ // List tools from all connected servers
33
+ mcpRouter.get('/tools', requireAuth, async (req, res, next) => {
34
+ try {
35
+ const tools = await mcpManager.listAllTools();
36
+ res.json({ tools });
37
+ }
38
+ catch (error) {
39
+ next(error);
40
+ }
41
+ });
42
+ // List tools from specific server
43
+ mcpRouter.get('/servers/:serverId/tools', requireAuth, async (req, res, next) => {
44
+ try {
45
+ const { serverId } = req.params;
46
+ const tools = await mcpManager.listTools(serverId);
47
+ res.json({ tools });
48
+ }
49
+ catch (error) {
50
+ next(error);
51
+ }
52
+ });
53
+ // Invoke a tool
54
+ mcpRouter.post('/tools/:serverId/:toolName', requireAuth, async (req, res, next) => {
55
+ try {
56
+ const config = getConfig();
57
+ const { serverId, toolName } = req.params;
58
+ const { arguments: args } = req.body;
59
+ if (!config.mcp) {
60
+ throw new AppError(HTTP_STATUS.BAD_REQUEST, 'MCP is not configured');
61
+ }
62
+ // Check if tool confirmation is required (for direct API calls, use simple mode check)
63
+ const confirmationMode = config.mcp.toolConfirmation?.mode || 'none';
64
+ if (confirmationMode !== 'none' && !req.body.confirmed) {
65
+ // Return the tool call for confirmation
66
+ res.json({
67
+ requiresConfirmation: true,
68
+ serverId,
69
+ toolName,
70
+ arguments: args,
71
+ });
72
+ return;
73
+ }
74
+ const result = await mcpManager.callTool(serverId, toolName, args);
75
+ res.json({ result });
76
+ }
77
+ catch (error) {
78
+ next(error);
79
+ }
80
+ });
81
+ // Connect to a server
82
+ mcpRouter.post('/servers/:serverId/connect', requireAuth, async (req, res, next) => {
83
+ try {
84
+ const config = getConfig();
85
+ const { serverId } = req.params;
86
+ if (!config.mcp) {
87
+ throw new AppError(HTTP_STATUS.BAD_REQUEST, 'MCP is not configured');
88
+ }
89
+ const serverConfig = config.mcp.servers.find((s) => s.id === serverId);
90
+ if (!serverConfig) {
91
+ throw new AppError(HTTP_STATUS.NOT_FOUND, 'Server not found');
92
+ }
93
+ await mcpManager.connect(serverConfig);
94
+ res.json({ connected: true });
95
+ }
96
+ catch (error) {
97
+ next(error);
98
+ }
99
+ });
100
+ // Disconnect from a server
101
+ mcpRouter.post('/servers/:serverId/disconnect', requireAuth, async (req, res, next) => {
102
+ try {
103
+ const { serverId } = req.params;
104
+ await mcpManager.disconnect(serverId);
105
+ res.json({ connected: false });
106
+ }
107
+ catch (error) {
108
+ next(error);
109
+ }
110
+ });
111
+ // ============================================================
112
+ // Credential Management Endpoints
113
+ // ============================================================
114
+ // List credential status for all servers requiring user credentials
115
+ mcpRouter.get('/credentials', requireAuth, async (req, res, next) => {
116
+ try {
117
+ const config = getConfig();
118
+ const userId = req.user.id;
119
+ if (!config.mcp) {
120
+ res.json({ credentials: [] });
121
+ return;
122
+ }
123
+ // Get user's existing credentials
124
+ const userCredentials = await db.mCPCredential.findMany({
125
+ where: { userId },
126
+ select: { serverId: true, credentialType: true },
127
+ });
128
+ const credentialMap = new Map(userCredentials.map((c) => [c.serverId, c.credentialType]));
129
+ // Build status for servers that require user credentials
130
+ const credentials = config.mcp.servers
131
+ .filter((server) => {
132
+ const authMode = server.authMode || 'none';
133
+ return authMode === 'user-apikey' || authMode === 'user-oauth';
134
+ })
135
+ .map((server) => ({
136
+ serverId: server.id,
137
+ serverName: server.name,
138
+ authMode: server.authMode || 'none',
139
+ hasCredential: credentialMap.has(server.id),
140
+ credentialType: credentialMap.get(server.id),
141
+ userInstructions: server.userInstructions,
142
+ }));
143
+ console.log(`[MCP] Returning ${credentials.length} credential statuses for user ${userId}`);
144
+ res.json({ credentials });
145
+ }
146
+ catch (error) {
147
+ next(error);
148
+ }
149
+ });
150
+ // Set API key for a server
151
+ mcpRouter.post('/credentials/:serverId/apikey', requireAuth, async (req, res, next) => {
152
+ try {
153
+ const config = getConfig();
154
+ const { serverId } = req.params;
155
+ const { apiKey } = req.body;
156
+ const userId = req.user.id;
157
+ if (!apiKey || typeof apiKey !== 'string') {
158
+ throw new AppError(HTTP_STATUS.BAD_REQUEST, 'API key is required');
159
+ }
160
+ if (!config.mcp) {
161
+ throw new AppError(HTTP_STATUS.BAD_REQUEST, 'MCP is not configured');
162
+ }
163
+ const serverConfig = config.mcp.servers.find((s) => s.id === serverId);
164
+ if (!serverConfig) {
165
+ throw new AppError(HTTP_STATUS.NOT_FOUND, 'Server not found');
166
+ }
167
+ if (serverConfig.authMode !== 'user-apikey') {
168
+ throw new AppError(HTTP_STATUS.BAD_REQUEST, 'Server does not accept user API keys');
169
+ }
170
+ // Encrypt and store the credential
171
+ const encryptedData = encryptCredential({ apiKey });
172
+ await db.mCPCredential.upsert({
173
+ where: {
174
+ userId_serverId: { userId, serverId },
175
+ },
176
+ update: {
177
+ credentialType: 'api_key',
178
+ encryptedData,
179
+ oauthState: null,
180
+ codeVerifier: null,
181
+ updatedAt: new Date(),
182
+ },
183
+ create: {
184
+ userId,
185
+ serverId,
186
+ credentialType: 'api_key',
187
+ encryptedData,
188
+ },
189
+ });
190
+ // Disconnect existing user client so it reconnects with new credentials
191
+ await mcpManager.disconnectUser(userId, serverId);
192
+ res.json({ success: true });
193
+ }
194
+ catch (error) {
195
+ next(error);
196
+ }
197
+ });
198
+ // Remove credential for a server
199
+ mcpRouter.delete('/credentials/:serverId', requireAuth, async (req, res, next) => {
200
+ try {
201
+ const { serverId } = req.params;
202
+ const userId = req.user.id;
203
+ await db.mCPCredential.deleteMany({
204
+ where: { userId, serverId },
205
+ });
206
+ // Disconnect user client
207
+ await mcpManager.disconnectUser(userId, serverId);
208
+ res.json({ success: true });
209
+ }
210
+ catch (error) {
211
+ next(error);
212
+ }
213
+ });
214
+ // ============================================================
215
+ // OAuth Flow Endpoints
216
+ // ============================================================
217
+ // Start OAuth flow - returns authorization URL
218
+ mcpRouter.get('/oauth/:serverId/authorize', requireAuth, async (req, res, next) => {
219
+ try {
220
+ const config = getConfig();
221
+ const { serverId } = req.params;
222
+ const userId = req.user.id;
223
+ if (!config.mcp) {
224
+ throw new AppError(HTTP_STATUS.BAD_REQUEST, 'MCP is not configured');
225
+ }
226
+ const serverConfig = config.mcp.servers.find((s) => s.id === serverId);
227
+ if (!serverConfig) {
228
+ throw new AppError(HTTP_STATUS.NOT_FOUND, 'Server not found');
229
+ }
230
+ if (serverConfig.authMode !== 'user-oauth') {
231
+ throw new AppError(HTTP_STATUS.BAD_REQUEST, 'Server does not support OAuth');
232
+ }
233
+ // Discover OAuth configuration (endpoints + client registration)
234
+ const oauthConfig = await discoverOAuthConfig(serverConfig);
235
+ if (!oauthConfig) {
236
+ throw new AppError(HTTP_STATUS.INTERNAL_SERVER_ERROR, 'Failed to discover OAuth configuration for this server');
237
+ }
238
+ // Generate PKCE challenge and state
239
+ const { codeVerifier, codeChallenge } = generatePKCE();
240
+ const state = generateOAuthState();
241
+ // Store state and code verifier in credential record
242
+ const encryptedVerifier = encryptCredential({ codeVerifier });
243
+ await db.mCPCredential.upsert({
244
+ where: {
245
+ userId_serverId: { userId, serverId },
246
+ },
247
+ update: {
248
+ oauthState: state,
249
+ codeVerifier: encryptedVerifier,
250
+ updatedAt: new Date(),
251
+ },
252
+ create: {
253
+ userId,
254
+ serverId,
255
+ credentialType: 'oauth',
256
+ encryptedData: '', // Will be updated after OAuth completes
257
+ oauthState: state,
258
+ codeVerifier: encryptedVerifier,
259
+ },
260
+ });
261
+ // Build authorization URL
262
+ const apiUrl = process.env.API_URL || 'http://localhost:3000';
263
+ const redirectUri = `${apiUrl}/api/mcp/oauth/callback`;
264
+ const authUrl = new URL(oauthConfig.authorizationEndpoint);
265
+ authUrl.searchParams.set('client_id', oauthConfig.clientId);
266
+ authUrl.searchParams.set('redirect_uri', redirectUri);
267
+ authUrl.searchParams.set('response_type', 'code');
268
+ authUrl.searchParams.set('state', `${serverId}:${state}`);
269
+ authUrl.searchParams.set('code_challenge', codeChallenge);
270
+ authUrl.searchParams.set('code_challenge_method', 'S256');
271
+ // RFC 8707: Resource indicator for audience binding (required by MCP spec)
272
+ if (serverConfig.url) {
273
+ authUrl.searchParams.set('resource', serverConfig.url);
274
+ }
275
+ if (oauthConfig.scopes && oauthConfig.scopes.length > 0) {
276
+ authUrl.searchParams.set('scope', oauthConfig.scopes.join(' '));
277
+ }
278
+ res.json({ authorizationUrl: authUrl.toString() });
279
+ }
280
+ catch (error) {
281
+ next(error);
282
+ }
283
+ });
284
+ // OAuth callback - exchanges code for tokens
285
+ mcpRouter.get('/oauth/callback', async (req, res, next) => {
286
+ try {
287
+ const config = getConfig();
288
+ const { code, state, error: oauthError, error_description } = req.query;
289
+ const appUrl = process.env.APP_URL || 'http://localhost:5173';
290
+ const settingsUrl = `${appUrl}/settings`;
291
+ if (oauthError) {
292
+ console.error(`[MCP OAuth] Error: ${oauthError} - ${error_description}`);
293
+ res.redirect(`${settingsUrl}?error=oauth_${oauthError}`);
294
+ return;
295
+ }
296
+ if (!code || !state || typeof code !== 'string' || typeof state !== 'string') {
297
+ res.redirect(`${settingsUrl}?error=oauth_invalid_response`);
298
+ return;
299
+ }
300
+ // Parse state: serverId:oauthState
301
+ const stateIndex = state.indexOf(':');
302
+ if (stateIndex === -1) {
303
+ res.redirect(`${settingsUrl}?error=oauth_invalid_state`);
304
+ return;
305
+ }
306
+ const serverId = state.slice(0, stateIndex);
307
+ const oauthState = state.slice(stateIndex + 1);
308
+ if (!config.mcp) {
309
+ res.redirect(`${settingsUrl}?error=mcp_not_configured`);
310
+ return;
311
+ }
312
+ const serverConfig = config.mcp.servers.find((s) => s.id === serverId);
313
+ if (!serverConfig) {
314
+ res.redirect(`${settingsUrl}?error=server_not_found`);
315
+ return;
316
+ }
317
+ // Find credential with matching state
318
+ const credential = await db.mCPCredential.findFirst({
319
+ where: {
320
+ serverId,
321
+ oauthState,
322
+ },
323
+ });
324
+ if (!credential || !credential.codeVerifier) {
325
+ res.redirect(`${settingsUrl}?error=oauth_state_mismatch`);
326
+ return;
327
+ }
328
+ // Discover OAuth configuration
329
+ const oauthConfig = await discoverOAuthConfig(serverConfig);
330
+ if (!oauthConfig) {
331
+ res.redirect(`${settingsUrl}?error=oauth_discovery_failed`);
332
+ return;
333
+ }
334
+ // Decrypt code verifier
335
+ const { decryptCredential } = await import('../services/encryption.js');
336
+ const { codeVerifier } = decryptCredential(credential.codeVerifier);
337
+ const apiUrl = process.env.API_URL || 'http://localhost:3000';
338
+ const redirectUri = `${apiUrl}/api/mcp/oauth/callback`;
339
+ // Exchange code for tokens
340
+ // RFC 8707: Include resource parameter for audience binding (required by MCP spec)
341
+ const tokenParams = {
342
+ grant_type: 'authorization_code',
343
+ code,
344
+ redirect_uri: redirectUri,
345
+ client_id: oauthConfig.clientId,
346
+ code_verifier: codeVerifier,
347
+ };
348
+ if (oauthConfig.clientSecret) {
349
+ tokenParams.client_secret = oauthConfig.clientSecret;
350
+ }
351
+ // Include resource parameter for token audience binding
352
+ if (serverConfig.url) {
353
+ tokenParams.resource = serverConfig.url;
354
+ }
355
+ const tokenResponse = await fetch(oauthConfig.tokenEndpoint, {
356
+ method: 'POST',
357
+ headers: {
358
+ 'Content-Type': 'application/x-www-form-urlencoded',
359
+ Accept: 'application/json',
360
+ },
361
+ body: new URLSearchParams(tokenParams),
362
+ });
363
+ if (!tokenResponse.ok) {
364
+ const errorText = await tokenResponse.text();
365
+ console.error(`[MCP OAuth] Token exchange failed:`, errorText);
366
+ res.redirect(`${settingsUrl}?error=oauth_token_exchange_failed`);
367
+ return;
368
+ }
369
+ const tokenData = await tokenResponse.json();
370
+ // Store tokens
371
+ const expiresAt = tokenData.expires_in
372
+ ? Math.floor(Date.now() / 1000) + tokenData.expires_in
373
+ : undefined;
374
+ const encryptedData = encryptCredential({
375
+ accessToken: tokenData.access_token,
376
+ refreshToken: tokenData.refresh_token,
377
+ expiresAt,
378
+ tokenType: tokenData.token_type || 'Bearer',
379
+ });
380
+ await db.mCPCredential.update({
381
+ where: { id: credential.id },
382
+ data: {
383
+ credentialType: 'oauth',
384
+ encryptedData,
385
+ oauthState: null,
386
+ codeVerifier: null,
387
+ updatedAt: new Date(),
388
+ },
389
+ });
390
+ // Disconnect existing user client so it reconnects with new credentials
391
+ await mcpManager.disconnectUser(credential.userId, serverId);
392
+ res.redirect(`${appUrl}/?openSettings=true&success=oauth_connected&server=${serverId}`);
393
+ }
394
+ catch (error) {
395
+ console.error('[MCP OAuth] Callback error:', error);
396
+ const appUrl = process.env.APP_URL || 'http://localhost:5173';
397
+ res.redirect(`${appUrl}/?openSettings=true&error=oauth_callback_error`);
398
+ }
399
+ });
400
+ //# sourceMappingURL=mcp.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mcp.js","sourceRoot":"","sources":["../../src/api/mcp.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,94 @@
1
+ import { Router } from 'express';
2
+ import { HTTP_STATUS } from '@chaaskit/shared';
3
+ import { requireAuth } from '../middleware/auth.js';
4
+ import { AppError } from '../middleware/errorHandler.js';
5
+ import { getConfig } from '../config/loader.js';
6
+ import { documentService } from '../services/documents.js';
7
+ export const mentionsRouter = Router();
8
+ // Middleware to check if documents feature is enabled
9
+ mentionsRouter.use((req, res, next) => {
10
+ const config = getConfig();
11
+ if (!config.documents?.enabled) {
12
+ return next(new AppError(HTTP_STATUS.FORBIDDEN, 'Documents are not enabled'));
13
+ }
14
+ next();
15
+ });
16
+ /**
17
+ * Search mentionable resources
18
+ * GET /api/mentions/search?q=query&scope=my|team|project&teamId=...&projectId=...
19
+ */
20
+ mentionsRouter.get('/search', requireAuth, async (req, res, next) => {
21
+ try {
22
+ const userId = req.user.id;
23
+ const { q, scope, teamId, projectId, limit } = req.query;
24
+ const documents = await documentService.search(userId, {
25
+ query: q,
26
+ scopes: scope ? [scope] : undefined,
27
+ teamId: teamId,
28
+ projectId: projectId,
29
+ limit: limit ? parseInt(limit, 10) : 20,
30
+ });
31
+ // Group by scope for easier frontend rendering
32
+ const grouped = {
33
+ my: documents.filter((d) => d.scope === 'my'),
34
+ team: documents.filter((d) => d.scope === 'team'),
35
+ project: documents.filter((d) => d.scope === 'project'),
36
+ };
37
+ res.json({
38
+ documents,
39
+ grouped,
40
+ hasMore: documents.length >= (limit ? parseInt(limit, 10) : 20),
41
+ });
42
+ }
43
+ catch (error) {
44
+ next(error);
45
+ }
46
+ });
47
+ /**
48
+ * Resolve document paths to metadata
49
+ * POST /api/mentions/resolve
50
+ * Body: { paths: ["@my/doc1", "@team/eng/doc2"] }
51
+ */
52
+ mentionsRouter.post('/resolve', requireAuth, async (req, res, next) => {
53
+ try {
54
+ const userId = req.user.id;
55
+ const { paths } = req.body;
56
+ if (!Array.isArray(paths)) {
57
+ throw new AppError(HTTP_STATUS.BAD_REQUEST, 'paths must be an array');
58
+ }
59
+ const resolved = await documentService.resolveMany(paths, userId);
60
+ // Convert Map to object for JSON response
61
+ const result = {};
62
+ for (const [path, doc] of resolved) {
63
+ result[path] = doc;
64
+ }
65
+ res.json({ resolved: result });
66
+ }
67
+ catch (error) {
68
+ next(error);
69
+ }
70
+ });
71
+ /**
72
+ * Get available mention types
73
+ * GET /api/mentions/types
74
+ */
75
+ mentionsRouter.get('/types', requireAuth, async (req, res, next) => {
76
+ try {
77
+ // For now, only documents are supported
78
+ // This endpoint can be extended when other mention providers are added
79
+ const types = [
80
+ {
81
+ type: 'document',
82
+ name: 'Documents',
83
+ icon: 'file-text',
84
+ description: 'Reference uploaded documents',
85
+ pathPattern: '@my/name, @team/slug/name, @project/slug/name',
86
+ },
87
+ ];
88
+ res.json({ types });
89
+ }
90
+ catch (error) {
91
+ next(error);
92
+ }
93
+ });
94
+ //# sourceMappingURL=mentions.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mentions.js","sourceRoot":"","sources":["../../src/api/mentions.ts"],"names":[],"mappings":"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"}