upfynai-code 2.9.1 → 2.9.3

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 (229) hide show
  1. package/README.md +91 -66
  2. package/client/dist/api-docs.html +838 -0
  3. package/client/dist/assets/AppContent-BXZDeSIC.js +545 -0
  4. package/client/dist/assets/CanvasFullScreen-mnpCnLZ9.js +1 -0
  5. package/client/dist/assets/CanvasWorkspace-4CqmjAVQ.js +163 -0
  6. package/client/dist/assets/DashboardPanel-zFIFlw56.js +1 -0
  7. package/client/dist/assets/FileTree-B0c_GaB3.js +1 -0
  8. package/client/dist/assets/GitPanel-DUP4zVU4.js +2 -0
  9. package/client/dist/assets/KaTeX_AMS-Regular-BQhdFMY1.woff2 +0 -0
  10. package/client/dist/assets/KaTeX_AMS-Regular-DMm9YOAa.woff +0 -0
  11. package/client/dist/assets/KaTeX_AMS-Regular-DRggAlZN.ttf +0 -0
  12. package/client/dist/assets/KaTeX_Caligraphic-Bold-ATXxdsX0.ttf +0 -0
  13. package/client/dist/assets/KaTeX_Caligraphic-Bold-BEiXGLvX.woff +0 -0
  14. package/client/dist/assets/KaTeX_Caligraphic-Bold-Dq_IR9rO.woff2 +0 -0
  15. package/client/dist/assets/KaTeX_Caligraphic-Regular-CTRA-rTL.woff +0 -0
  16. package/client/dist/assets/KaTeX_Caligraphic-Regular-Di6jR-x-.woff2 +0 -0
  17. package/client/dist/assets/KaTeX_Caligraphic-Regular-wX97UBjC.ttf +0 -0
  18. package/client/dist/assets/KaTeX_Fraktur-Bold-BdnERNNW.ttf +0 -0
  19. package/client/dist/assets/KaTeX_Fraktur-Bold-BsDP51OF.woff +0 -0
  20. package/client/dist/assets/KaTeX_Fraktur-Bold-CL6g_b3V.woff2 +0 -0
  21. package/client/dist/assets/KaTeX_Fraktur-Regular-CB_wures.ttf +0 -0
  22. package/client/dist/assets/KaTeX_Fraktur-Regular-CTYiF6lA.woff2 +0 -0
  23. package/client/dist/assets/KaTeX_Fraktur-Regular-Dxdc4cR9.woff +0 -0
  24. package/client/dist/assets/KaTeX_Main-Bold-Cx986IdX.woff2 +0 -0
  25. package/client/dist/assets/KaTeX_Main-Bold-Jm3AIy58.woff +0 -0
  26. package/client/dist/assets/KaTeX_Main-Bold-waoOVXN0.ttf +0 -0
  27. package/client/dist/assets/KaTeX_Main-BoldItalic-DxDJ3AOS.woff2 +0 -0
  28. package/client/dist/assets/KaTeX_Main-BoldItalic-DzxPMmG6.ttf +0 -0
  29. package/client/dist/assets/KaTeX_Main-BoldItalic-SpSLRI95.woff +0 -0
  30. package/client/dist/assets/KaTeX_Main-Italic-3WenGoN9.ttf +0 -0
  31. package/client/dist/assets/KaTeX_Main-Italic-BMLOBm91.woff +0 -0
  32. package/client/dist/assets/KaTeX_Main-Italic-NWA7e6Wa.woff2 +0 -0
  33. package/client/dist/assets/KaTeX_Main-Regular-B22Nviop.woff2 +0 -0
  34. package/client/dist/assets/KaTeX_Main-Regular-Dr94JaBh.woff +0 -0
  35. package/client/dist/assets/KaTeX_Main-Regular-ypZvNtVU.ttf +0 -0
  36. package/client/dist/assets/KaTeX_Math-BoldItalic-B3XSjfu4.ttf +0 -0
  37. package/client/dist/assets/KaTeX_Math-BoldItalic-CZnvNsCZ.woff2 +0 -0
  38. package/client/dist/assets/KaTeX_Math-BoldItalic-iY-2wyZ7.woff +0 -0
  39. package/client/dist/assets/KaTeX_Math-Italic-DA0__PXp.woff +0 -0
  40. package/client/dist/assets/KaTeX_Math-Italic-flOr_0UB.ttf +0 -0
  41. package/client/dist/assets/KaTeX_Math-Italic-t53AETM-.woff2 +0 -0
  42. package/client/dist/assets/KaTeX_SansSerif-Bold-CFMepnvq.ttf +0 -0
  43. package/client/dist/assets/KaTeX_SansSerif-Bold-D1sUS0GD.woff2 +0 -0
  44. package/client/dist/assets/KaTeX_SansSerif-Bold-DbIhKOiC.woff +0 -0
  45. package/client/dist/assets/KaTeX_SansSerif-Italic-C3H0VqGB.woff2 +0 -0
  46. package/client/dist/assets/KaTeX_SansSerif-Italic-DN2j7dab.woff +0 -0
  47. package/client/dist/assets/KaTeX_SansSerif-Italic-YYjJ1zSn.ttf +0 -0
  48. package/client/dist/assets/KaTeX_SansSerif-Regular-BNo7hRIc.ttf +0 -0
  49. package/client/dist/assets/KaTeX_SansSerif-Regular-CS6fqUqJ.woff +0 -0
  50. package/client/dist/assets/KaTeX_SansSerif-Regular-DDBCnlJ7.woff2 +0 -0
  51. package/client/dist/assets/KaTeX_Script-Regular-C5JkGWo-.ttf +0 -0
  52. package/client/dist/assets/KaTeX_Script-Regular-D3wIWfF6.woff2 +0 -0
  53. package/client/dist/assets/KaTeX_Script-Regular-D5yQViql.woff +0 -0
  54. package/client/dist/assets/KaTeX_Size1-Regular-C195tn64.woff +0 -0
  55. package/client/dist/assets/KaTeX_Size1-Regular-Dbsnue_I.ttf +0 -0
  56. package/client/dist/assets/KaTeX_Size1-Regular-mCD8mA8B.woff2 +0 -0
  57. package/client/dist/assets/KaTeX_Size2-Regular-B7gKUWhC.ttf +0 -0
  58. package/client/dist/assets/KaTeX_Size2-Regular-Dy4dx90m.woff2 +0 -0
  59. package/client/dist/assets/KaTeX_Size2-Regular-oD1tc_U0.woff +0 -0
  60. package/client/dist/assets/KaTeX_Size3-Regular-CTq5MqoE.woff +0 -0
  61. package/client/dist/assets/KaTeX_Size3-Regular-DgpXs0kz.ttf +0 -0
  62. package/client/dist/assets/KaTeX_Size4-Regular-BF-4gkZK.woff +0 -0
  63. package/client/dist/assets/KaTeX_Size4-Regular-DWFBv043.ttf +0 -0
  64. package/client/dist/assets/KaTeX_Size4-Regular-Dl5lxZxV.woff2 +0 -0
  65. package/client/dist/assets/KaTeX_Typewriter-Regular-C0xS9mPB.woff +0 -0
  66. package/client/dist/assets/KaTeX_Typewriter-Regular-CO6r4hn1.woff2 +0 -0
  67. package/client/dist/assets/KaTeX_Typewriter-Regular-D3Ib7_Hf.ttf +0 -0
  68. package/client/dist/assets/LoginModal-BRycfsyD.js +13 -0
  69. package/client/dist/assets/MarkdownPreview-DHmk3qzu.js +1 -0
  70. package/client/dist/assets/MermaidBlock-BuBc_G-F.js +2 -0
  71. package/client/dist/assets/Onboarding-BcnaZZ0o.js +1 -0
  72. package/client/dist/assets/PreviewPanel-CqCa92Tf.js +32 -0
  73. package/client/dist/assets/SetupForm-S0g6u5yT.js +1 -0
  74. package/client/dist/assets/WorkflowsPanel-CouH9JDO.js +1 -0
  75. package/client/dist/assets/index-BFuqS0tY.css +1 -0
  76. package/client/dist/assets/index-CNDcVl2g.js +68 -0
  77. package/client/dist/assets/pdf-CE_K4jFx.js +12 -0
  78. package/client/dist/assets/vendor-canvas-BZV40eAE.css +1 -0
  79. package/client/dist/assets/vendor-canvas-D39yWul6.js +49 -0
  80. package/client/dist/assets/vendor-codemirror-CbtmxxaB.js +35 -0
  81. package/client/dist/assets/vendor-diff-DNQpbhrT.js +69 -0
  82. package/client/dist/assets/vendor-i18n-DCFGyhQR.js +1 -0
  83. package/client/dist/assets/vendor-icons-BaD0x9SL.js +711 -0
  84. package/client/dist/assets/vendor-markdown-CimbIo6Y.js +296 -0
  85. package/client/dist/assets/vendor-mermaid-CH7SGc99.js +2556 -0
  86. package/client/dist/assets/vendor-react-96lCPsRK.js +67 -0
  87. package/client/dist/assets/vendor-syntax-DuHI9Ok6.js +16 -0
  88. package/client/dist/assets/vendor-xterm-CZq1hqo1.js +66 -0
  89. package/client/dist/assets/vendor-xterm-qxJ8_QYu.css +32 -0
  90. package/client/dist/clear-cache.html +85 -0
  91. package/client/dist/convert-icons.md +53 -0
  92. package/client/dist/favicon.png +0 -0
  93. package/client/dist/favicon.svg +5 -0
  94. package/client/dist/generate-icons.js +49 -0
  95. package/client/dist/icons/claude-ai-icon.svg +1 -0
  96. package/client/dist/icons/codex-white.svg +3 -0
  97. package/client/dist/icons/codex.svg +3 -0
  98. package/client/dist/icons/cursor-white.svg +12 -0
  99. package/client/dist/icons/cursor.svg +1 -0
  100. package/client/dist/icons/icon-128x128.png +0 -0
  101. package/client/dist/icons/icon-128x128.svg +5 -0
  102. package/client/dist/icons/icon-144x144.png +0 -0
  103. package/client/dist/icons/icon-144x144.svg +5 -0
  104. package/client/dist/icons/icon-152x152.png +0 -0
  105. package/client/dist/icons/icon-152x152.svg +5 -0
  106. package/client/dist/icons/icon-192x192.png +0 -0
  107. package/client/dist/icons/icon-192x192.svg +5 -0
  108. package/client/dist/icons/icon-384x384.png +0 -0
  109. package/client/dist/icons/icon-384x384.svg +5 -0
  110. package/client/dist/icons/icon-512x512.png +0 -0
  111. package/client/dist/icons/icon-512x512.svg +5 -0
  112. package/client/dist/icons/icon-72x72.png +0 -0
  113. package/client/dist/icons/icon-72x72.svg +5 -0
  114. package/client/dist/icons/icon-96x96.png +0 -0
  115. package/client/dist/icons/icon-96x96.svg +5 -0
  116. package/client/dist/icons/icon-template.svg +5 -0
  117. package/client/dist/index.html +119 -0
  118. package/client/dist/logo-128.png +0 -0
  119. package/client/dist/logo-256.png +0 -0
  120. package/client/dist/logo-32.png +0 -0
  121. package/client/dist/logo-512.png +0 -0
  122. package/client/dist/logo-64.png +0 -0
  123. package/client/dist/logo.svg +14 -0
  124. package/client/dist/manifest.json +61 -0
  125. package/client/dist/mcp-docs.html +108 -0
  126. package/client/dist/offline.html +84 -0
  127. package/client/dist/screenshots/cli-selection.png +0 -0
  128. package/client/dist/screenshots/desktop-main.png +0 -0
  129. package/client/dist/screenshots/mobile-chat.png +0 -0
  130. package/client/dist/screenshots/tools-modal.png +0 -0
  131. package/client/dist/sw.js +82 -0
  132. package/commands/upfynai-connect.md +59 -0
  133. package/commands/upfynai-disconnect.md +31 -0
  134. package/commands/upfynai-doctor.md +99 -0
  135. package/commands/upfynai-export.md +49 -0
  136. package/commands/upfynai-local.md +82 -0
  137. package/commands/upfynai-status.md +75 -0
  138. package/commands/upfynai-stop.md +49 -0
  139. package/commands/upfynai-uninstall.md +58 -0
  140. package/commands/upfynai.md +69 -0
  141. package/package.json +143 -82
  142. package/scripts/build-client.js +17 -0
  143. package/scripts/fix-node-pty.js +67 -0
  144. package/scripts/install-commands.js +78 -0
  145. package/server/agent-loop.js +242 -0
  146. package/server/auto-compact.js +99 -0
  147. package/server/claude-sdk.js +797 -0
  148. package/server/cli-ui.js +785 -0
  149. package/server/cli.js +596 -0
  150. package/server/constants/config.js +31 -0
  151. package/server/cursor-cli.js +270 -0
  152. package/server/database/auth.db +0 -0
  153. package/server/database/db.js +1391 -0
  154. package/server/database/init.sql +70 -0
  155. package/server/index.js +3799 -0
  156. package/server/load-env.js +26 -0
  157. package/server/mcp-server.js +621 -0
  158. package/server/middleware/auth.js +176 -0
  159. package/server/middleware/relayHelpers.js +44 -0
  160. package/server/middleware/sandboxRouter.js +174 -0
  161. package/server/openai-codex.js +403 -0
  162. package/server/openrouter.js +137 -0
  163. package/server/projects.js +1807 -0
  164. package/server/provider-factory.js +174 -0
  165. package/server/relay-client.js +379 -0
  166. package/server/routes/agent.js +1226 -0
  167. package/server/routes/auth.js +554 -0
  168. package/server/routes/canvas.js +53 -0
  169. package/server/routes/cli-auth.js +263 -0
  170. package/server/routes/codex.js +396 -0
  171. package/server/routes/commands.js +707 -0
  172. package/server/routes/composio.js +176 -0
  173. package/server/routes/cursor.js +770 -0
  174. package/server/routes/dashboard.js +295 -0
  175. package/server/routes/git.js +1208 -0
  176. package/server/routes/keys.js +34 -0
  177. package/server/routes/mcp-utils.js +48 -0
  178. package/server/routes/mcp.js +661 -0
  179. package/server/routes/payments.js +227 -0
  180. package/server/routes/projects.js +655 -0
  181. package/server/routes/sessions.js +146 -0
  182. package/server/routes/settings.js +261 -0
  183. package/server/routes/taskmaster.js +1928 -0
  184. package/server/routes/user.js +106 -0
  185. package/server/routes/vapi-chat.js +624 -0
  186. package/server/routes/voice.js +235 -0
  187. package/server/routes/webhooks.js +166 -0
  188. package/server/routes/workflows.js +312 -0
  189. package/server/sandbox.js +120 -0
  190. package/server/services/composio.js +204 -0
  191. package/server/services/sessionRegistry.js +139 -0
  192. package/server/services/whisperService.js +84 -0
  193. package/server/services/workflowScheduler.js +206 -0
  194. package/server/tests/relay-flow.test.js +570 -0
  195. package/server/tests/sessions.test.js +259 -0
  196. package/server/utils/commandParser.js +303 -0
  197. package/server/utils/email.js +61 -0
  198. package/server/utils/gitConfig.js +24 -0
  199. package/server/utils/mcp-detector.js +198 -0
  200. package/server/utils/taskmaster-websocket.js +129 -0
  201. package/shared/integrationCatalog.d.ts +12 -0
  202. package/shared/integrationCatalog.js +172 -0
  203. package/shared/modelConstants.js +96 -0
  204. package/bin/cli.js +0 -97
  205. package/dist/agents/claude.js +0 -229
  206. package/dist/agents/codex.js +0 -48
  207. package/dist/agents/cursor.js +0 -48
  208. package/dist/agents/detect.js +0 -51
  209. package/dist/agents/exec.js +0 -31
  210. package/dist/agents/files.js +0 -105
  211. package/dist/agents/git.js +0 -18
  212. package/dist/agents/gitagent.js +0 -67
  213. package/dist/agents/index.js +0 -88
  214. package/dist/agents/shell.js +0 -38
  215. package/dist/agents/utils.js +0 -136
  216. package/scripts/postinstall.js +0 -9
  217. package/scripts/prepublish.js +0 -58
  218. package/src/animation.js +0 -228
  219. package/src/auth.js +0 -122
  220. package/src/config.js +0 -40
  221. package/src/connect.js +0 -416
  222. package/src/launch.js +0 -78
  223. package/src/mcp.js +0 -57
  224. package/src/permissions.js +0 -140
  225. package/src/persistent-shell.js +0 -261
  226. package/src/server.js +0 -54
  227. /package/{dist → shared}/gitagent/index.js +0 -0
  228. /package/{dist → shared}/gitagent/parser.js +0 -0
  229. /package/{dist → shared}/gitagent/prompt-builder.js +0 -0
