@goscribe/server 1.3.4 → 1.5.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 (378) hide show
  1. package/.env.example +12 -0
  2. package/.vscode/settings.json +3 -0
  3. package/REFACTOR_NOTES.md +60 -0
  4. package/TESTING_PROMPT.md +225 -0
  5. package/dist/controllers/admin.controller.d.ts +715 -0
  6. package/dist/controllers/admin.controller.js +9 -0
  7. package/dist/controllers/annotations.controller.d.ts +439 -0
  8. package/dist/controllers/annotations.controller.js +9 -0
  9. package/dist/controllers/app-router.controller.d.ts +3011 -0
  10. package/dist/controllers/app-router.controller.js +38 -0
  11. package/dist/controllers/app-router.controller.test.d.ts +1 -0
  12. package/dist/controllers/app-router.controller.test.js +36 -0
  13. package/dist/controllers/auth.controller.d.ts +323 -0
  14. package/dist/controllers/auth.controller.js +9 -0
  15. package/dist/controllers/base.controller.d.ts +4 -0
  16. package/dist/controllers/base.controller.js +5 -0
  17. package/dist/controllers/chat.controller.d.ts +341 -0
  18. package/dist/controllers/chat.controller.js +9 -0
  19. package/dist/controllers/copilot.controller.d.ts +397 -0
  20. package/dist/controllers/copilot.controller.js +9 -0
  21. package/dist/controllers/flashcards.controller.d.ts +651 -0
  22. package/dist/controllers/flashcards.controller.js +9 -0
  23. package/dist/controllers/members.controller.d.ts +339 -0
  24. package/dist/controllers/members.controller.js +9 -0
  25. package/dist/controllers/notifications.controller.d.ts +199 -0
  26. package/dist/controllers/notifications.controller.js +9 -0
  27. package/dist/controllers/payment.controller.d.ts +181 -0
  28. package/dist/controllers/payment.controller.js +9 -0
  29. package/dist/controllers/podcast.controller.d.ts +575 -0
  30. package/dist/controllers/podcast.controller.js +9 -0
  31. package/dist/controllers/router-module.controller.d.ts +5 -0
  32. package/dist/controllers/router-module.controller.js +6 -0
  33. package/dist/controllers/studyguide.controller.d.ts +73 -0
  34. package/dist/controllers/studyguide.controller.js +9 -0
  35. package/dist/controllers/worksheets.controller.d.ts +829 -0
  36. package/dist/controllers/worksheets.controller.js +9 -0
  37. package/dist/controllers/workspace.controller.d.ts +1207 -0
  38. package/dist/controllers/workspace.controller.js +9 -0
  39. package/dist/lib/activity_human_description.test.js +16 -15
  40. package/dist/lib/activity_log_service.test.js +28 -23
  41. package/dist/lib/ai/config.d.ts +20 -0
  42. package/dist/lib/ai/config.js +31 -0
  43. package/dist/lib/ai/embedding-client.d.ts +8 -0
  44. package/dist/lib/ai/embedding-client.js +30 -0
  45. package/dist/lib/ai/index.d.ts +47 -0
  46. package/dist/lib/ai/index.js +28 -0
  47. package/dist/lib/ai/inference-backend/client.d.ts +28 -0
  48. package/dist/lib/ai/inference-backend/client.js +301 -0
  49. package/dist/lib/ai/inference-backend/mocks.d.ts +12 -0
  50. package/dist/lib/ai/inference-backend/mocks.js +133 -0
  51. package/dist/lib/ai/inference-backend/types.d.ts +44 -0
  52. package/dist/lib/ai/inference-backend/types.js +1 -0
  53. package/dist/lib/ai/json-parse.d.ts +2 -0
  54. package/dist/lib/ai/json-parse.js +34 -0
  55. package/dist/lib/ai/llm-client.d.ts +6 -0
  56. package/dist/lib/ai/llm-client.js +19 -0
  57. package/dist/lib/ai/mock.d.ts +2 -0
  58. package/dist/lib/ai/mock.js +10 -0
  59. package/dist/lib/ai/types.d.ts +9 -0
  60. package/dist/lib/ai/types.js +1 -0
  61. package/dist/lib/chunking.d.ts +19 -0
  62. package/dist/lib/chunking.js +47 -0
  63. package/dist/lib/curated-kb-seed.d.ts +12 -0
  64. package/dist/lib/curated-kb-seed.js +155 -0
  65. package/dist/lib/email.js +67 -108
  66. package/dist/lib/embeddings.d.ts +2 -0
  67. package/dist/lib/embeddings.js +1 -0
  68. package/dist/lib/ensure-curated-kb-catalog.d.ts +6 -0
  69. package/dist/lib/ensure-curated-kb-catalog.js +53 -0
  70. package/dist/lib/env.d.ts +1 -5
  71. package/dist/lib/env.js +2 -7
  72. package/dist/lib/inference.d.ts +1 -8
  73. package/dist/lib/inference.js +1 -19
  74. package/dist/lib/kb-meta.d.ts +8 -0
  75. package/dist/lib/kb-meta.js +77 -0
  76. package/dist/lib/note-text.d.ts +1 -0
  77. package/dist/lib/note-text.js +47 -0
  78. package/dist/lib/notification-service.test.js +37 -36
  79. package/dist/lib/pdf.d.ts +11 -0
  80. package/dist/lib/pdf.js +11 -0
  81. package/dist/lib/usage_service.d.ts +2 -1
  82. package/dist/lib/usage_service.js +30 -12
  83. package/dist/lib/worksheet-generation.js +4 -4
  84. package/dist/lib/worksheet-generation.test.js +32 -17
  85. package/dist/lib/workspace-kb.d.ts +5 -0
  86. package/dist/lib/workspace-kb.js +7 -0
  87. package/dist/models/controller-context.model.d.ts +8 -0
  88. package/dist/models/controller-context.model.js +1 -0
  89. package/dist/repositories/artifact.repository.d.ts +60 -0
  90. package/dist/repositories/artifact.repository.js +40 -0
  91. package/dist/repositories/base.repository.d.ts +14 -0
  92. package/dist/repositories/base.repository.js +14 -0
  93. package/dist/repositories/invitation.repository.d.ts +94 -0
  94. package/dist/repositories/invitation.repository.js +44 -0
  95. package/dist/repositories/notification.repository.d.ts +72 -0
  96. package/dist/repositories/notification.repository.js +44 -0
  97. package/dist/repositories/router-module.repository.d.ts +10 -0
  98. package/dist/repositories/router-module.repository.js +14 -0
  99. package/dist/repositories/user.repository.d.ts +74 -0
  100. package/dist/repositories/user.repository.js +37 -0
  101. package/dist/repositories/workspace-member.repository.d.ts +31 -0
  102. package/dist/repositories/workspace-member.repository.js +31 -0
  103. package/dist/repositories/workspace.repository.d.ts +97 -0
  104. package/dist/repositories/workspace.repository.js +79 -0
  105. package/dist/routers/_app.d.ts +528 -33
  106. package/dist/routers/_app.js +4 -0
  107. package/dist/routers/admin.d.ts +0 -4
  108. package/dist/routers/admin.js +21 -549
  109. package/dist/routers/annotations.js +12 -170
  110. package/dist/routers/artifactVersions.d.ts +65 -0
  111. package/dist/routers/artifactVersions.js +14 -0
  112. package/dist/routers/auth.d.ts +0 -6
  113. package/dist/routers/auth.js +36 -421
  114. package/dist/routers/chat.js +15 -229
  115. package/dist/routers/copilot.d.ts +14 -13
  116. package/dist/routers/copilot.js +13 -532
  117. package/dist/routers/flashcards.d.ts +5 -5
  118. package/dist/routers/flashcards.js +23 -349
  119. package/dist/routers/knowledgeBase.d.ts +421 -0
  120. package/dist/routers/knowledgeBase.js +118 -0
  121. package/dist/routers/members.d.ts +0 -41
  122. package/dist/routers/members.js +22 -710
  123. package/dist/routers/notes.d.ts +94 -0
  124. package/dist/routers/notes.js +37 -0
  125. package/dist/routers/notifications.js +7 -109
  126. package/dist/routers/payment.d.ts +3 -2
  127. package/dist/routers/payment.js +11 -393
  128. package/dist/routers/podcast.d.ts +1 -1
  129. package/dist/routers/podcast.js +11 -784
  130. package/dist/routers/studyguide.js +3 -129
  131. package/dist/routers/worksheets.d.ts +29 -14
  132. package/dist/routers/worksheets.js +49 -628
  133. package/dist/routers/workspace.d.ts +0 -4
  134. package/dist/routers/workspace.js +28 -922
  135. package/dist/scripts/purge-deleted-users.js +2 -2
  136. package/dist/server.js +10 -3
  137. package/dist/services/activity/activity-human-description.service.d.ts +13 -0
  138. package/dist/services/activity/activity-human-description.service.js +221 -0
  139. package/dist/services/activity/activity-human-description.service.test.d.ts +1 -0
  140. package/dist/services/activity/activity-human-description.service.test.js +16 -0
  141. package/dist/services/activity/activity-log.service.d.ts +87 -0
  142. package/dist/services/activity/activity-log.service.js +276 -0
  143. package/dist/services/activity/activity-log.service.test.d.ts +1 -0
  144. package/dist/services/activity/activity-log.service.test.js +27 -0
  145. package/dist/services/activity-human-description.service.d.ts +13 -0
  146. package/dist/services/activity-human-description.service.js +221 -0
  147. package/dist/services/activity-human-description.service.test.d.ts +1 -0
  148. package/dist/services/activity-human-description.service.test.js +16 -0
  149. package/dist/services/activity-log.service.d.ts +87 -0
  150. package/dist/services/activity-log.service.js +276 -0
  151. package/dist/services/activity-log.service.test.d.ts +1 -0
  152. package/dist/services/activity-log.service.test.js +27 -0
  153. package/dist/services/admin/admin.service.d.ts +270 -0
  154. package/dist/services/admin/admin.service.js +476 -0
  155. package/dist/services/admin.service.d.ts +270 -0
  156. package/dist/services/admin.service.js +476 -0
  157. package/dist/services/ai/ai-session.service.d.ts +5 -0
  158. package/dist/services/ai/ai-session.service.js +4 -0
  159. package/dist/services/ai-session.service.d.ts +60 -0
  160. package/dist/services/ai-session.service.js +561 -0
  161. package/dist/services/annotation.service.d.ts +177 -0
  162. package/dist/services/annotation.service.js +154 -0
  163. package/dist/services/artifact-notification.service.d.ts +14 -0
  164. package/dist/services/artifact-notification.service.js +20 -0
  165. package/dist/services/artifact-version.service.d.ts +38 -0
  166. package/dist/services/artifact-version.service.js +129 -0
  167. package/dist/services/artifacts/annotation.service.d.ts +177 -0
  168. package/dist/services/artifacts/annotation.service.js +154 -0
  169. package/dist/services/artifacts/artifact-version.service.d.ts +38 -0
  170. package/dist/services/artifacts/artifact-version.service.js +129 -0
  171. package/dist/services/artifacts/chat.service.d.ts +127 -0
  172. package/dist/services/artifacts/chat.service.js +182 -0
  173. package/dist/services/artifacts/study-guide.service.d.ts +18 -0
  174. package/dist/services/artifacts/study-guide.service.js +65 -0
  175. package/dist/services/auth/auth.service.d.ts +94 -0
  176. package/dist/services/auth/auth.service.js +368 -0
  177. package/dist/services/auth.service.d.ts +94 -0
  178. package/dist/services/auth.service.js +368 -0
  179. package/dist/services/base.service.d.ts +14 -0
  180. package/dist/services/base.service.js +14 -0
  181. package/dist/services/billing/payment.service.d.ts +55 -0
  182. package/dist/services/billing/payment.service.js +368 -0
  183. package/dist/services/billing/subscription.service.d.ts +37 -0
  184. package/dist/services/billing/subscription.service.js +654 -0
  185. package/dist/services/billing/usage.service.d.ts +27 -0
  186. package/dist/services/billing/usage.service.js +77 -0
  187. package/dist/services/chat.service.d.ts +127 -0
  188. package/dist/services/chat.service.js +182 -0
  189. package/dist/services/content/copilot.service.d.ts +113 -0
  190. package/dist/services/content/copilot.service.js +453 -0
  191. package/dist/services/content/flashcard-progress.service.d.ts +159 -0
  192. package/dist/services/content/flashcard-progress.service.js +432 -0
  193. package/dist/services/content/flashcard.service.d.ts +140 -0
  194. package/dist/services/content/flashcard.service.js +326 -0
  195. package/dist/services/content/media-analysis.service.d.ts +23 -0
  196. package/dist/services/content/media-analysis.service.js +404 -0
  197. package/dist/services/content/podcast.service.d.ts +267 -0
  198. package/dist/services/content/podcast.service.js +653 -0
  199. package/dist/services/content/worksheet-content.service.d.ts +37 -0
  200. package/dist/services/content/worksheet-content.service.js +84 -0
  201. package/dist/services/content/worksheet-content.service.test.d.ts +1 -0
  202. package/dist/services/content/worksheet-content.service.test.js +69 -0
  203. package/dist/services/content/worksheet-generation.service.d.ts +91 -0
  204. package/dist/services/content/worksheet-generation.service.js +95 -0
  205. package/dist/services/content/worksheet-generation.service.test.d.ts +1 -0
  206. package/dist/services/content/worksheet-generation.service.test.js +20 -0
  207. package/dist/services/content/worksheet.service.d.ts +347 -0
  208. package/dist/services/content/worksheet.service.js +599 -0
  209. package/dist/services/copilot.service.d.ts +116 -0
  210. package/dist/services/copilot.service.js +447 -0
  211. package/dist/services/flashcard-progress.service.d.ts +2 -2
  212. package/dist/services/flashcard-progress.service.js +3 -2
  213. package/dist/services/flashcard.service.d.ts +140 -0
  214. package/dist/services/flashcard.service.js +325 -0
  215. package/dist/services/invitation.service.d.ts +66 -0
  216. package/dist/services/invitation.service.js +348 -0
  217. package/dist/services/knowledge/knowledge-base.service.d.ts +316 -0
  218. package/dist/services/knowledge/knowledge-base.service.js +544 -0
  219. package/dist/services/knowledge-base.service.d.ts +316 -0
  220. package/dist/services/knowledge-base.service.js +536 -0
  221. package/dist/services/media-analysis.service.d.ts +23 -0
  222. package/dist/services/media-analysis.service.js +384 -0
  223. package/dist/services/member.service.d.ts +36 -0
  224. package/dist/services/member.service.js +193 -0
  225. package/dist/services/members/invitation.service.d.ts +66 -0
  226. package/dist/services/members/invitation.service.js +348 -0
  227. package/dist/services/members/member.service.d.ts +36 -0
  228. package/dist/services/members/member.service.js +193 -0
  229. package/dist/services/note.service.d.ts +55 -0
  230. package/dist/services/note.service.js +111 -0
  231. package/dist/services/notification.service.d.ts +214 -0
  232. package/dist/services/notification.service.js +550 -0
  233. package/dist/services/notification.service.test.d.ts +1 -0
  234. package/dist/services/notification.service.test.js +87 -0
  235. package/dist/services/notifications/notification.service.d.ts +214 -0
  236. package/dist/services/notifications/notification.service.js +550 -0
  237. package/dist/services/notifications/notification.service.test.d.ts +1 -0
  238. package/dist/services/notifications/notification.service.test.js +87 -0
  239. package/dist/services/payment.service.d.ts +55 -0
  240. package/dist/services/payment.service.js +368 -0
  241. package/dist/services/podcast.service.d.ts +267 -0
  242. package/dist/services/podcast.service.js +654 -0
  243. package/dist/services/router-module.service.d.ts +7 -0
  244. package/dist/services/router-module.service.js +10 -0
  245. package/dist/services/study-guide.service.d.ts +18 -0
  246. package/dist/services/study-guide.service.js +65 -0
  247. package/dist/services/subscription.service.d.ts +37 -0
  248. package/dist/services/subscription.service.js +654 -0
  249. package/dist/services/usage-limit-policy.service.d.ts +12 -0
  250. package/dist/services/usage-limit-policy.service.js +22 -0
  251. package/dist/services/usage-limit-policy.service.test.d.ts +1 -0
  252. package/dist/services/usage-limit-policy.service.test.js +46 -0
  253. package/dist/services/usage.service.d.ts +27 -0
  254. package/dist/services/usage.service.js +77 -0
  255. package/dist/services/worksheet-content.service.d.ts +42 -0
  256. package/dist/services/worksheet-content.service.js +84 -0
  257. package/dist/services/worksheet-content.service.test.d.ts +1 -0
  258. package/dist/services/worksheet-content.service.test.js +69 -0
  259. package/dist/services/worksheet-generation.service.d.ts +91 -0
  260. package/dist/services/worksheet-generation.service.js +95 -0
  261. package/dist/services/worksheet-generation.service.test.d.ts +1 -0
  262. package/dist/services/worksheet-generation.service.test.js +20 -0
  263. package/dist/services/worksheet.service.d.ts +385 -0
  264. package/dist/services/worksheet.service.js +596 -0
  265. package/dist/services/workspace/workspace-analytics.service.d.ts +24 -0
  266. package/dist/services/workspace/workspace-analytics.service.js +95 -0
  267. package/dist/services/workspace/workspace-kb.service.d.ts +40 -0
  268. package/dist/services/workspace/workspace-kb.service.js +184 -0
  269. package/dist/services/workspace/workspace.service.d.ts +307 -0
  270. package/dist/services/workspace/workspace.service.js +394 -0
  271. package/dist/services/workspace-analytics.service.d.ts +24 -0
  272. package/dist/services/workspace-analytics.service.js +95 -0
  273. package/dist/services/workspace-kb.service.d.ts +40 -0
  274. package/dist/services/workspace-kb.service.js +184 -0
  275. package/dist/services/workspace-progress.service.d.ts +27 -0
  276. package/dist/services/workspace-progress.service.js +56 -0
  277. package/dist/services/workspace-progress.service.test.d.ts +1 -0
  278. package/dist/services/workspace-progress.service.test.js +49 -0
  279. package/dist/services/workspace.service.d.ts +307 -0
  280. package/dist/services/workspace.service.js +390 -0
  281. package/dist/trpc.js +2 -2
  282. package/package.json +5 -6
  283. package/prisma/migrations/20260509000001_add_knowledge_base/migration.sql +99 -0
  284. package/prisma/migrations/20260509000002_curate_knowledge_base/migration.sql +52 -0
  285. package/prisma/migrations/20260522000000_add_notes/migration.sql +27 -0
  286. package/prisma/migrations/20260524000000_remove_notes/migration.sql +3 -0
  287. package/prisma/schema.prisma +150 -48
  288. package/prisma/seed.mjs +67 -0
  289. package/scripts/debug/README.md +4 -0
  290. package/src/README.md +63 -0
  291. package/src/lib/ai/config.ts +34 -0
  292. package/src/lib/ai/embedding-client.ts +47 -0
  293. package/src/lib/ai/index.ts +62 -0
  294. package/src/lib/ai/inference-backend/client.ts +479 -0
  295. package/src/lib/ai/inference-backend/mocks.ts +171 -0
  296. package/src/lib/ai/inference-backend/types.ts +50 -0
  297. package/src/lib/ai/json-parse.ts +35 -0
  298. package/src/lib/ai/llm-client.ts +31 -0
  299. package/src/lib/ai/mock.ts +12 -0
  300. package/src/lib/ai/types.ts +11 -0
  301. package/src/lib/chunking.ts +81 -0
  302. package/src/lib/curated-kb-seed.ts +164 -0
  303. package/src/lib/email.ts +77 -115
  304. package/src/lib/embeddings.ts +9 -0
  305. package/src/lib/ensure-curated-kb-catalog.ts +60 -0
  306. package/src/lib/env.ts +2 -7
  307. package/src/lib/inference.ts +1 -21
  308. package/src/lib/kb-meta.ts +81 -0
  309. package/src/lib/pdf.ts +23 -0
  310. package/src/lib/workspace-kb.ts +7 -0
  311. package/src/repositories/artifact.repository.ts +55 -0
  312. package/src/repositories/base.repository.ts +19 -0
  313. package/src/repositories/invitation.repository.ts +53 -0
  314. package/src/repositories/notification.repository.ts +53 -0
  315. package/src/repositories/user.repository.ts +44 -0
  316. package/src/repositories/workspace-member.repository.ts +38 -0
  317. package/src/repositories/workspace.repository.ts +89 -0
  318. package/src/routers/_app.ts +4 -0
  319. package/src/routers/admin.ts +124 -692
  320. package/src/routers/annotations.ts +25 -203
  321. package/src/routers/artifactVersions.ts +32 -0
  322. package/src/routers/auth.ts +81 -519
  323. package/src/routers/chat.ts +42 -245
  324. package/src/routers/copilot.ts +41 -666
  325. package/src/routers/flashcards.ts +108 -404
  326. package/src/routers/knowledgeBase.ts +216 -0
  327. package/src/routers/members.ts +60 -782
  328. package/src/routers/notifications.ts +15 -117
  329. package/src/routers/payment.ts +37 -446
  330. package/src/routers/podcast.ts +36 -898
  331. package/src/routers/studyguide.ts +5 -144
  332. package/src/routers/worksheets.ts +171 -735
  333. package/src/routers/workspace.ts +138 -1109
  334. package/src/scripts/purge-deleted-users.ts +2 -2
  335. package/src/server.ts +10 -3
  336. package/src/{lib/activity_human_description.test.ts → services/activity/activity-human-description.service.test.ts} +1 -1
  337. package/src/{lib/activity_log_service.test.ts → services/activity/activity-log.service.test.ts} +1 -1
  338. package/src/{lib/activity_log_service.ts → services/activity/activity-log.service.ts} +2 -2
  339. package/src/services/admin/admin.service.ts +612 -0
  340. package/src/services/ai/ai-session.service.ts +5 -0
  341. package/src/services/artifacts/annotation.service.ts +189 -0
  342. package/src/services/artifacts/artifact-version.service.ts +151 -0
  343. package/src/services/artifacts/chat.service.ts +197 -0
  344. package/src/services/artifacts/study-guide.service.ts +72 -0
  345. package/src/services/auth/auth.service.ts +473 -0
  346. package/src/services/base.service.ts +19 -0
  347. package/src/services/billing/payment.service.ts +436 -0
  348. package/src/{lib/subscription_service.ts → services/billing/subscription.service.ts} +5 -5
  349. package/src/{lib/usage_service.ts → services/billing/usage.service.ts} +32 -12
  350. package/src/services/content/copilot.service.ts +596 -0
  351. package/src/services/{flashcard-progress.service.ts → content/flashcard-progress.service.ts} +6 -3
  352. package/src/services/content/flashcard.service.ts +394 -0
  353. package/src/services/content/media-analysis.service.ts +556 -0
  354. package/src/services/content/podcast.service.ts +777 -0
  355. package/src/services/content/worksheet-content.service.test.ts +83 -0
  356. package/src/services/content/worksheet-content.service.ts +117 -0
  357. package/src/{lib/worksheet-generation.test.ts → services/content/worksheet-generation.service.test.ts} +1 -1
  358. package/src/services/content/worksheet.service.ts +751 -0
  359. package/src/services/knowledge/knowledge-base.service.ts +705 -0
  360. package/src/services/members/invitation.service.ts +427 -0
  361. package/src/services/members/member.service.ts +241 -0
  362. package/src/{lib/notification-service.test.ts → services/notifications/notification.service.test.ts} +2 -2
  363. package/src/{lib/notification-service.ts → services/notifications/notification.service.ts} +102 -1
  364. package/src/services/workspace/workspace-analytics.service.ts +107 -0
  365. package/src/services/workspace/workspace-kb.service.ts +273 -0
  366. package/src/services/workspace/workspace.service.ts +481 -0
  367. package/src/trpc.ts +2 -2
  368. package/src/lib/ai-session.ts +0 -704
  369. package/src/lib/workspace-access.ts +0 -13
  370. /package/{check-difficulty.cjs → scripts/debug/check-difficulty.cjs} +0 -0
  371. /package/{check-questions.cjs → scripts/debug/check-questions.cjs} +0 -0
  372. /package/{db-summary.cjs → scripts/debug/db-summary.cjs} +0 -0
  373. /package/{mcq-test.cjs → scripts/debug/mcq-test.cjs} +0 -0
  374. /package/{test-generate.js → scripts/debug/test-generate.js} +0 -0
  375. /package/{test-ratio.cjs → scripts/debug/test-ratio.cjs} +0 -0
  376. /package/{zod-test.cjs → scripts/debug/zod-test.cjs} +0 -0
  377. /package/src/{lib/activity_human_description.ts → services/activity/activity-human-description.service.ts} +0 -0
  378. /package/src/{lib/worksheet-generation.ts → services/content/worksheet-generation.service.ts} +0 -0
