@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,438 @@
1
+ import { Router } from 'express';
2
+ import { db } from '@chaaskit/db';
3
+ import { z } from 'zod';
4
+ import { HTTP_STATUS } from '@chaaskit/shared';
5
+ import { requireAuth, requireAdmin } from '../middleware/auth.js';
6
+ import { AppError } from '../middleware/errorHandler.js';
7
+ export const adminRouter = Router();
8
+ // All admin routes require authentication and admin role
9
+ adminRouter.use(requireAuth, requireAdmin);
10
+ // Get admin dashboard stats
11
+ adminRouter.get('/stats', async (req, res, next) => {
12
+ try {
13
+ const thirtyDaysAgo = new Date();
14
+ thirtyDaysAgo.setDate(thirtyDaysAgo.getDate() - 30);
15
+ const [totalUsers, totalTeams, totalThreads, totalMessages, newUsersLast30Days, messagesLast30Days, planCounts,] = await Promise.all([
16
+ db.user.count(),
17
+ db.team.count({ where: { archivedAt: null } }),
18
+ db.thread.count(),
19
+ db.message.count(),
20
+ db.user.count({
21
+ where: { createdAt: { gte: thirtyDaysAgo } },
22
+ }),
23
+ db.message.count({
24
+ where: { createdAt: { gte: thirtyDaysAgo } },
25
+ }),
26
+ db.user.groupBy({
27
+ by: ['plan'],
28
+ _count: { plan: true },
29
+ }),
30
+ ]);
31
+ const planDistribution = {};
32
+ for (const item of planCounts) {
33
+ planDistribution[item.plan] = item._count.plan;
34
+ }
35
+ const stats = {
36
+ totalUsers,
37
+ totalTeams,
38
+ totalThreads,
39
+ totalMessages,
40
+ planDistribution,
41
+ newUsersLast30Days,
42
+ messagesLast30Days,
43
+ };
44
+ res.json(stats);
45
+ }
46
+ catch (error) {
47
+ next(error);
48
+ }
49
+ });
50
+ // Get usage data over time
51
+ adminRouter.get('/usage', async (req, res, next) => {
52
+ try {
53
+ const periodParam = req.query.period || 'day';
54
+ const daysParam = parseInt(req.query.days) || 30;
55
+ // Limit to reasonable ranges
56
+ const days = Math.min(Math.max(daysParam, 1), 365);
57
+ const startDate = new Date();
58
+ startDate.setDate(startDate.getDate() - days);
59
+ startDate.setHours(0, 0, 0, 0);
60
+ // Get messages with token counts grouped by date
61
+ const messages = await db.message.findMany({
62
+ where: {
63
+ createdAt: { gte: startDate },
64
+ },
65
+ select: {
66
+ createdAt: true,
67
+ inputTokens: true,
68
+ outputTokens: true,
69
+ },
70
+ });
71
+ // Group by date
72
+ const usageByDate = new Map();
73
+ for (const msg of messages) {
74
+ const dateKey = msg.createdAt.toISOString().split('T')[0];
75
+ if (!usageByDate.has(dateKey)) {
76
+ usageByDate.set(dateKey, {
77
+ date: dateKey,
78
+ messages: 0,
79
+ inputTokens: 0,
80
+ outputTokens: 0,
81
+ });
82
+ }
83
+ const data = usageByDate.get(dateKey);
84
+ data.messages += 1;
85
+ data.inputTokens += msg.inputTokens || 0;
86
+ data.outputTokens += msg.outputTokens || 0;
87
+ }
88
+ // Fill in missing dates with zeros
89
+ const usage = [];
90
+ const currentDate = new Date(startDate);
91
+ const endDate = new Date();
92
+ while (currentDate <= endDate) {
93
+ const dateKey = currentDate.toISOString().split('T')[0];
94
+ usage.push(usageByDate.get(dateKey) || {
95
+ date: dateKey,
96
+ messages: 0,
97
+ inputTokens: 0,
98
+ outputTokens: 0,
99
+ });
100
+ currentDate.setDate(currentDate.getDate() + 1);
101
+ }
102
+ res.json({
103
+ usage,
104
+ period: periodParam,
105
+ });
106
+ }
107
+ catch (error) {
108
+ next(error);
109
+ }
110
+ });
111
+ // Get feedback stats and recent feedback
112
+ adminRouter.get('/feedback', async (req, res, next) => {
113
+ try {
114
+ const limit = Math.min(parseInt(req.query.limit) || 20, 100);
115
+ const [totalUp, totalDown, recentFeedback] = await Promise.all([
116
+ db.messageFeedback.count({ where: { type: 'up' } }),
117
+ db.messageFeedback.count({ where: { type: 'down' } }),
118
+ db.messageFeedback.findMany({
119
+ take: limit,
120
+ orderBy: { createdAt: 'desc' },
121
+ include: {
122
+ user: {
123
+ select: {
124
+ id: true,
125
+ email: true,
126
+ name: true,
127
+ },
128
+ },
129
+ message: {
130
+ select: {
131
+ id: true,
132
+ content: true,
133
+ threadId: true,
134
+ },
135
+ },
136
+ },
137
+ }),
138
+ ]);
139
+ const feedbackItems = recentFeedback.map((f) => ({
140
+ id: f.id,
141
+ type: f.type,
142
+ comment: f.comment,
143
+ createdAt: f.createdAt,
144
+ user: f.user,
145
+ message: {
146
+ ...f.message,
147
+ content: f.message.content.slice(0, 200) + (f.message.content.length > 200 ? '...' : ''),
148
+ },
149
+ }));
150
+ const stats = {
151
+ totalUp,
152
+ totalDown,
153
+ recentFeedback: feedbackItems,
154
+ };
155
+ res.json(stats);
156
+ }
157
+ catch (error) {
158
+ next(error);
159
+ }
160
+ });
161
+ // Get paginated user list
162
+ adminRouter.get('/users', async (req, res, next) => {
163
+ try {
164
+ const page = Math.max(parseInt(req.query.page) || 1, 1);
165
+ const pageSize = Math.min(Math.max(parseInt(req.query.pageSize) || 20, 1), 100);
166
+ const search = req.query.search || '';
167
+ const skip = (page - 1) * pageSize;
168
+ const whereClause = search
169
+ ? {
170
+ OR: [
171
+ { email: { contains: search, mode: 'insensitive' } },
172
+ { name: { contains: search, mode: 'insensitive' } },
173
+ ],
174
+ }
175
+ : {};
176
+ const [users, total] = await Promise.all([
177
+ db.user.findMany({
178
+ where: whereClause,
179
+ skip,
180
+ take: pageSize,
181
+ orderBy: { createdAt: 'desc' },
182
+ select: {
183
+ id: true,
184
+ email: true,
185
+ name: true,
186
+ avatarUrl: true,
187
+ isAdmin: true,
188
+ plan: true,
189
+ messagesThisMonth: true,
190
+ credits: true,
191
+ emailVerified: true,
192
+ oauthProvider: true,
193
+ createdAt: true,
194
+ _count: {
195
+ select: {
196
+ threads: true,
197
+ teamMemberships: true,
198
+ },
199
+ },
200
+ teamMemberships: {
201
+ select: {
202
+ role: true,
203
+ team: {
204
+ select: {
205
+ id: true,
206
+ name: true,
207
+ },
208
+ },
209
+ },
210
+ },
211
+ },
212
+ }),
213
+ db.user.count({ where: whereClause }),
214
+ ]);
215
+ const adminUsers = users.map((u) => ({
216
+ id: u.id,
217
+ email: u.email,
218
+ name: u.name,
219
+ avatarUrl: u.avatarUrl,
220
+ isAdmin: u.isAdmin,
221
+ plan: u.plan,
222
+ messagesThisMonth: u.messagesThisMonth,
223
+ credits: u.credits,
224
+ emailVerified: u.emailVerified,
225
+ oauthProvider: u.oauthProvider,
226
+ createdAt: u.createdAt,
227
+ threadCount: u._count.threads,
228
+ teamCount: u._count.teamMemberships,
229
+ teams: u.teamMemberships.map((tm) => ({
230
+ id: tm.team.id,
231
+ name: tm.team.name,
232
+ role: tm.role,
233
+ })),
234
+ }));
235
+ res.json({
236
+ users: adminUsers,
237
+ total,
238
+ page,
239
+ pageSize,
240
+ });
241
+ }
242
+ catch (error) {
243
+ next(error);
244
+ }
245
+ });
246
+ // Update user (admin toggle, plan change)
247
+ const updateUserSchema = z.object({
248
+ isAdmin: z.boolean().optional(),
249
+ plan: z.string().optional(),
250
+ });
251
+ adminRouter.patch('/users/:userId', async (req, res, next) => {
252
+ try {
253
+ const { userId } = req.params;
254
+ const updates = updateUserSchema.parse(req.body);
255
+ // Prevent self-demotion
256
+ if (userId === req.user.id && updates.isAdmin === false) {
257
+ throw new AppError(HTTP_STATUS.BAD_REQUEST, 'Cannot remove your own admin status');
258
+ }
259
+ const user = await db.user.findUnique({
260
+ where: { id: userId },
261
+ });
262
+ if (!user) {
263
+ throw new AppError(HTTP_STATUS.NOT_FOUND, 'User not found');
264
+ }
265
+ const updatedUser = await db.user.update({
266
+ where: { id: userId },
267
+ data: updates,
268
+ select: {
269
+ id: true,
270
+ email: true,
271
+ name: true,
272
+ avatarUrl: true,
273
+ isAdmin: true,
274
+ plan: true,
275
+ messagesThisMonth: true,
276
+ credits: true,
277
+ emailVerified: true,
278
+ oauthProvider: true,
279
+ createdAt: true,
280
+ teamMemberships: {
281
+ select: {
282
+ role: true,
283
+ team: {
284
+ select: {
285
+ id: true,
286
+ name: true,
287
+ },
288
+ },
289
+ },
290
+ },
291
+ _count: {
292
+ select: {
293
+ threads: true,
294
+ teamMemberships: true,
295
+ },
296
+ },
297
+ },
298
+ });
299
+ const adminUser = {
300
+ id: updatedUser.id,
301
+ email: updatedUser.email,
302
+ name: updatedUser.name,
303
+ avatarUrl: updatedUser.avatarUrl,
304
+ isAdmin: updatedUser.isAdmin,
305
+ plan: updatedUser.plan,
306
+ messagesThisMonth: updatedUser.messagesThisMonth,
307
+ credits: updatedUser.credits,
308
+ emailVerified: updatedUser.emailVerified,
309
+ oauthProvider: updatedUser.oauthProvider,
310
+ createdAt: updatedUser.createdAt,
311
+ threadCount: updatedUser._count.threads,
312
+ teamCount: updatedUser._count.teamMemberships,
313
+ teams: updatedUser.teamMemberships.map((tm) => ({
314
+ id: tm.team.id,
315
+ name: tm.team.name,
316
+ role: tm.role,
317
+ })),
318
+ };
319
+ res.json({ user: adminUser });
320
+ }
321
+ catch (error) {
322
+ next(error);
323
+ }
324
+ });
325
+ // Get paginated team list
326
+ adminRouter.get('/teams', async (req, res, next) => {
327
+ try {
328
+ const page = Math.max(parseInt(req.query.page) || 1, 1);
329
+ const pageSize = Math.min(Math.max(parseInt(req.query.pageSize) || 20, 1), 100);
330
+ const search = req.query.search || '';
331
+ const includeArchived = req.query.includeArchived === 'true';
332
+ const skip = (page - 1) * pageSize;
333
+ const whereClause = {
334
+ ...(search ? { name: { contains: search, mode: 'insensitive' } } : {}),
335
+ ...(includeArchived ? {} : { archivedAt: null }),
336
+ };
337
+ const [teams, total] = await Promise.all([
338
+ db.team.findMany({
339
+ where: whereClause,
340
+ skip,
341
+ take: pageSize,
342
+ orderBy: { createdAt: 'desc' },
343
+ select: {
344
+ id: true,
345
+ name: true,
346
+ createdAt: true,
347
+ archivedAt: true,
348
+ _count: {
349
+ select: {
350
+ members: true,
351
+ threads: true,
352
+ },
353
+ },
354
+ },
355
+ }),
356
+ db.team.count({ where: whereClause }),
357
+ ]);
358
+ const adminTeams = teams.map((t) => ({
359
+ id: t.id,
360
+ name: t.name,
361
+ memberCount: t._count.members,
362
+ threadCount: t._count.threads,
363
+ createdAt: t.createdAt,
364
+ archivedAt: t.archivedAt,
365
+ }));
366
+ res.json({
367
+ teams: adminTeams,
368
+ total,
369
+ page,
370
+ pageSize,
371
+ });
372
+ }
373
+ catch (error) {
374
+ next(error);
375
+ }
376
+ });
377
+ // Get single team with members
378
+ adminRouter.get('/teams/:teamId', async (req, res, next) => {
379
+ try {
380
+ const { teamId } = req.params;
381
+ const team = await db.team.findUnique({
382
+ where: { id: teamId },
383
+ select: {
384
+ id: true,
385
+ name: true,
386
+ context: true,
387
+ createdAt: true,
388
+ archivedAt: true,
389
+ _count: {
390
+ select: {
391
+ members: true,
392
+ threads: true,
393
+ },
394
+ },
395
+ members: {
396
+ select: {
397
+ role: true,
398
+ createdAt: true,
399
+ user: {
400
+ select: {
401
+ id: true,
402
+ email: true,
403
+ name: true,
404
+ avatarUrl: true,
405
+ },
406
+ },
407
+ },
408
+ orderBy: { createdAt: 'asc' },
409
+ },
410
+ },
411
+ });
412
+ if (!team) {
413
+ throw new AppError(HTTP_STATUS.NOT_FOUND, 'Team not found');
414
+ }
415
+ const adminTeamDetails = {
416
+ id: team.id,
417
+ name: team.name,
418
+ context: team.context,
419
+ memberCount: team._count.members,
420
+ threadCount: team._count.threads,
421
+ createdAt: team.createdAt,
422
+ archivedAt: team.archivedAt,
423
+ members: team.members.map((m) => ({
424
+ id: m.user.id,
425
+ email: m.user.email,
426
+ name: m.user.name,
427
+ avatarUrl: m.user.avatarUrl,
428
+ role: m.role,
429
+ joinedAt: m.createdAt,
430
+ })),
431
+ };
432
+ res.json(adminTeamDetails);
433
+ }
434
+ catch (error) {
435
+ next(error);
436
+ }
437
+ });
438
+ //# sourceMappingURL=admin.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"admin.js","sourceRoot":"","sources":["../../src/api/admin.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,21 @@
1
+ import { Router } from 'express';
2
+ import { optionalAuth } from '../middleware/auth.js';
3
+ import { getAgentsForPlan, getAgentClientInfo } from '../services/agents.js';
4
+ export const agentsRouter = Router();
5
+ /**
6
+ * GET /api/agents
7
+ * List agents available to the current user based on their plan.
8
+ * Returns agent info suitable for display (id, name, isDefault).
9
+ */
10
+ agentsRouter.get('/', optionalAuth, async (req, res, next) => {
11
+ try {
12
+ const userPlan = req.user?.plan;
13
+ const agents = getAgentsForPlan(userPlan);
14
+ const agentList = agents.map(getAgentClientInfo);
15
+ res.json({ agents: agentList });
16
+ }
17
+ catch (error) {
18
+ next(error);
19
+ }
20
+ });
21
+ //# sourceMappingURL=agents.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"agents.js","sourceRoot":"","sources":["../../src/api/agents.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AACjC,OAAO,EAAE,YAAY,EAAE,MAAM,uBAAuB,CAAC;AACrD,OAAO,EAAE,gBAAgB,EAAE,kBAAkB,EAAE,MAAM,uBAAuB,CAAC;AAE7E,MAAM,CAAC,MAAM,YAAY,GAAG,MAAM,EAAE,CAAC;AAErC;;;;GAIG;AACH,YAAY,CAAC,GAAG,CAAC,GAAG,EAAE,YAAY,EAAE,KAAK,EAAE,GAAG,EAAE,GAAG,EAAE,IAAI,EAAE,EAAE;IAC3D,IAAI,CAAC;QACH,MAAM,QAAQ,GAAG,GAAG,CAAC,IAAI,EAAE,IAAI,CAAC;QAChC,MAAM,MAAM,GAAG,gBAAgB,CAAC,QAAQ,CAAC,CAAC;QAE1C,MAAM,SAAS,GAAG,MAAM,CAAC,GAAG,CAAC,kBAAkB,CAAC,CAAC;QAEjD,GAAG,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,SAAS,EAAE,CAAC,CAAC;IAClC,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,IAAI,CAAC,KAAK,CAAC,CAAC;IACd,CAAC;AACH,CAAC,CAAC,CAAC"}
@@ -0,0 +1,122 @@
1
+ import { Router } from 'express';
2
+ import { randomBytes } from 'crypto';
3
+ import bcrypt from 'bcryptjs';
4
+ import { db } from '@chaaskit/db';
5
+ import { HTTP_STATUS } from '@chaaskit/shared';
6
+ import { requireAuth } from '../middleware/auth.js';
7
+ import { getConfig } from '../config/loader.js';
8
+ export const apiKeysRouter = Router();
9
+ // Check if user can access API keys feature
10
+ function canAccessApiKeys(user) {
11
+ const config = getConfig();
12
+ if (!config.api?.enabled)
13
+ return false;
14
+ if (!config.api.allowedPlans)
15
+ return true;
16
+ return config.api.allowedPlans.includes(user.plan);
17
+ }
18
+ // List user's API keys (masked)
19
+ apiKeysRouter.get('/', requireAuth, async (req, res, next) => {
20
+ try {
21
+ if (!canAccessApiKeys(req.user)) {
22
+ res.status(HTTP_STATUS.FORBIDDEN).json({ error: 'API access not available for your plan' });
23
+ return;
24
+ }
25
+ const keys = await db.apiKey.findMany({
26
+ where: { userId: req.user.id },
27
+ select: {
28
+ id: true,
29
+ name: true,
30
+ keyPrefix: true,
31
+ teamId: true,
32
+ team: { select: { id: true, name: true } },
33
+ lastUsedAt: true,
34
+ expiresAt: true,
35
+ createdAt: true,
36
+ },
37
+ orderBy: { createdAt: 'desc' },
38
+ });
39
+ res.json({ keys });
40
+ }
41
+ catch (error) {
42
+ next(error);
43
+ }
44
+ });
45
+ // Create new API key
46
+ apiKeysRouter.post('/', requireAuth, async (req, res, next) => {
47
+ try {
48
+ if (!canAccessApiKeys(req.user)) {
49
+ res.status(HTTP_STATUS.FORBIDDEN).json({ error: 'API access not available for your plan' });
50
+ return;
51
+ }
52
+ const { name, expiresAt, teamId } = req.body;
53
+ // If teamId provided, verify user is a member of that team
54
+ if (teamId) {
55
+ const membership = await db.teamMember.findFirst({
56
+ where: { userId: req.user.id, teamId },
57
+ });
58
+ if (!membership) {
59
+ res.status(HTTP_STATUS.FORBIDDEN).json({ error: 'You are not a member of this team' });
60
+ return;
61
+ }
62
+ }
63
+ // Generate key: <prefix><48 random chars>
64
+ const config = getConfig();
65
+ const prefix = config.api?.keyPrefix || 'sk-';
66
+ const rawKey = `${prefix}${randomBytes(24).toString('hex')}`;
67
+ // Store enough chars to identify the key (prefix + first few random chars)
68
+ const keyPrefix = rawKey.slice(0, prefix.length + 6);
69
+ const keyHash = await bcrypt.hash(rawKey, 10);
70
+ const apiKey = await db.apiKey.create({
71
+ data: {
72
+ userId: req.user.id,
73
+ teamId: teamId || null,
74
+ name: name || 'Untitled Key',
75
+ keyPrefix,
76
+ keyHash,
77
+ expiresAt: expiresAt ? new Date(expiresAt) : null,
78
+ },
79
+ include: { team: { select: { id: true, name: true } } },
80
+ });
81
+ // Return full key ONLY on creation (never again)
82
+ res.json({
83
+ key: {
84
+ id: apiKey.id,
85
+ name: apiKey.name,
86
+ keyPrefix,
87
+ teamId: apiKey.teamId,
88
+ teamName: apiKey.team?.name,
89
+ createdAt: apiKey.createdAt,
90
+ expiresAt: apiKey.expiresAt,
91
+ },
92
+ secret: rawKey, // Show once, user must copy
93
+ });
94
+ }
95
+ catch (error) {
96
+ next(error);
97
+ }
98
+ });
99
+ // Delete API key
100
+ apiKeysRouter.delete('/:keyId', requireAuth, async (req, res, next) => {
101
+ try {
102
+ const { keyId } = req.params;
103
+ const key = await db.apiKey.findFirst({
104
+ where: { id: keyId, userId: req.user.id },
105
+ });
106
+ if (!key) {
107
+ res.status(HTTP_STATUS.NOT_FOUND).json({ error: 'API key not found' });
108
+ return;
109
+ }
110
+ await db.apiKey.delete({ where: { id: keyId } });
111
+ res.json({ success: true });
112
+ }
113
+ catch (error) {
114
+ next(error);
115
+ }
116
+ });
117
+ // Check if user can access API keys (for UI)
118
+ apiKeysRouter.get('/access', requireAuth, async (req, res) => {
119
+ const canAccess = canAccessApiKeys(req.user);
120
+ res.json({ canAccess });
121
+ });
122
+ //# sourceMappingURL=api-keys.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"api-keys.js","sourceRoot":"","sources":["../../src/api/api-keys.ts"],"names":[],"mappings":"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"}