@@ -0,0 +1,295 @@
1
+ import { Router } from 'express';
2
+ import { getProjects, getSessions } from '../projects.js';
3
+ import { userDb, subscriptionDb, paymentDb, apiKeysDb, credentialsDb, relayTokensDb, webhookDb, workflowDb } from '../database/db.js';
4
+
5
+ const router = Router();
6
+
7
+ // AI provider credential types
8
+ const AI_PROVIDER_TYPES = ['anthropic_key', 'openai_key', 'openrouter_key', 'google_key'];
9
+ const PROVIDER_LABELS = {
10
+ anthropic_key: 'Anthropic',
11
+ openai_key: 'OpenAI',
12
+ openrouter_key: 'OpenRouter',
13
+ google_key: 'Google',
14
+ };
15
+
16
+ // Per-user stats cache (60s TTL)
17
+ const statsCache = new Map();
18
+ const CACHE_TTL = 60_000;
19
+
20
+ function getCachedStats(userId) {
21
+ const entry = statsCache.get(userId);
22
+ if (entry && Date.now() - entry.ts < CACHE_TTL) return entry.data;
23
+ return null;
24
+ }
25
+ function setCachedStats(userId, data) {
26
+ statsCache.set(userId, { data, ts: Date.now() });
27
+ }
28
+ function invalidateStatsCache(userId) {
29
+ statsCache.delete(userId);
30
+ }
31
+
32
+ // Periodic expiry of overdue subs (every 5 min, not per-request)
33
+ let lastExpireRun = 0;
34
+ async function maybeExpireOverdue() {
35
+ if (Date.now() - lastExpireRun < 300_000) return;
36
+ lastExpireRun = Date.now();
37
+ try { await subscriptionDb.expireOverdue(); } catch { /* non-critical */ }
38
+ }
39
+
40
+ /**
41
+ * GET /api/dashboard/user-stats — Per-user dashboard data from Turso
42
+ * Aggregates: account, subscription, payments, API keys, AI providers, relay tokens, webhooks, workflows
43
+ */
44
+ router.get('/user-stats', async (req, res) => {
45
+ try {
46
+ const userId = req.user.id;
47
+
48
+ // Return cached stats if fresh (< 60s)
49
+ const cached = getCachedStats(userId);
50
+ if (cached) return res.json(cached);
51
+
52
+ // Run all DB queries in parallel
53
+ const [
54
+ fullUser,
55
+ activeSub,
56
+ allSubs,
57
+ payments,
58
+ apiKeys,
59
+ credentials,
60
+ relayTokens,
61
+ webhooks,
62
+ workflows,
63
+ ] = await Promise.all([
64
+ userDb.getUserById(userId),
65
+ subscriptionDb.getActiveSub(userId).catch(() => null),
66
+ subscriptionDb.getAllSubs(userId).catch(() => []),
67
+ paymentDb.getUserPayments(userId).catch(() => []),
68
+ apiKeysDb.getApiKeys(userId).catch(() => []),
69
+ credentialsDb.getCredentials(userId).catch(() => []),
70
+ relayTokensDb.getTokens(userId).catch(() => []),
71
+ webhookDb.getAll(userId).catch(() => []),
72
+ workflowDb.getAll(userId).catch(() => []),
73
+ ]);
74
+
75
+ // Expire overdue subs periodically (not per-request)
76
+ maybeExpireOverdue();
77
+
78
+ // Account
79
+ const account = fullUser ? {
80
+ username: fullUser.username,
81
+ email: fullUser.email,
82
+ phone: fullUser.phone,
83
+ firstName: fullUser.first_name,
84
+ lastName: fullUser.last_name,
85
+ userCode: fullUser.user_code,
86
+ createdAt: fullUser.created_at,
87
+ lastLogin: fullUser.last_login,
88
+ accessOverride: fullUser.access_override,
89
+ } : null;
90
+
91
+ // Subscription
92
+ const subscription = {
93
+ active: activeSub ? {
94
+ id: activeSub.id,
95
+ planId: activeSub.plan_id,
96
+ status: activeSub.status,
97
+ startsAt: activeSub.starts_at,
98
+ expiresAt: activeSub.expires_at,
99
+ amount: activeSub.amount,
100
+ currency: activeSub.currency || 'INR',
101
+ } : null,
102
+ history: allSubs.map(s => ({
103
+ id: s.id,
104
+ planId: s.plan_id,
105
+ status: s.status,
106
+ startsAt: s.starts_at,
107
+ expiresAt: s.expires_at,
108
+ amount: s.amount,
109
+ currency: s.currency || 'INR',
110
+ createdAt: s.created_at,
111
+ })),
112
+ };
113
+
114
+ // Payments
115
+ const paidPayments = payments.filter(p => p.status === 'paid');
116
+ const totalAmountPaise = paidPayments.reduce((sum, p) => sum + (Number(p.amount) || 0), 0);
117
+ const paymentData = {
118
+ total: payments.length,
119
+ paid: paidPayments.length,
120
+ totalAmountPaise,
121
+ recent: payments.slice(0, 10).map(p => ({
122
+ id: p.id,
123
+ planId: p.plan_id,
124
+ amount: p.amount,
125
+ currency: p.currency || 'INR',
126
+ status: p.status,
127
+ createdAt: p.created_at,
128
+ })),
129
+ };
130
+
131
+ // API Keys (include full list for dashboard display)
132
+ const activeKeys = apiKeys.filter(k => k.is_active);
133
+ const apiKeyData = {
134
+ total: apiKeys.length,
135
+ active: activeKeys.length,
136
+ keys: apiKeys.map(k => ({
137
+ id: k.id,
138
+ keyName: k.key_name || 'default',
139
+ keyPrefix: k.api_key ? k.api_key.slice(0, 12) : '',
140
+ keySuffix: k.api_key ? k.api_key.slice(-6) : '',
141
+ fullKey: k.api_key,
142
+ isActive: !!k.is_active,
143
+ createdAt: k.created_at,
144
+ lastUsed: k.last_used,
145
+ })),
146
+ };
147
+
148
+ // AI Providers (filter credentials by AI provider types)
149
+ const aiCreds = credentials.filter(c => AI_PROVIDER_TYPES.includes(c.credential_type));
150
+ const activeAiCreds = aiCreds.filter(c => c.is_active);
151
+ const aiProviderData = {
152
+ total: aiCreds.length,
153
+ active: activeAiCreds.length,
154
+ providers: activeAiCreds.map(c => PROVIDER_LABELS[c.credential_type] || c.credential_type),
155
+ };
156
+
157
+ // Relay Tokens
158
+ const activeTokens = relayTokens.filter(t => t.is_active);
159
+ const lastConnected = relayTokens
160
+ .filter(t => t.last_connected)
161
+ .sort((a, b) => new Date(b.last_connected) - new Date(a.last_connected))[0]?.last_connected || null;
162
+ const relayData = {
163
+ total: relayTokens.length,
164
+ active: activeTokens.length,
165
+ lastConnected,
166
+ };
167
+
168
+ // Webhooks
169
+ const activeWebhooks = webhooks.filter(w => w.is_active);
170
+ const webhookData = {
171
+ total: webhooks.length,
172
+ active: activeWebhooks.length,
173
+ };
174
+
175
+ // Workflows
176
+ const activeWorkflows = workflows.filter(w => w.is_active);
177
+ const workflowData = {
178
+ total: workflows.length,
179
+ active: activeWorkflows.length,
180
+ };
181
+
182
+ const result = {
183
+ account,
184
+ subscription,
185
+ payments: paymentData,
186
+ apiKeys: apiKeyData,
187
+ aiProviders: aiProviderData,
188
+ relayTokens: relayData,
189
+ webhooks: webhookData,
190
+ workflows: workflowData,
191
+ };
192
+
193
+ setCachedStats(userId, result);
194
+ res.json(result);
195
+ } catch (error) {
196
+ res.status(500).json({ error: 'Failed to fetch user stats' });
197
+ }
198
+ });
199
+
200
+ /**
201
+ * GET /api/dashboard/stats — Dashboard usage analytics (local projects)
202
+ * Returns session counts, provider breakdown, and today's activity.
203
+ */
204
+ router.get('/stats', async (req, res) => {
205
+ try {
206
+ const userId = req.user?.id || req.user?.userId;
207
+ const projects = await getProjects(null, userId);
208
+ let totalSessions = 0;
209
+ let todaySessions = 0;
210
+ const providers = {};
211
+ const today = new Date().toISOString().slice(0, 10);
212
+
213
+ for (const project of projects) {
214
+ // Count sessions per provider
215
+ for (const provider of ['claude', 'cursor', 'codex']) {
216
+ try {
217
+ const sessions = await getSessions(project.name, provider);
218
+ if (sessions && sessions.length) {
219
+ totalSessions += sessions.length;
220
+ providers[provider] = (providers[provider] || 0) + sessions.length;
221
+
222
+ // Count today's sessions
223
+ for (const s of sessions) {
224
+ const created = s.created_at || s.createdAt || '';
225
+ if (created.startsWith(today)) {
226
+ todaySessions++;
227
+ }
228
+ }
229
+ }
230
+ } catch (e) {
231
+ // Provider not available for this project
232
+ }
233
+ }
234
+ }
235
+
236
+ res.json({
237
+ total: totalSessions,
238
+ today: todaySessions,
239
+ providers,
240
+ projectCount: projects.length,
241
+ });
242
+ } catch (error) {
243
+ res.status(500).json({ error: 'Failed to fetch dashboard stats' });
244
+ }
245
+ });
246
+
247
+ /**
248
+ * POST /api/dashboard/api-keys — Create a new API key
249
+ */
250
+ router.post('/api-keys', async (req, res) => {
251
+ try {
252
+ const userId = req.user.id;
253
+ const keyName = req.body.keyName || 'default';
254
+ const result = await apiKeysDb.createApiKey(userId, keyName);
255
+ invalidateStatsCache(userId);
256
+ res.json({ success: true, key: result });
257
+ } catch (error) {
258
+ res.status(500).json({ error: 'Failed to create API key' });
259
+ }
260
+ });
261
+
262
+ /**
263
+ * DELETE /api/dashboard/api-keys/:id — Delete a specific API key
264
+ */
265
+ router.delete('/api-keys/:id', async (req, res) => {
266
+ try {
267
+ const userId = req.user.id;
268
+ const keyId = Number(req.params.id);
269
+ const deleted = await apiKeysDb.deleteApiKey(userId, keyId);
270
+ if (!deleted) return res.status(404).json({ error: 'API key not found' });
271
+ invalidateStatsCache(userId);
272
+ res.json({ success: true });
273
+ } catch (error) {
274
+ res.status(500).json({ error: 'Failed to delete API key' });
275
+ }
276
+ });
277
+
278
+ /**
279
+ * PATCH /api/dashboard/api-keys/:id/toggle — Toggle API key active/inactive
280
+ */
281
+ router.patch('/api-keys/:id/toggle', async (req, res) => {
282
+ try {
283
+ const userId = req.user.id;
284
+ const keyId = Number(req.params.id);
285
+ const isActive = !!req.body.isActive;
286
+ const toggled = await apiKeysDb.toggleApiKey(userId, keyId, isActive);
287
+ invalidateStatsCache(userId);
288
+ if (!toggled) return res.status(404).json({ error: 'API key not found' });
289
+ res.json({ success: true });
290
+ } catch (error) {
291
+ res.status(500).json({ error: 'Failed to toggle API key' });
292
+ }
293
+ });
294
+
295
+ export default router;