@@ -1,710 +1,142 @@
1
1
  import { z } from 'zod';
2
2
  import { router, adminProcedure } from '../trpc.js';
3
- import { logger } from '../lib/logger.js';
4
- import { stripe } from '../lib/stripe.js';
5
3
  import { ActivityLogCategory, ActivityLogStatus, ArtifactType, InvoiceType } from '@prisma/client';
6
- import {
7
- buildActivityLogWhere,
8
- deleteActivityLogsOlderThan,
9
- getActivityRetentionDays,
10
- } from '../lib/activity_log_service.js';
11
- import { getActivityHumanDescription } from '../lib/activity_human_description.js';
12
-
13
- function csvEscapeCell(value: string | number | null | undefined): string {
14
- if (value === null || value === undefined) return '';
15
- const s = String(value);
16
- if (/[",\n\r]/.test(s)) return `"${s.replace(/"/g, '""')}"`;
17
- return s;
18
- }
4
+ import { AdminService } from '../services/admin/admin.service.js';
19
5
 
20
6
  const activityLogFiltersInput = z.object({
21
- from: z.coerce.date().optional(),
22
- to: z.coerce.date().optional(),
23
- actorUserId: z.string().optional(),
24
- workspaceId: z.string().optional(),
25
- category: z.nativeEnum(ActivityLogCategory).optional(),
26
- status: z.nativeEnum(ActivityLogStatus).optional(),
27
- search: z.string().max(200).optional(),
7
+ from: z.coerce.date().optional(),
8
+ to: z.coerce.date().optional(),
9
+ actorUserId: z.string().optional(),
10
+ workspaceId: z.string().optional(),
11
+ category: z.nativeEnum(ActivityLogCategory).optional(),
12
+ status: z.nativeEnum(ActivityLogStatus).optional(),
13
+ search: z.string().max(200).optional(),
28
14
  });
29
15
 
30
16
  const activityListInput = activityLogFiltersInput.extend({
31
- page: z.number().int().min(1).default(1),
32
- limit: z.number().min(1).max(100).default(20),
17
+ page: z.number().int().min(1).default(1),
18
+ limit: z.number().min(1).max(100).default(20),
33
19
  });
34
20
 
35
21
  const listUsersInput = z.object({
36
- page: z.number().int().min(1).default(1),
37
- pageSize: z.number().int().min(1).max(100).default(10),
38
- /** Trims; matches name, email, or id (substring). */
39
- search: z.string().max(200).optional(),
40
- emailVerified: z.enum(['all', 'yes', 'no']).default('all'),
41
- /** Filter by account creation time (joined date). */
42
- joinedFrom: z.coerce.date().optional(),
43
- joinedTo: z.coerce.date().optional(),
22
+ page: z.number().int().min(1).default(1),
23
+ pageSize: z.number().int().min(1).max(100).default(10),
24
+ search: z.string().max(200).optional(),
25
+ emailVerified: z.enum(['all', 'yes', 'no']).default('all'),
26
+ joinedFrom: z.coerce.date().optional(),
27
+ joinedTo: z.coerce.date().optional(),
44
28
  });
45
29
 
46
30
  const listInvoicesInput = z.object({
47
- page: z.number().int().min(1).default(1),
48
- pageSize: z.number().int().min(1).max(100).default(10),
49
- search: z.string().max(200).optional(),
50
- status: z.string().max(32).optional(),
51
- type: z.nativeEnum(InvoiceType).optional(),
52
- userId: z.string().optional(),
53
- from: z.coerce.date().optional(),
54
- to: z.coerce.date().optional(),
31
+ page: z.number().int().min(1).default(1),
32
+ pageSize: z.number().int().min(1).max(100).default(10),
33
+ search: z.string().max(200).optional(),
34
+ status: z.string().max(32).optional(),
35
+ type: z.nativeEnum(InvoiceType).optional(),
36
+ userId: z.string().optional(),
37
+ from: z.coerce.date().optional(),
38
+ to: z.coerce.date().optional(),
55
39
  });
56
40
 
57
41
  export const admin = router({
58
- getSystemStats: adminProcedure
59
- .query(async ({ ctx }) => {
60
- const totalUsers = await ctx.db.user.count();
61
- const totalWorkspaces = await ctx.db.workspace.count();
62
- const totalSubscriptions = await ctx.db.subscription.count({
63
- where: { status: 'active' }
64
- });
65
-
66
- // Calculate revenue breakdown
67
- const subRevenueResult = await (ctx.db as any).invoice.aggregate({
68
- where: { status: 'paid', type: 'SUBSCRIPTION' },
69
- _sum: { amountPaid: true }
70
- });
71
- const topupRevenueResult = await (ctx.db as any).invoice.aggregate({
72
- where: { status: 'paid', type: 'TOPUP' },
73
- _sum: { amountPaid: true }
74
- });
75
-
76
- const subRevenue = Number(subRevenueResult._sum.amountPaid || 0) / 100;
77
- const topupRevenue = Number(topupRevenueResult._sum.amountPaid || 0) / 100;
78
-
79
- return {
80
- totalUsers,
81
- totalWorkspaces,
82
- totalSubscriptions,
83
- revenue: subRevenue + topupRevenue,
84
- subscriptionRevenue: subRevenue,
85
- topupRevenue: topupRevenue,
86
- };
87
- }),
88
-
89
- listUsers: adminProcedure
90
- .input(listUsersInput)
91
- .query(async ({ ctx, input }) => {
92
- const search = input.search?.trim();
93
- const baseRoleWhere = {
94
- role: {
95
- name: { not: 'System Admin' },
96
- },
97
- } as const;
98
-
99
- const searchWhere =
100
- search && search.length > 0
101
- ? {
102
- OR: [
103
- { email: { contains: search, mode: 'insensitive' as const } },
104
- { name: { contains: search, mode: 'insensitive' as const } },
105
- { id: { contains: search } },
106
- ],
107
- }
108
- : undefined;
109
-
110
- const verifiedWhere =
111
- input.emailVerified === 'yes'
112
- ? { emailVerified: { not: null } }
113
- : input.emailVerified === 'no'
114
- ? { emailVerified: null }
115
- : undefined;
116
-
117
- const joinedWhere =
118
- input.joinedFrom || input.joinedTo
119
- ? {
120
- createdAt: {
121
- ...(input.joinedFrom ? { gte: input.joinedFrom } : {}),
122
- ...(input.joinedTo ? { lte: input.joinedTo } : {}),
123
- },
124
- }
125
- : undefined;
126
-
127
- const where = {
128
- AND: [
129
- baseRoleWhere,
130
- ...(searchWhere ? [searchWhere] : []),
131
- ...(verifiedWhere ? [verifiedWhere] : []),
132
- ...(joinedWhere ? [joinedWhere] : []),
133
- ],
134
- };
135
-
136
- const totalCount = await ctx.db.user.count({ where });
137
- const pageSize = input.pageSize;
138
- const totalPages = Math.max(1, Math.ceil(totalCount / pageSize));
139
- const page = Math.min(Math.max(1, input.page), totalPages);
140
- const skip = (page - 1) * pageSize;
141
-
142
- const users = await ctx.db.user.findMany({
143
- where,
144
- skip,
145
- take: pageSize,
146
- orderBy: { createdAt: 'desc' },
147
- include: {
148
- role: true,
149
- profilePicture: true,
150
- subscriptions: {
151
- orderBy: { createdAt: 'desc' },
152
- take: 1,
153
- include: {
154
- plan: true,
155
- },
156
- },
157
- },
158
- });
159
-
160
- return {
161
- users: users.map((user: any) => ({
162
- ...user,
163
- profilePicture: user.profilePicture?.objectKey
164
- ? `/profile-picture/${user.profilePicture.objectKey}?t=${new Date(user.updatedAt).getTime()}`
165
- : null,
166
- })),
167
- page,
168
- pageSize,
169
- totalCount,
170
- totalPages,
171
- };
172
- }),
173
-
174
- /**
175
- * Paginated global invoice list for admin Invoices page (search + filters).
176
- */
177
- listInvoices: adminProcedure
178
- .input(listInvoicesInput)
179
- .query(async ({ ctx, input }) => {
180
- const search = input.search?.trim();
181
-
182
- const searchWhere =
183
- search && search.length > 0
184
- ? {
185
- OR: [
186
- { id: { contains: search } },
187
- { stripeInvoiceId: { contains: search, mode: 'insensitive' as const } },
188
- {
189
- user: {
190
- email: { contains: search, mode: 'insensitive' as const },
191
- },
192
- },
193
- {
194
- user: {
195
- name: { contains: search, mode: 'insensitive' as const },
196
- },
197
- },
198
- ],
199
- }
200
- : undefined;
201
-
202
- const userIdWhere = input.userId?.trim()
203
- ? { userId: input.userId.trim() }
204
- : undefined;
205
-
206
- const statusWhere = input.status?.trim()
207
- ? { status: input.status.trim() }
208
- : undefined;
209
-
210
- const typeWhere = input.type ? { type: input.type } : undefined;
211
-
212
- const dateWhere =
213
- input.from || input.to
214
- ? {
215
- createdAt: {
216
- ...(input.from ? { gte: input.from } : {}),
217
- ...(input.to ? { lte: input.to } : {}),
218
- },
219
- }
220
- : undefined;
221
-
222
- const andParts = [
223
- ...(searchWhere ? [searchWhere] : []),
224
- ...(userIdWhere ? [userIdWhere] : []),
225
- ...(statusWhere ? [statusWhere] : []),
226
- ...(typeWhere ? [typeWhere] : []),
227
- ...(dateWhere ? [dateWhere] : []),
228
- ];
229
- const where = andParts.length > 0 ? { AND: andParts } : {};
230
-
231
- const totalCount = await (ctx.db as any).invoice.count({ where });
232
- const pageSize = input.pageSize;
233
- const totalPages = Math.max(1, Math.ceil(totalCount / pageSize));
234
- const page = Math.min(Math.max(1, input.page), totalPages);
235
- const skip = (page - 1) * pageSize;
236
-
237
- const invoices = await (ctx.db as any).invoice.findMany({
238
- where,
239
- skip,
240
- take: pageSize,
241
- orderBy: { createdAt: 'desc' },
242
- include: {
243
- user: {
244
- include: {
245
- profilePicture: true,
246
- },
247
- },
248
- subscription: {
249
- include: { plan: true },
250
- },
251
- },
252
- });
253
-
254
- const items = invoices.map((invoice: any) => ({
255
- ...invoice,
256
- user: invoice.user
257
- ? {
258
- ...invoice.user,
259
- profilePicture: invoice.user.profilePicture?.objectKey
260
- ? `/profile-picture/${invoice.user.profilePicture.objectKey}?t=${new Date(invoice.user.updatedAt).getTime()}`
261
- : null,
262
- }
263
- : invoice.user,
264
- }));
265
-
266
- return {
267
- items,
268
- page,
269
- pageSize,
270
- totalCount,
271
- totalPages,
272
- };
273
- }),
274
-
275
- listWorkspaces: adminProcedure
276
- .input(z.object({
277
- limit: z.number().min(1).max(100).default(50),
278
- cursor: z.string().nullish(),
279
- }))
280
- .query(async ({ ctx, input }) => {
281
- const workspaces = await ctx.db.workspace.findMany({
282
- take: input.limit + 1,
283
- cursor: input.cursor ? { id: input.cursor } : undefined,
284
- orderBy: { createdAt: 'desc' },
285
- include: {
286
- owner: {
287
- select: {
288
- name: true,
289
- email: true,
290
- }
291
- }
292
- }
293
- });
294
-
295
- let nextCursor: typeof input.cursor | undefined = undefined;
296
- if (workspaces.length > input.limit) {
297
- const nextItem = workspaces.pop();
298
- nextCursor = nextItem!.id;
299
- }
300
-
301
- return {
302
- workspaces,
303
- nextCursor,
304
- };
305
- }),
306
-
307
- updateUserRole: adminProcedure
308
- .input(z.object({
309
- userId: z.string(),
310
- roleName: z.string(),
311
- }))
312
- .mutation(async ({ ctx, input }) => {
313
- const role = await ctx.db.role.findUnique({
314
- where: { name: input.roleName }
315
- });
316
-
317
- if (!role) {
318
- throw new Error(`Role ${input.roleName} not found`);
319
- }
320
-
321
- const updatedUser = await ctx.db.user.update({
322
- where: { id: input.userId },
323
- data: { roleId: role.id },
324
- });
325
-
326
- logger.info(`User ${input.userId} role updated to ${input.roleName} by admin ${ctx.userId}`, 'ADMIN');
327
- return updatedUser;
328
- }),
329
-
330
- listPlans: adminProcedure
331
- .query(async ({ ctx }) => {
332
- return ctx.db.plan.findMany({
333
- include: {
334
- limit: true
335
- },
336
- orderBy: { price: 'asc' }
337
- });
338
- }),
339
-
340
- upsertPlan: adminProcedure
341
- .input(z.object({
342
- id: z.string().optional(),
343
- name: z.string(),
344
- description: z.string().optional(),
345
- type: z.string(), // subscription or credits
346
- price: z.number(),
347
- stripePriceId: z.string().optional(),
348
- interval: z.string().nullable(),
349
- active: z.boolean().default(true),
350
- limits: z.object({
351
- maxStorageBytes: z.number(),
352
- maxWorksheets: z.number(),
353
- maxFlashcards: z.number(),
354
- maxPodcasts: z.number().default(0),
355
- maxStudyGuides: z.number().default(0),
356
- })
357
- }))
358
- .mutation(async ({ ctx, input }) => {
359
- let { limits, id, stripePriceId, ...planData } = input;
360
-
361
- // Automation: Create Stripe Product and Price if stripePriceId is missing (new plan)
362
- if (!stripePriceId && !id) {
363
- if (!stripe) {
364
- throw new Error("Stripe is not configured on the server");
365
- }
366
-
367
- try {
368
- // 1. Create Product
369
- const product = await stripe.products.create({
370
- name: planData.name,
371
- description: planData.description,
372
- });
373
-
374
- // 2. Create Price
375
- const price = await stripe.prices.create({
376
- product: product.id,
377
- unit_amount: Math.round(planData.price * 100), // convert to cents
378
- currency: 'usd',
379
- recurring: planData.type === 'subscription'
380
- ? { interval: (planData.interval as any) || 'month' }
381
- : undefined,
382
- });
383
-
384
- stripePriceId = price.id;
385
- logger.info(`Automatically created Stripe Product (${product.id}) and Price (${price.id}) for plan: ${planData.name}`, 'STRIPE');
386
- } catch (err: any) {
387
- logger.error(`Failed to automate Stripe creation: ${err.message}`, 'STRIPE');
388
- throw new Error(`Stripe error: ${err.message}`);
389
- }
390
- }
391
-
392
- if (id) {
393
- // Update
394
- const updatedPlan = await ctx.db.plan.update({
395
- where: { id },
396
- data: {
397
- ...planData,
398
- stripePriceId,
399
- limit: {
400
- update: {
401
- maxStorageBytes: BigInt(limits.maxStorageBytes),
402
- maxWorksheets: limits.maxWorksheets,
403
- maxFlashcards: limits.maxFlashcards,
404
- maxPodcasts: limits.maxPodcasts,
405
- maxStudyGuides: limits.maxStudyGuides,
406
- }
407
- }
408
- }
409
- });
410
- logger.info(`Plan ${id} updated by admin ${ctx.userId}`, 'ADMIN');
411
- return updatedPlan;
412
- } else {
413
- // Create
414
- const newPlan = await ctx.db.plan.create({
415
- data: {
416
- ...planData,
417
- stripePriceId: stripePriceId || "", // Should be set by automation above
418
- limit: {
419
- create: {
420
- maxStorageBytes: BigInt(limits.maxStorageBytes),
421
- maxWorksheets: limits.maxWorksheets,
422
- maxFlashcards: limits.maxFlashcards,
423
- maxPodcasts: limits.maxPodcasts,
424
- maxStudyGuides: limits.maxStudyGuides,
425
- }
426
- }
427
- }
428
- });
429
- logger.info(`New plan ${newPlan.id} created by admin ${ctx.userId}`, 'ADMIN');
430
- return newPlan;
431
- }
432
- }),
433
-
434
- deletePlan: adminProcedure
435
- .input(z.object({ id: z.string() }))
436
- .mutation(async ({ ctx, input }) => {
437
- // Check if plan has active subscriptions
438
- const activeSubs = await ctx.db.subscription.count({
439
- where: { planId: input.id, status: 'active' }
440
- });
441
-
442
- if (activeSubs > 0) {
443
- // If it has active subs, we should just deactivate it instead of deleting
444
- await ctx.db.plan.update({
445
- where: { id: input.id },
446
- data: { active: false }
447
- });
448
- return { success: true, message: "Plan deactivated because it has active subscribers." };
449
- }
450
-
451
- await ctx.db.plan.delete({ where: { id: input.id } });
452
- logger.info(`Plan ${input.id} deleted by admin ${ctx.userId}`, 'ADMIN');
453
- return { success: true, message: "Plan deleted." };
454
- }),
455
-
456
- getUserInvoices: adminProcedure
457
- .input(
458
- z.object({
459
- userId: z.string(),
460
- /** When set, return at most this many newest invoices (e.g. 5 for user detail sheet). Omit for full history. */
461
- limit: z.number().int().min(1).max(500).optional(),
462
- })
463
- )
464
- .query(async ({ ctx, input }) => {
465
- return (ctx.db as any).invoice.findMany({
466
- where: { userId: input.userId },
467
- orderBy: { createdAt: 'desc' },
468
- take: input.limit,
469
- include: {
470
- subscription: {
471
- include: { plan: true }
472
- }
473
- }
474
- });
475
- }),
476
-
477
- getUserDetailedInfo: adminProcedure
478
- .input(z.object({ userId: z.string() }))
479
- .query(async ({ ctx, input }) => {
480
- const user = await ctx.db.user.findUnique({
481
- where: { id: input.userId },
482
- include: {
483
- role: true,
484
- profilePicture: true,
485
- subscriptions: {
486
- orderBy: { createdAt: 'desc' },
487
- take: 1,
488
- include: { plan: true }
489
- },
490
- _count: {
491
- select: {
492
- workspaces: true,
493
- artifacts: true,
494
- }
495
- }
496
- }
497
- });
498
-
499
- if (!user) throw new Error("User not found");
500
-
501
- // Calculate total spent by this user
502
- const totalSpentResult = await (ctx.db as any).invoice.aggregate({
503
- where: { userId: input.userId, status: 'paid' },
504
- _sum: { amountPaid: true }
505
- });
506
- const totalSpent = Number(totalSpentResult._sum.amountPaid || 0) / 100;
507
-
508
- // Get purchased credits summary
509
- const purchasedCredits = await ctx.db.userCredit.groupBy({
510
- by: ['resourceType'],
511
- where: { userId: input.userId },
512
- _sum: { amount: true }
513
- });
514
-
515
- const profilePictureUrl = (user as any).profilePicture?.objectKey
516
- ? `/profile-picture/${(user as any).profilePicture.objectKey}?t=${new Date(user.updatedAt).getTime()}`
517
- : null;
518
-
519
- return {
520
- ...user,
521
- totalSpent,
522
- purchasedCredits: purchasedCredits.map(c => ({
523
- type: c.resourceType,
524
- amount: c._sum.amount || 0
525
- })),
526
- profilePicture: profilePictureUrl,
527
- };
528
- }),
529
-
530
- debugInvoices: adminProcedure
531
- .query(async ({ ctx }) => {
532
- const count = await (ctx.db as any).invoice.count();
533
- const all = await (ctx.db as any).invoice.findMany({ take: 10 });
534
- return { count, all };
535
- }),
536
-
537
- listResourcePrices: adminProcedure
538
- .query(async ({ ctx }) => {
539
- return ctx.db.resourcePrice.findMany({
540
- orderBy: { resourceType: 'asc' }
541
- });
542
- }),
543
-
544
- upsertResourcePrice: adminProcedure
545
- .input(z.object({
546
- resourceType: z.nativeEnum(ArtifactType),
547
- priceCents: z.number().min(0),
548
- }))
549
- .mutation(async ({ ctx, input }) => {
550
- const price = await ctx.db.resourcePrice.upsert({
551
- where: { resourceType: input.resourceType },
552
- update: { priceCents: input.priceCents },
553
- create: {
554
- resourceType: input.resourceType,
555
- priceCents: input.priceCents,
556
- },
557
- });
558
- logger.info(`Resource price for ${input.resourceType} updated to ${input.priceCents} by admin ${ctx.userId}`, 'ADMIN');
559
- return price;
560
- }),
561
-
562
- listRecentInvoices: adminProcedure
563
- .input(z.object({ limit: z.number().default(10) }))
564
- .query(async ({ ctx, input }) => {
565
- const invoices = await (ctx.db as any).invoice.findMany({
566
- take: input.limit,
567
- orderBy: { createdAt: 'desc' },
568
- include: {
569
- user: {
570
- include: {
571
- profilePicture: true,
572
- }
573
- }
574
- }
575
- });
576
-
577
- return invoices.map((invoice: any) => ({
578
- ...invoice,
579
- user: {
580
- ...invoice.user,
581
- profilePicture: invoice.user.profilePicture?.objectKey
582
- ? `/profile-picture/${invoice.user.profilePicture.objectKey}?t=${new Date(invoice.user.updatedAt).getTime()}`
583
- : null,
584
- }
585
- }));
586
- }),
587
-
588
- activityList: adminProcedure
589
- .input(activityListInput)
590
- .query(async ({ ctx, input }) => {
591
- const where = buildActivityLogWhere({
592
- from: input.from,
593
- to: input.to,
594
- actorUserId: input.actorUserId,
595
- workspaceId: input.workspaceId,
596
- category: input.category,
597
- status: input.status,
598
- search: input.search,
599
- });
600
-
601
- const totalCount = await ctx.db.activityLog.count({ where });
602
- const totalPages = Math.max(1, Math.ceil(totalCount / input.limit));
603
- const page = Math.min(Math.max(1, input.page), totalPages);
604
- const skip = (page - 1) * input.limit;
605
-
606
- const rows = await ctx.db.activityLog.findMany({
607
- where,
608
- skip,
609
- take: input.limit,
610
- orderBy: { createdAt: 'desc' },
611
- include: {
612
- actor: { select: { id: true, email: true, name: true } },
613
- workspace: { select: { id: true, title: true } },
614
- },
615
- });
616
-
617
- return {
618
- items: rows.map((r) => ({
619
- id: r.id,
620
- createdAt: r.createdAt,
621
- action: r.action,
622
- description: getActivityHumanDescription(r.trpcPath, r.action),
623
- category: r.category,
624
- trpcPath: r.trpcPath,
625
- status: r.status,
626
- durationMs: r.durationMs,
627
- errorCode: r.errorCode,
628
- ipAddress: r.ipAddress,
629
- actor: r.actor,
630
- workspace: r.workspace,
631
- metadata: r.metadata,
632
- })),
633
- page,
634
- pageSize: input.limit,
635
- totalCount,
636
- totalPages,
637
- };
638
- }),
639
-
640
- activityExportCsv: adminProcedure
641
- .input(
642
- activityLogFiltersInput.extend({
643
- maxRows: z.number().min(1).max(5000).default(2000),
644
- })
645
- )
646
- .query(async ({ ctx, input }) => {
647
- const where = buildActivityLogWhere({
648
- from: input.from,
649
- to: input.to,
650
- actorUserId: input.actorUserId,
651
- workspaceId: input.workspaceId,
652
- category: input.category,
653
- status: input.status,
654
- search: input.search,
655
- });
656
-
657
- const rows = await ctx.db.activityLog.findMany({
658
- where,
659
- orderBy: { createdAt: 'desc' },
660
- take: input.maxRows,
661
- include: {
662
- actor: { select: { email: true, name: true } },
663
- workspace: { select: { title: true } },
664
- },
665
- });
666
-
667
- const header = [
668
- 'createdAt',
669
- 'actorEmail',
670
- 'actorName',
671
- 'description',
672
- 'trpcPath',
673
- 'category',
674
- 'status',
675
- 'durationMs',
676
- 'workspaceTitle',
677
- 'errorCode',
678
- ].join(',');
679
-
680
- const lines = rows.map((r) =>
681
- [
682
- csvEscapeCell(r.createdAt.toISOString()),
683
- csvEscapeCell(r.actor?.email),
684
- csvEscapeCell(r.actor?.name),
685
- csvEscapeCell(getActivityHumanDescription(r.trpcPath, r.action)),
686
- csvEscapeCell(r.trpcPath),
687
- csvEscapeCell(r.category),
688
- csvEscapeCell(r.status),
689
- csvEscapeCell(r.durationMs),
690
- csvEscapeCell(r.workspace?.title),
691
- csvEscapeCell(r.errorCode),
692
- ].join(',')
693
- );
694
-
695
- return {
696
- csv: [header, ...lines].join('\n'),
697
- count: rows.length,
698
- };
699
- }),
700
-
701
- /** Deletes rows older than ACTIVITY_LOG_RETENTION_DAYS (default 365). Run manually or via cron. */
702
- activityPurgeRetention: adminProcedure.mutation(async ({ ctx }) => {
703
- const days = getActivityRetentionDays();
704
- const cutoff = new Date();
705
- cutoff.setDate(cutoff.getDate() - days);
706
- const { deleted } = await deleteActivityLogsOlderThan(ctx.db, cutoff);
707
- logger.info(`ActivityLog retention purge: deleted ${deleted} rows older than ${days} days (cutoff ${cutoff.toISOString()})`, 'ADMIN');
708
- return { deleted, retentionDays: days, cutoff };
709
- }),
42
+ getSystemStats: adminProcedure.query(({ ctx }) => new AdminService(ctx.db).getSystemStats()),
43
+
44
+ listUsers: adminProcedure
45
+ .input(listUsersInput)
46
+ .query(({ ctx, input }) => new AdminService(ctx.db).listUsers(input)),
47
+
48
+ listInvoices: adminProcedure
49
+ .input(listInvoicesInput)
50
+ .query(({ ctx, input }) => new AdminService(ctx.db).listInvoices(input)),
51
+
52
+ listWorkspaces: adminProcedure
53
+ .input(
54
+ z.object({
55
+ limit: z.number().min(1).max(100).default(50),
56
+ cursor: z.string().nullish(),
57
+ }),
58
+ )
59
+ .query(({ ctx, input }) => new AdminService(ctx.db).listWorkspaces(input)),
60
+
61
+ updateUserRole: adminProcedure
62
+ .input(z.object({ userId: z.string(), roleName: z.string() }))
63
+ .mutation(({ ctx, input }) => new AdminService(ctx.db).updateUserRole(ctx.userId!, input)),
64
+
65
+ listPlans: adminProcedure.query(({ ctx }) => new AdminService(ctx.db).listPlans()),
66
+
67
+ upsertPlan: adminProcedure
68
+ .input(
69
+ z.object({
70
+ id: z.string().optional(),
71
+ name: z.string(),
72
+ description: z.string().optional(),
73
+ type: z.string(),
74
+ price: z.number(),
75
+ stripePriceId: z.string().optional(),
76
+ interval: z.string().nullable(),
77
+ active: z.boolean().default(true),
78
+ limits: z.object({
79
+ maxStorageBytes: z.number(),
80
+ maxWorksheets: z.number(),
81
+ maxFlashcards: z.number(),
82
+ maxPodcasts: z.number().default(0),
83
+ maxStudyGuides: z.number().default(0),
84
+ }),
85
+ }),
86
+ )
87
+ .mutation(({ ctx, input }) => new AdminService(ctx.db).upsertPlan(ctx.userId!, input)),
88
+
89
+ deletePlan: adminProcedure
90
+ .input(z.object({ id: z.string() }))
91
+ .mutation(({ ctx, input }) => new AdminService(ctx.db).deletePlan(ctx.userId!, input.id)),
92
+
93
+ getUserInvoices: adminProcedure
94
+ .input(
95
+ z.object({
96
+ userId: z.string(),
97
+ limit: z.number().int().min(1).max(500).optional(),
98
+ }),
99
+ )
100
+ .query(({ ctx, input }) => new AdminService(ctx.db).getUserInvoices(input)),
101
+
102
+ getUserDetailedInfo: adminProcedure
103
+ .input(z.object({ userId: z.string() }))
104
+ .query(({ ctx, input }) => new AdminService(ctx.db).getUserDetailedInfo(input.userId)),
105
+
106
+ debugInvoices: adminProcedure.query(({ ctx }) => new AdminService(ctx.db).debugInvoices()),
107
+
108
+ listResourcePrices: adminProcedure.query(({ ctx }) =>
109
+ new AdminService(ctx.db).listResourcePrices(),
110
+ ),
111
+
112
+ upsertResourcePrice: adminProcedure
113
+ .input(
114
+ z.object({
115
+ resourceType: z.nativeEnum(ArtifactType),
116
+ priceCents: z.number().min(0),
117
+ }),
118
+ )
119
+ .mutation(({ ctx, input }) =>
120
+ new AdminService(ctx.db).upsertResourcePrice(ctx.userId!, input),
121
+ ),
122
+
123
+ listRecentInvoices: adminProcedure
124
+ .input(z.object({ limit: z.number().default(10) }))
125
+ .query(({ ctx, input }) => new AdminService(ctx.db).listRecentInvoices(input.limit)),
126
+
127
+ activityList: adminProcedure
128
+ .input(activityListInput)
129
+ .query(({ ctx, input }) => new AdminService(ctx.db).activityList(input)),
130
+
131
+ activityExportCsv: adminProcedure
132
+ .input(
133
+ activityLogFiltersInput.extend({
134
+ maxRows: z.number().min(1).max(5000).default(2000),
135
+ }),
136
+ )
137
+ .query(({ ctx, input }) => new AdminService(ctx.db).activityExportCsv(input)),
138
+
139
+ activityPurgeRetention: adminProcedure.mutation(({ ctx }) =>
140
+ new AdminService(ctx.db).activityPurgeRetention(),
141
+ ),
710
142
  });